質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
React Native

React Nativeは、ネイティブモバイルアプリ(iOS/Android)を作成できるJavaScriptフレームワークです。Reactと同じ設計のため、宣言的なコンポーネントでリッチなUIを開発することが可能です。

React.js

Reactは、アプリケーションのインターフェースを構築するためのオープンソースJavaScriptライブラリです。

Q&A

解決済

1回答

1542閲覧

stateで管理する配列[]の値入力と、表示の不具合を解決したい

tomaa

総合スコア84

React Native

React Nativeは、ネイティブモバイルアプリ(iOS/Android)を作成できるJavaScriptフレームワークです。Reactと同じ設計のため、宣言的なコンポーネントでリッチなUIを開発することが可能です。

React.js

Reactは、アプリケーションのインターフェースを構築するためのオープンソースJavaScriptライブラリです。

0グッド

1クリップ

投稿2020/08/11 04:06

前提・実現したいこと

React Nativeを利用してアプリの開発をしています。

下記の3つの要素が1組となるコンポーネントがあります。

各コンポーネントは配列で管理され、それぞれのTextInputで入力された値は、コンポーネントと同じindexで管理する配列で、追加、削除の管理がされます。

  • TextInput
  • Button(追加)
  • Button(削除)

追加ボタンを押すと、コンポーネントが追加され、削除を押すと削除されます。

イメージ説明

発生している問題・エラーメッセージ

一度TextInputに入力し、追加をすると、入力を終えた値の再編集ができずに困っています。

例えば、最初のコンポーネント(配列0番目)の入力欄に「000」と入力し、追加を押すと、入力されたとおりの値が表示されコンポーネントも追加されます。

イメージ説明

しかし、この状態で最初のコンポーネント(配列0番目)を編集しようとすると、入力に以下のような不具合が起こります。

入力欄の表示が000から変わらない(削除できずに、新しい数字の表示もできない)

数字を入力するonChangeの際、表示の元となる配列の値をconsol.log()(下記コードの①の箇所)で確認するために、

1→2→3
と入力するとコンソールでは、

Array [ "0001", ] Array [ "0002", ] Array [ "0003", ]

というように、元々の000に入力した数字の末尾に、1文字追加され、4文字で更新されるようになります。

このような問題を解決し、値の再編集をするにはどのような対策行えばよいか試していますが、解決できません。

方法等、ご存知の方おりましたら、ご教授いただけないでしょうか?

該当のソースコード

JS

1import React, { useState, useEffect } from "react"; 2import { View, Text, Button, TextInput, StyleSheet } from "react-native"; 3 4function Item({ 5 setNumber, 6 number, 7 handleInput, 8 handleAdd, 9 handleDelete, 10 index, 11}) { 12 return ( 13 <View style={styles.list}> 14 <Text>{index}</Text> 15 <TextInput 16 style={{ borderWidth: 1 }} 17 value={number[index]} //②試したこと 18 onChange={(e) => { 19 handleInput(index, e.nativeEvent.text); 20 console.log(number); //①値のコンソール確認 21 }} 22 ></TextInput> 23 <Button 24 title="追加" 25 onPress={() => { 26 handleAdd(); 27 }} 28 /> 29 <Button 30 title="削除" 31 onPress={() => { 32 handleDelete(index); 33 }} 34 /> 35 </View> 36 ); 37} 38 39export default function TestStateArray() { 40 const [count, setCount] = useState(1); 41 const [number, setNumber] = useState([]); 42 43 function handleAdd() { 44 setCount((v) => v + 1); 45 } 46 47 function handleDelete(index) { 48 setCount((v) => v - 1); 49 setNumber((v) => { 50 v.splice(index, 1); 51 return v; 52 }); 53 } 54 55 function handleInput(index, text) { 56 setNumber((v) => { 57 v[index] = text; 58 return v; 59 }); 60 } 61 62 return ( 63 <View> 64 {Array.from({ length: count }, (_, i) => ( 65 <Item 66 setNumber={setNumber} 67 number={number} 68 handleInput={handleInput} 69 handleAdd={handleAdd} 70 handleDelete={handleDelete} 71 key={i + "-" + number} 72 index={i} 73 /> 74 ))} 75 </View> 76 ); 77} 78 79const styles = StyleSheet.create({ 80 list: { 81 margin: 10, 82 padding: 10, 83 backgroundColor: "#ddd", 84 }, 85}); 86

試したこと

上記コード②のコメント部分である、value={number[index]}を削除すると、

追加ボタンを押し、コンポーネントが増えた時点で、入力欄の値が消えてしまします(再編集できない)ので、対策にはなりませんでした。

補足情報(FW/ツールのバージョンなど)

node : 12.18.3
react native : 4.10.1
expo : 3.22.3

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

javascript

1 function handleInput(index, text) { 2 setNumber((v) => { 3 v[index] = text; 4 return v; 5 }); 6 }

配列を破壊的に変更すると、「変更された」と認識されない場合があるので、React関係では基本的に避けるべきパターンです。いったんコピーしてから変更しましょう。

javascript

1 function handleInput(index, text) { 2 setNumber((v) => { 3 const ret = v.slice(); 4 ret[index] = text; 5 return ret; 6 }); 7 }

投稿2020/08/11 04:41

maisumakun

総合スコア146018

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

maisumakun

2020/08/11 04:42

同様に、破壊的にspliceしているhandleDeleteも対応が必要です。
tomaa

2020/08/11 05:10

ご回答いただきありがとうございます。 紹介していただいた方法を試しました。 再編集はできるよになりましたが、入力する時に不具合が出るようになりました。 1文字づつしか入力できず、入力毎にカーソルを当て直す必要があります。 こちらの対策について、調べてみたいと思います。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問