diff --git a/packages/ra-ui-materialui/src/input/ArrayInput/SimpleFormIterator.tsx b/packages/ra-ui-materialui/src/input/ArrayInput/SimpleFormIterator.tsx index 76269f9c113..e06184a1464 100644 --- a/packages/ra-ui-materialui/src/input/ArrayInput/SimpleFormIterator.tsx +++ b/packages/ra-ui-materialui/src/input/ArrayInput/SimpleFormIterator.tsx @@ -44,14 +44,34 @@ export const SimpleFormIterator = (props: SimpleFormIteratorProps) => { disableRemove, disableReordering, TransitionProps, + defaultValue, getItemLabel = DefaultLabelFn, } = props; const { append, fields, move, remove } = useArrayInput(props); const nodeRef = useRef(); // const { error, submitFailed } = meta; + // We need a unique id for each field for a proper enter/exit animation + // so we keep an internal map between the field position and an auto-increment id + const nextId = useRef( + fields && fields.length + ? fields.length + : defaultValue + ? defaultValue.length + : 0 + ); + + // We check whether we have a defaultValue (which must be an array) before checking + // the fields prop which will always be empty for a new record. + // Without it, our ids wouldn't match the default value and we would get key warnings + // on the CssTransition element inside our render method + const ids = useRef( + nextId.current > 0 ? Array.from(Array(nextId.current).keys()) : [] + ); + const removeField = useCallback( (index: number) => { + ids.current.splice(index, 1); remove(index); }, [remove] @@ -59,6 +79,7 @@ export const SimpleFormIterator = (props: SimpleFormIteratorProps) => { const addField = useCallback( (item: any = undefined) => { + ids.current.push(nextId.current++); append(item); }, [append] @@ -76,6 +97,9 @@ export const SimpleFormIterator = (props: SimpleFormIteratorProps) => { const handleReorder = useCallback( (origin: number, destination: number) => { + const item = ids.current[origin]; + ids.current[origin] = ids.current[destination]; + ids.current[destination] = item; move(origin, destination); }, [move] @@ -99,7 +123,7 @@ export const SimpleFormIterator = (props: SimpleFormIteratorProps) => { {fields.map((member, index) => (