Elección de la estructura del estado
Estructurar bien el estado puede marcar la diferencia entre un componente que es agradable de modificar y depurar, y uno que es una fuente constante de errores. Estos son algunos consejos que debe tener en cuenta al estructurar el estado.
Aprenderás
- Cuando usar una versus multiples variables de estado.
- Qué evitar al organizar el estado.
- Cómo solucionar problemas comunes con la estructura del estado.
Principios para la estructuración del estado
When you write a component that holds some state, you’ll have to make choices about how many state variables to use and what the shape of their data should be. While it’s possible to write correct programs even with a suboptimal state structure, there are a few principles that can guide you to make better choices:
- Group related state. If you always update two or more state variables at the same time, consider merging them into a single state variable.
- Avoid contradictions in state. When the state is structured in a way that several pieces of state may contradict and “disagree” with each other, you leave room for mistakes. Try to avoid this.
- Avoid redundant state. If you can calculate some information from the component’s props or its existing state variables during rendering, you should not put that information into that component’s state.
- Avoid duplication in state. When the same data is duplicated between multiple state variables, or within nested objects, it is difficult to keep them in sync. Reduce duplication when you can.
- Avoid deeply nested state. Deeply hierarchical state is not very convenient to update. When possible, prefer to structure state in a flat way.
The goal behind these principles is to make state easy to update without introducing mistakes. Removing redundant and duplicate data from state helps ensure that all its pieces stay in sync. This is similar to how a database engineer might want to “normalize” the database structure to reduce the chance of bugs. To paraphrase Albert Einstein, “Make your state as simple as it can be—but no simpler.”
Now let’s see how these principles apply in action.
Estado relativo al grupo
En ocasiones, es posible que no esté seguro entre usar una o varias variables de estado.
¿Deberías hacer esto?
const [x, setX] = useState(0);
const [y, setY] = useState(0);
¿O esto?
const [position, setPosition] = useState({ x: 0, y: 0 });
Técnicamente, puedes usar cualquiera de estos enfoques. Pero si algunas de las dos variables de estado siempre cambian juntas, podría ser una buena idea unificarlas en una sola variable de estado. Entonces no olvidará mantenerlos siempre sincronizados, como en este ejemplo donde al mover el cursor se actualizan ambas coordenadas del punto rojo:
import { useState } from 'react'; export default function MovingDot() { const [position, setPosition] = useState({ x: 0, y: 0 }); return ( <div onPointerMove={e => { setPosition({ x: e.clientX, y: e.clientY }); }} style={{ position: 'relative', width: '100vw', height: '100vh', }}> <div style={{ position: 'absolute', backgroundColor: 'red', borderRadius: '50%', transform: `translate(${position.x}px, ${position.y}px)`, left: -10, top: -10, width: 20, height: 20, }} /> </div> ) }
Otro caso en el que agrupará datos en un objeto o una matriz es cuando no sabe cuántas partes diferentes del estado se necesitarán. Por ejemplo, es útil cuando tienes un formulario en el que el usuario puede agregar campos personalizados.
Evitar contradicciones en el estado
Aquí hay un formulario de comentarios de un hotel con variables de estado isSending
y isSent
:
import { useState } from 'react'; export default function FeedbackForm() { const [text, setText] = useState(''); const [isSending, setIsSending] = useState(false); const [isSent, setIsSent] = useState(false); async function handleSubmit(e) { e.preventDefault(); setIsSending(true); await sendMessage(text); setIsSending(false); setIsSent(true); } if (isSent) { return <h1>¡Gracias por tu retroalimentación!</h1> } return ( <form onSubmit={handleSubmit}> <p>¿Cómo fue tu estadía en The Prancing Pony?</p> <textarea disabled={isSending} value={text} onChange={e => setText(e.target.value)} /> <br /> <button disabled={isSending} type="submit" > Enviar </button> {isSending && <p>Enviando...</p>} </form> ); } // Pretender enviar un mensaje. function sendMessage(text) { return new Promise(resolve => { setTimeout(resolve, 2000); }); }
Si bien este código funciona, deja la puerta abierta para estados “imposibles”. Por ejemplo, si olvida llamar a setIsSent
y setIsSending
juntos, puede terminar en una situación en la que tanto isSending
como isSent
son true
al mismo tiempo. Cuanto más complejo sea su componente, más difícil será entender lo que sucedió.
Dado que isSending
y isSent
nunca deben ser true
al mismo tiempo, es mejor reemplazarlos con una variable de estado status
que puede tomar uno de tres estados válidos: 'typing '
(initial), 'sending'
y 'sent'
:
import { useState } from 'react'; export default function FeedbackForm() { const [text, setText] = useState(''); const [status, setStatus] = useState('typing'); async function handleSubmit(e) { e.preventDefault(); setStatus('sending'); await sendMessage(text); setStatus('sent'); } const isSending = status === 'sending'; const isSent = status === 'sent'; if (isSent) { return <h1>¡Gracias por tu retroalimentación!</h1> } return ( <form onSubmit={handleSubmit}> <p>¿Cómo fue tu estadía en The Prancing Pony?</p> <textarea disabled={isSending} value={text} onChange={e => setText(e.target.value)} /> <br /> <button disabled={isSending} type="submit" > Enviar </button> {isSending && <p>Enviando...</p>} </form> ); } // Pretender enviar un mensaje. function sendMessage(text) { return new Promise(resolve => { setTimeout(resolve, 2000); }); }
Todavía puedes declarar algunas constantes para mejorar la legibilidad:
const isSending = status === 'sending';
const isSent = status === 'sent';
Pero no son variables de estado, por lo que no debe preocuparse de que no estén sincronizadas entre sí.
Evitar estado redundante
Si puede calcular alguna información de las props del componente o sus variables de estado existentes durante el renderizado, no debe poner esa información en el estado de ese componente.
Por ejemplo, toma este formulario. Funciona, pero ¿puedes encontrar algún estado redundante en él?
import { useState } from 'react'; export default function Form() { const [firstName, setFirstName] = useState(''); const [lastName, setLastName] = useState(''); const [fullName, setFullName] = useState(''); function handleFirstNameChange(e) { setFirstName(e.target.value); setFullName(e.target.value + ' ' + lastName); } function handleLastNameChange(e) { setLastName(e.target.value); setFullName(firstName + ' ' + e.target.value); } return ( <> <h2>Vamos a registrarte</h2> <label> Nombre:{' '} <input value={firstName} onChange={handleFirstNameChange} /> </label> <label> Apellido:{' '} <input value={lastName} onChange={handleLastNameChange} /> </label> <p> Su boleto será emitido a:<b>{fullName}</b> </p> </> ); }
Este formulario tiene tres variables de estado: firstName
, lastName
y fullName
. Sin embargo, fullName
es redundante. Siempre puedes calcular fullName
a partir de firstName
y lastName
durante el renderizado, así que quítalo del estado.
Así es como puedes hacerlo:
import { useState } from 'react'; export default function Form() { const [firstName, setFirstName] = useState(''); const [lastName, setLastName] = useState(''); const fullName = firstName + ' ' + lastName; function handleFirstNameChange(e) { setFirstName(e.target.value); } function handleLastNameChange(e) { setLastName(e.target.value); } return ( <> <h2>Vamos a registrarte</h2> <label> Nombre:{' '} <input value={firstName} onChange={handleFirstNameChange} /> </label> <label> Apellido:{' '} <input value={lastName} onChange={handleLastNameChange} /> </label> <p> Su boleto será emitido a: <b>{fullName}</b> </p> </> ); }
aquí, fullName
no es una variable de estado. En cambio, se calcula durante el renderizado:
const fullName = firstName + ' ' + lastName;
Como resultado, los controladores de cambios no necesitan hacer nada especial para actualizarlo. Cuando llamas a setFirstName
o setLastName
, activas una nueva representación y luego el siguiente fullName
se calculará a partir de los nuevos datos.
Deep Dive
Un ejemplo común de estado redundante es un código como este:
function Message({ messageColor }) {
const [color, setColor] = useState(messageColor);
Aquí, una variable de estado color
se inicializa en la prop messageColor
. El problema es que si el componente principal pasa un valor diferente de messageColor
más adelante (por ejemplo, 'red'
en lugar de 'blue'
), ¡la variable de estado color
¡no se actualizará! el estado solo se inicializa durante el primer renderizado.
Esta es la razón por la que “reflejar” alguna prop en una variable de estado puede generar confusión. En su lugar, usa el accesorio messageColor
directamente en tu código. Si deseas darle un nombre más corto, use una constante:
function Message({ messageColor }) {
const color = messageColor;
“Mirroring” props into state only makes sense when you want to ignore all updates for a specific prop. By convention, start the prop name with initial
or default
to clarify that its new values are ignored:
De esta forma, no se sincroniza con la propiedad que se pasa desde el componente principal.
“Reflejar” props en estado solo tiene sentido cuando quieres ignorar todas las actualizaciones de un prop en específico. Por convención, comience el nombre de la prop con initial
or default
para aclarar que sus nuevos valores se ignoran:
function Message({ initialColor }) {
// La variable de estado `color` contiene el *primer* valor de `initialColor`.
// Se ignoran los cambios posteriores a la prop `initialColor`.
const [color, setColor] = useState(initialColor);
Evite la duplicación en el estado
Este componente de lista de menú le permite elegir un solo refrigerio de viaje entre varios:
import { useState } from 'react'; const initialItems = [ { title: 'pretzels', id: 0 }, { title: 'crispy seaweed', id: 1 }, { title: 'granola bar', id: 2 }, ]; export default function Menu() { const [items, setItems] = useState(initialItems); const [selectedItem, setSelectedItem] = useState( items[0] ); return ( <> <h2>¿Cuál es tu merienda de viaje?</h2> <ul> {items.map(item => ( <li key={item.id}> {item.title} {' '} <button onClick={() => { setSelectedItem(item); }}>Seleccionar</button> </li> ))} </ul> <p>Seleccionaste {selectedItem.title}.</p> </> ); }
Actualmente, almacena el elemento seleccionado como un objeto en la variable de estado selectedItem
. Sin embargo, esto no esta bien: el contenido de selectedItem
es el mismo objeto que uno de los elementos dentro de la lista de items
. Esto significa que la información sobre el elemento en sí está duplicada en dos lugares.
¿Por qué es esto un problema? Hagamos que cada elemento sea editable:
import { useState } from 'react'; const initialItems = [ { title: 'pretzels', id: 0 }, { title: 'crispy seaweed', id: 1 }, { title: 'granola bar', id: 2 }, ]; export default function Menu() { const [items, setItems] = useState(initialItems); const [selectedItem, setSelectedItem] = useState( items[0] ); function handleItemChange(id, e) { setItems(items.map(item => { if (item.id === id) { return { ...item, title: e.target.value, }; } else { return item; } })); } return ( <> <h2>¿Cuál es tu merienda de viaje?</h2> <ul> {items.map((item, index) => ( <li key={item.id}> <input value={item.title} onChange={e => { handleItemChange(item.id, e) }} /> {' '} <button onClick={() => { setSelectedItem(item); }}>Seleccionar</button> </li> ))} </ul> <p>Seleccionaste {selectedItem.title}.</p> </> ); }
Observe cómo si primero hace clic en “Seleccionar” en un elemento y luego lo edita, la entrada se actualiza, pero la etiqueta en la parte inferior no refleja las ediciones. Esto se debe a que tiene un estado duplicado y se olvidó de actualizar selectedItem
.
Aunque también podría actualizar selectedItem
, una solución más fácil es eliminar la duplicación. En este ejemplo, en lugar de un objeto selectedItem
(que crea una duplicación con objetos dentro de items
), usted mantiene selectedId
en el estado, y luego obtiene el selectedItem
buscando en la matriz items
un artículo con esa identificación:
import { useState } from 'react'; const initialItems = [ { title: 'pretzels', id: 0 }, { title: 'crispy seaweed', id: 1 }, { title: 'granola bar', id: 2 }, ]; export default function Menu() { const [items, setItems] = useState(initialItems); const [selectedId, setSelectedId] = useState(0); const selectedItem = items.find(item => item.id === selectedId ); function handleItemChange(id, e) { setItems(items.map(item => { if (item.id === id) { return { ...item, title: e.target.value, }; } else { return item; } })); } return ( <> <h2>¿Cuál es tu merienda de viaje?</h2> <ul> {items.map((item, index) => ( <li key={item.id}> <input value={item.title} onChange={e => { handleItemChange(item.id, e) }} /> {' '} <button onClick={() => { setSelectedId(item.id); }}>Seleccionar</button> </li> ))} </ul> <p>Seleccionaste {selectedItem.title}.</p> </> ); }
(Alternativamente, puede mantener el índice seleccionado en el estado).
El estado solía duplicarse de esta manera:
items = [{ id: 0, title: 'pretzels'}, ...]
selectedItem = {id: 0, title: 'pretzels'}
Pero después del cambio es asi:
items = [{ id: 0, title: 'pretzels'}, ...]
selectedId = 0
¡La duplicación se ha ido, y solo conservas el estado esencial!
Ahora, si edita el item seleccionado, el siguiente mensaje se actualizará inmediatamente. Esto se debe a que setItems
desencadena una nueva representación, y items.find(...)
encontraría el item con el título actualizado. No era necesario mantener el item seleccionado en el estado, porque solo el ID seleccionado es esencial. El resto podría calcularse durante el renderizado.
Evitar el estado profundamente anidado
Imagina un plan de viaje compuesto por planetas, continentes y países. Es posible que sienta la tentación de estructurar su estado mediante objetos y matrices anidados, como en este ejemplo:
export const initialTravelPlan = { id: 0, title: '(Root)', childPlaces: [{ id: 1, title: 'Earth', childPlaces: [{ id: 2, title: 'Africa', childPlaces: [{ id: 3, title: 'Botswana', childPlaces: [] }, { id: 4, title: 'Egypt', childPlaces: [] }, { id: 5, title: 'Kenya', childPlaces: [] }, { id: 6, title: 'Madagascar', childPlaces: [] }, { id: 7, title: 'Morocco', childPlaces: [] }, { id: 8, title: 'Nigeria', childPlaces: [] }, { id: 9, title: 'South Africa', childPlaces: [] }] }, { id: 10, title: 'Americas', childPlaces: [{ id: 11, title: 'Argentina', childPlaces: [] }, { id: 12, title: 'Brazil', childPlaces: [] }, { id: 13, title: 'Barbados', childPlaces: [] }, { id: 14, title: 'Canada', childPlaces: [] }, { id: 15, title: 'Jamaica', childPlaces: [] }, { id: 16, title: 'Mexico', childPlaces: [] }, { id: 17, title: 'Trinidad and Tobago', childPlaces: [] }, { id: 18, title: 'Venezuela', childPlaces: [] }] }, { id: 19, title: 'Asia', childPlaces: [{ id: 20, title: 'China', childPlaces: [] }, { id: 21, title: 'Hong Kong', childPlaces: [] }, { id: 22, title: 'India', childPlaces: [] }, { id: 23, title: 'Singapore', childPlaces: [] }, { id: 24, title: 'South Korea', childPlaces: [] }, { id: 25, title: 'Thailand', childPlaces: [] }, { id: 26, title: 'Vietnam', childPlaces: [] }] }, { id: 27, title: 'Europe', childPlaces: [{ id: 28, title: 'Croatia', childPlaces: [], }, { id: 29, title: 'France', childPlaces: [], }, { id: 30, title: 'Germany', childPlaces: [], }, { id: 31, title: 'Italy', childPlaces: [], }, { id: 32, title: 'Portugal', childPlaces: [], }, { id: 33, title: 'Spain', childPlaces: [], }, { id: 34, title: 'Turkey', childPlaces: [], }] }, { id: 35, title: 'Oceania', childPlaces: [{ id: 36, title: 'Australia', childPlaces: [], }, { id: 37, title: 'Bora Bora (French Polynesia)', childPlaces: [], }, { id: 38, title: 'Easter Island (Chile)', childPlaces: [], }, { id: 39, title: 'Fiji', childPlaces: [], }, { id: 40, title: 'Hawaii (the USA)', childPlaces: [], }, { id: 41, title: 'New Zealand', childPlaces: [], }, { id: 42, title: 'Vanuatu', childPlaces: [], }] }] }, { id: 43, title: 'Moon', childPlaces: [{ id: 44, title: 'Rheita', childPlaces: [] }, { id: 45, title: 'Piccolomini', childPlaces: [] }, { id: 46, title: 'Tycho', childPlaces: [] }] }, { id: 47, title: 'Mars', childPlaces: [{ id: 48, title: 'Corn Town', childPlaces: [] }, { id: 49, title: 'Green Hill', childPlaces: [] }] }] };
Ahora, supongamos que deseas agregar un botón para eliminar un lugar que ya visitaste. ¿Cómo lo harías? Actualizar el estado anidado implica hacer copias de objetos desde la parte que cambió. La eliminación de un lugar profundamente anidado implica copiar toda la cadena de lugares principal. Dicho código puede ser muy detallado.
Si el estado está demasiado anidado para actualizarse fácilmente, considere hacerlo “plano”. Esta es una manera de reestructurar estos datos. En lugar de una estructura similar a un árbol donde cada lugar
tiene una matriz de sus lugares secundarios, puede hacer que cada lugar contenga una matriz de sus ID de lugares secundarios. Luego puede almacenar un mapeo de cada ID de lugar al lugar correspondiente.
Esta reestructuración de datos puede recordarle ver una tabla de base de datos:
export const initialTravelPlan = { 0: { id: 0, title: '(Root)', childIds: [1, 43, 47], }, 1: { id: 1, title: 'Earth', childIds: [2, 10, 19, 27, 35] }, 2: { id: 2, title: 'Africa', childIds: [3, 4, 5, 6 , 7, 8, 9] }, 3: { id: 3, title: 'Botswana', childIds: [] }, 4: { id: 4, title: 'Egypt', childIds: [] }, 5: { id: 5, title: 'Kenya', childIds: [] }, 6: { id: 6, title: 'Madagascar', childIds: [] }, 7: { id: 7, title: 'Morocco', childIds: [] }, 8: { id: 8, title: 'Nigeria', childIds: [] }, 9: { id: 9, title: 'South Africa', childIds: [] }, 10: { id: 10, title: 'Americas', childIds: [11, 12, 13, 14, 15, 16, 17, 18], }, 11: { id: 11, title: 'Argentina', childIds: [] }, 12: { id: 12, title: 'Brazil', childIds: [] }, 13: { id: 13, title: 'Barbados', childIds: [] }, 14: { id: 14, title: 'Canada', childIds: [] }, 15: { id: 15, title: 'Jamaica', childIds: [] }, 16: { id: 16, title: 'Mexico', childIds: [] }, 17: { id: 17, title: 'Trinidad and Tobago', childIds: [] }, 18: { id: 18, title: 'Venezuela', childIds: [] }, 19: { id: 19, title: 'Asia', childIds: [20, 21, 22, 23, 24, 25, 26], }, 20: { id: 20, title: 'China', childIds: [] }, 21: { id: 21, title: 'Hong Kong', childIds: [] }, 22: { id: 22, title: 'India', childIds: [] }, 23: { id: 23, title: 'Singapore', childIds: [] }, 24: { id: 24, title: 'South Korea', childIds: [] }, 25: { id: 25, title: 'Thailand', childIds: [] }, 26: { id: 26, title: 'Vietnam', childIds: [] }, 27: { id: 27, title: 'Europe', childIds: [28, 29, 30, 31, 32, 33, 34], }, 28: { id: 28, title: 'Croatia', childIds: [] }, 29: { id: 29, title: 'France', childIds: [] }, 30: { id: 30, title: 'Germany', childIds: [] }, 31: { id: 31, title: 'Italy', childIds: [] }, 32: { id: 32, title: 'Portugal', childIds: [] }, 33: { id: 33, title: 'Spain', childIds: [] }, 34: { id: 34, title: 'Turkey', childIds: [] }, 35: { id: 35, title: 'Oceania', childIds: [36, 37, 38, 39, 40, 41, 42], }, 36: { id: 36, title: 'Australia', childIds: [] }, 37: { id: 37, title: 'Bora Bora (French Polynesia)', childIds: [] }, 38: { id: 38, title: 'Easter Island (Chile)', childIds: [] }, 39: { id: 39, title: 'Fiji', childIds: [] }, 40: { id: 40, title: 'Hawaii (the USA)', childIds: [] }, 41: { id: 41, title: 'New Zealand', childIds: [] }, 42: { id: 42, title: 'Vanuatu', childIds: [] }, 43: { id: 43, title: 'Moon', childIds: [44, 45, 46] }, 44: { id: 44, title: 'Rheita', childIds: [] }, 45: { id: 45, title: 'Piccolomini', childIds: [] }, 46: { id: 46, title: 'Tycho', childIds: [] }, 47: { id: 47, title: 'Mars', childIds: [48, 49] }, 48: { id: 48, title: 'Corn Town', childIds: [] }, 49: { id: 49, title: 'Green Hill', childIds: [] } };
Ahora que el estado es “plano” (también conocido como “normalizado”), la actualización de elementos anidados se vuelve más fácil.
Para eliminar un lugar ahora, solo necesita actualizar dos niveles de estado:
- La versión actualizada de su lugar principal debería excluir el ID eliminado de su matriz
childIds
. - La versión actualizada del objeto raíz de “tabla” debe incluir la versión actualizada del lugar principal.
Este es un ejemplo de cómo podrías hacerlo:
import { useState } from 'react'; import { initialTravelPlan } from './places.js'; export default function TravelPlan() { const [plan, setPlan] = useState(initialTravelPlan); function handleComplete(parentId, childId) { const parent = plan[parentId]; // Crear una nueva versión del lugar principal // que no incluye ID del hijo. const nextParent = { ...parent, childIds: parent.childIds .filter(id => id !== childId) }; // Actualizar el objeto de estado raíz... setPlan({ ...plan, // ...para que tenga el padre este actualizado. [parentId]: nextParent }); } const root = plan[0]; const planetIds = root.childIds; return ( <> <h2>Logares a visitar</h2> <ol> {planetIds.map(id => ( <PlaceTree key={id} id={id} parentId={0} placesById={plan} onComplete={handleComplete} /> ))} </ol> </> ); } function PlaceTree({ id, parentId, placesById, onComplete }) { const place = placesById[id]; const childIds = place.childIds; return ( <li> {place.title} <button onClick={() => { onComplete(parentId, id); }}> Completado </button> {childIds.length > 0 && <ol> {childIds.map(childId => ( <PlaceTree key={childId} id={childId} parentId={id} placesById={placesById} onComplete={onComplete} /> ))} </ol> } </li> ); }
Puede anidar el estado tanto como desee, pero hacerlo “plano” puede resolver numerosos problemas. Facilita la actualización del estado y ayuda a garantizar que no haya duplicación en diferentes partes de un objeto anidado.
Deep Dive
Idealmente, también eliminaría los elementos eliminados (¡y sus hijos!) del objeto “tabla” para mejorar el uso de la memoria. Esta versión lo hace. También usa Immer para hacer que la lógica de actualización sea más concisa.
import { useImmer } from 'use-immer'; import { initialTravelPlan } from './places.js'; export default function TravelPlan() { const [plan, updatePlan] = useImmer(initialTravelPlan); function handleComplete(parentId, childId) { updatePlan(draft => { // Elimina los ID secundarios del lugar principal. const parent = draft[parentId]; parent.childIds = parent.childIds .filter(id => id !== childId); // Olvida este lugar y todo su subárbol. deleteAllChildren(childId); function deleteAllChildren(id) { const place = draft[id]; place.childIds.forEach(deleteAllChildren); delete draft[id]; } }); } const root = plan[0]; const planetIds = root.childIds; return ( <> <h2>Lugares a visitar</h2> <ol> {planetIds.map(id => ( <PlaceTree key={id} id={id} parentId={0} placesById={plan} onComplete={handleComplete} /> ))} </ol> </> ); } function PlaceTree({ id, parentId, placesById, onComplete }) { const place = placesById[id]; const childIds = place.childIds; return ( <li> {place.title} <button onClick={() => { onComplete(parentId, id); }}> Completado </button> {childIds.length > 0 && <ol> {childIds.map(childId => ( <PlaceTree key={childId} id={childId} parentId={id} placesById={placesById} onComplete={onComplete} /> ))} </ol> } </li> ); }
A veces, también puede reducir el anidamiento de estados moviendo algunos de los estados anidados a los componentes secundarios. Esto funciona bien para el estado efímero de la interfaz de usuario que no necesita almacenarse, por ejemplo, si se pasa el cursor por encima de un elemento.
Recapitulación
- Si dos variables de estado siempre se actualizan juntas, considere combinarlas en una.
- Elija cuidadosamente sus variables de estado para evitar crear estados “imposibles”.
- Estructure su estado de una manera que reduzca las posibilidades de que cometa un error al actualizarlo.
- Evite el estado redundante y duplicado para que no necesite mantenerlo sincronizado.
- No ponga props en estado a menos que desee evitar específicamente las actualizaciones.
- Para patrones de interfaz de usuario como la selección, mantenga el ID o el índice en estado en lugar del objeto mismo.
- Si actualizar el estado profundamente anidado es complicado, intente aplanarlo.
Desafío 1 de 4: Arreglar un componente que no se actualiza
Este componente Reloj
recibe dos accesorios: color
y tiempo
. Cuando selecciona un color diferente en el cuadro de selección, el componente Reloj
recibe una prop de color
diferente de su componente principal. Sin embargo, por alguna razón, el color mostrado no se actualiza. ¿Por qué? Arregla el problema.
import { useState } from 'react'; export default function Clock(props) { const [color, setColor] = useState(props.color); return ( <h1 style={{ color: color }}> {props.time} </h1> ); }