This question is more to learn more about how react handles and reacts to changes rather than the implementation, so I let immutable-props-apprach develop a bit.
I'm trying to get the first element of an array and remove it from the original array that is passed to the component as a prop:
const ChildComponent = ( arrayToChange: Array<any>[] ) => { const elementFromArray = arrayToChange.shift(); }
From the definition of shift() method::
shift() method removes the first element from the array and returns the removed element. This method changes the length of the array.
Although the elementFromArray variable now contains the elements in the array, the array is complete, it is not affected in any way and still contains all elements.
But how is this possible? React should pass props by reference, so the original array should be affected. I'd understand if React put some protections in place and those changes wouldn't be reflected in the parent, however, I'd still like the changes to be reflected in the child. I can't find anything useful to explain this behavior, most resources only mention immutable methods for props and how to find a workaround, but not the reason or logic behind it.
Although the elementFromArray variable now contains the elements in the array, the array is complete, it is not affected in any way and still contains all elements. However, if I use the Push() method, the changes are reflected and arrayToChange contains one more element.
My question is - why does arrayToChange react differently to these methods? If shift() doesn't change the content, I hope push() won't either.
The behavior in the code snippet can be explained if you think of the rendering process as a breadth-first algorithm.
JSX will convert:
Enter the following JavaScript:
React.createElement(ChildShowArray, { array: letter })
Creates a structure that does not immediately call theChildShowArray
component. It will create some kind of intermediate structure/object that will only run when the renderer asks it to.JavaScript placed inside
{...}
(JSX context) is passed directly as a parameter and therefore parsed directly. This means that all{JSON.stringify(letters)}
inParent
are run before any code in the child component is run.When building the parent structure is complete, the renderer will access each intermediate structure/object and ask it to render. This is done from top to bottom, which is why the first
ChildShowArray
render still shows the full array. Then renderChildChangeArray
and remove the first element. The secondChildShowArray
render reflects this change and is rendered without the first element.Note that
shift()
does change the contents ofletters
, but when it is called, the contents ofParent
are already rendered and no longer change. This change does affect theParent
the next time it is rendered (click the "Rerender" button in the snippet). It also affects the rendering of other child components below it that use the same array reference.I'm not entirely sure what the issue is, but I'm going to hazard a guess here, so please leave a comment here and I'll change it before voting.
I think you should try this in your child component:
Then use "data" to map to the output in jsx
Then in the parent component, shift arrayToChange. You can think of useEffect as an "observer" that will fire when the array length changes.