Vinculando inputs a um array em React
Suponha que o estado notes armazene um array:
function App() {
const [notes, setNotes] = useState([1, 2, 3]);
return <div>
</div>;
}
Suponha também que temos uma função auxiliar, que calcula a soma dos elementos do array:
function getSum(arr) {
let sum = 0;
for (const elem of arr) {
sum += +elem;
}
return sum;
}
function App() {
...
}
Vamos calcular e exibir a soma dos elementos do nosso array do estado, usando nossa função auxiliar:
function App() {
const [notes, setNotes] = useState([1, 2, 3]);
return <div>
{getSum(notes)}
</div>;
}
Agora, vamos criar três inputs e no value
de cada input, colocar um dos elementos do array:
function App() {
const [notes, setNotes] = useState([1, 2, 3]);
return <div>
<input value={notes[0]} />
<input value={notes[1]} />
<input value={notes[2]} />
{getSum(notes)}
</div>;
}
Agora, vamos adicionar o evento onChange
aos nossos inputs. Vamos criar uma única função
de manipulação comum para este evento:
function App() {
const [notes, setNotes] = useState([1, 2, 3]);
function changeHandler(index, event) {
// função de manipulação comum
}
return <div>
<input value={notes[0]} onChange={event => changeHandler(0, event)} />
<input value={notes[1]} onChange={event => changeHandler(1, event)} />
<input value={notes[2]} onChange={event => changeHandler(2, event)} />
{getSum(notes)}
</div>;
}
Como você pode ver, a função changeHandler
recebe como primeiro parâmetro o número do elemento
do array que este input está editando.
Por meio deste número, podemos substituir o elemento do array pelo conteúdo do input.
Vamos fazer isso:
function changeHandler(index, event) {
setNotes([...notes.slice(0, index), event.target.value, ...notes.slice(index + 1)]);
}
Agora será possível editar qualquer input, e o array será alterado de forma reativa e, consequentemente, a soma dos seus elementos será recalculada.
Vamos juntar todo o nosso código:
function App() {
const [notes, setNotes] = useState([1, 2, 3]);
function changeHandler(index, event) {
setNotes([...notes.slice(0, index), event.target.value, ...notes.slice(index + 1)]);
}
return <div>
<input value={notes[0]} onChange={event => changeHandler(0, event)} />
<input value={notes[1]} onChange={event => changeHandler(1, event)} />
<input value={notes[2]} onChange={event => changeHandler(2, event)} />
{getSum(notes)}
</div>;
}
É possível fazer com que os inputs sejam gerados em um loop:
function App() {
const [notes, setNotes] = useState([1, 2, 3]);
function changeHandler(index, event) {
setNotes([...notes.slice(0, index), event.target.value, ...notes.slice(index + 1)]);
}
const result = notes.map((note, index) => {
return <input
key={index}
value={note}
onChange={event => changeHandler(index, event)}
/>;
});
return <div>
{result}
{getSum(notes)}
</div>;
}
Dado o seguinte array:
[1, 2, 3, 4, 5, 6, 7, 8, 9]
Exiba na tela a média aritmética dos elementos deste array. Em um loop, crie inputs para editar os elementos.