I have two components, a Parent and a Child. The Parent components keeps a list of Children in state and renders it using the map method.
The parent:
import { useState } from "react";
import Child from "./Child";
export default function Parent(){
const [counter, setCounter] = useState(0)
const [childKey, setChildKey] = useState(0);
const [children, setChildren] = useState([]);
function addChild(){
setChildren([...children, <Child counter={counter} index={childKey}/>]);
setChildKey(childKey + 1);
}
return (
<>
<h2>The parent component</h2>
<p>The counter is set to {counter}</p>
<p><button onClick={() => {setCounter(counter + 1)}}>Increment</button></p>
<p><button onClick={addChild}>Add child</button></p>
{children.map((c, i) => {
return (
<div key={i}>{c}</div>
);
})}
</>
);
}
The Child:
export default function Child({counter, index}){
return (
<>
<p>Child {index}: the counter is set to {counter}</p>
</>
);
}
Clicking the "add child" button adds an child to the children array.
Clicking the "increment" button updates the counter on the parent, but the counter in the children does not change.
Is there a way to re-render the children when the counter increments? (while preferably keeping the children in an array)
setChildren([...children, <Child counter={counter} index={childKey}/>]);
Don't put elements into state. It makes it really easy to have bugs like the one you're having right now. You've effectively "locked in" what the props to the child are, so changes to the counter will not take effect.
Instead, just store the minimal data that's needed to create the elements, and then create those elements during rendering. In your case, i think you just need a number saying how many children to render. Conveniently, you already have that in childKey
, so i recommend:
export default function Parent(){
const [counter, setCounter] = useState(0)
const [childKey, setChildKey] = useState(0);
const children = [];
for (let i = 1; i <= childKey; i++) {
children.push((
<div key={i}>
<Child counter={counter} index={i} />
</div>
));
}
function addChild(){
setChildKey(childKey + 1);
}
return (
<>
<h2>The parent component</h2>
<p>The counter is set to {counter}</p>
<p><button onClick={() => {setCounter(counter + 1)}}>Increment</button></p>
<p><button onClick={addChild}>Add child</button></p>
{children}
</>
);
}
Maybe childKey
could be renamed to childCount
.
When you do:
setChildren([...children, <Child counter={counter} index={childKey}/>]);
You tell React to add <Child counter={counter} index={childKey}/>
element to the children
state. It creates an element with the props counter
and childKey
with the values that they have at the time you add them to the state. Re-rendering these elements with a different value for childKey
will have no impact.
Depending on what you want to do, there are many solutions. If you want to render Child
with the value of counter
at the time of re-render I would do something like this:
export default function Parent(){
const [counter, setCounter] = useState(0)
const [childKey, setChildKey] = useState(0);
const [children, setChildren] = useState([]);
function addChild(){
setChildren([...children, ({ counter }) => <Child counter={counter} index={childKey}/>]);
setChildKey(childKey + 1);
}
return (
<>
<h2>The parent component</h2>
<p>The counter is set to {counter}</p>
<p><button onClick={() => {setCounter(counter + 1)}}>Increment</button></p>
<p><button onClick={addChild}>Add child</button></p>
{children.map((childFn, i) => {
return (
<div key={i}>{childFn({ counter })}</div>
);
})}
</>
);
}
Or may be simpler (and less error prone):
export default function Parent(){
const [counter, setCounter] = useState(0)
const [childKey, setChildKey] = useState(0);
const [children, setChildren] = useState([]);
function addChild(){
setChildren([...children, childKey]);
setChildKey(childKey + 1);
}
return (
<>
<h2>The parent component</h2>
<p>The counter is set to {counter}</p>
<p><button onClick={() => {setCounter(counter + 1)}}>Increment</button></p>
<p><button onClick={addChild}>Add child</button></p>
{children.map((childKey, i) => {
return (
<div key={i}><Child counter={counter} index={childKey} /></div>
);
})}
</>
);
}
Yes like other answers keeping components in state is not a good idea - well explained by Nicholas above
Same answer but different style of rendering
Just updated the Parent.js
as below
import { useState } from "react";
import Child from "./Child";
export default function Parent() {
const [counter, setCounter] = useState(0);
const [childKey, setChildKey] = useState(0);
function addChild() {
setChildKey(childKey + 1);
}
return (
<>
<h2>The parent component</h2>
<p>The counter is set to {counter}</p>
<p>
<button
onClick={() => {
setCounter(counter + 1);
}}
>
Increment
</button>
</p>
<p>
<button onClick={addChild}>Add child</button>
</p>
{/* create an array of length of childKey number */}
{Array.from({ length: childKey }, (_, i) => {
return <Child key={i} counter={counter} index={i} />;
})}
</>
);
}
The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.