useContext es un Hook de React que te permite leer y suscribirte a un contexto desde tu componente.

const value = useContext(SomeContext)

Referencia

useContext(SomeContext)

Llama useContext en el nivel superior de tu componente para leer y suscribirte al contexto.

import { useContext } from 'react';

function MyComponent() {
const theme = useContext(ThemeContext);
// ...

Ver más ejemplos abajo.

Parámetros

  • SomeContext: El contexto que creaste previamente con createContext. El propio contexto no guarda información, solo representa el tipo de información que puedes proporcionar o leer desde tus componentes.

Devuelve

useContext devuelve el valor del contexto para el componente que lo llama. Está determinado como el value pasado al SomeContext.Provider más cercano arriba del componente que llama en el árbol. Si no existe tal proveedor, entonces el valor devuelto será el defaultValue que le pasaste a createContext para ese contexto. El valor devuelto siempre está actualizado. React rerenderiza automáticamente los componentes que leen algún contexto si este cambia.

Advertencias

  • La llamada de useContext() en un componente no es afectada por los proveedores devueltos desde el mismo componente. El <Context.Provider> correspondiente necesita estar arriba del componente que hace la llamada de useContext().
  • React rerenderiza automáticamente todos los hijos que usen un contexto particular empezando desde el proveedor que recibe un value diferente. Los valores anteriores y los siguientes son comparados con Object.is. Saltarse el rerenderizado con memo no evita que los hijos reciban valores de contexto frescos de arriba.
  • Si tu sistema de compilación produce módulos duplicados en la salida (lo cual puede pasar si usas enlaces simbólicos), esto puede romper el contexto. Pasar algo a través del contexto solo funciona si SomeContext que usas para proporcionar el contexto y SomeContext que usas para leerlo son exactamente el mismo objeto, como está determinado por la comparación ===.

Uso

Pasar datos de manera profunda en el árbol

Llama useContext en el nivel superior de tu componente para leer y suscribirte al contexto.

import { useContext } from 'react';

function Button() {
const theme = useContext(ThemeContext);
// ...

useContext devuelve el valor del contexto para el contexto que le pasaste. Para determinar el valor del contexto, React busca en el árbol de componentes y encuentra el proveedor de contexto más cercano arriba para ese contexto en particular.

Para pasar el contexto a un Button, envuélvelo o envuelve a uno de sus componentes padres dentro del proveedor de contexto correspondiente:

function MyPage() {
return (
<ThemeContext.Provider value="dark">
<Form />
</ThemeContext.Provider>
);
}

function Form() {
// ... renderiza botones dentro ...
}

No importa cuántas capas de componentes hay entre el proveedor y el Button. Cuando un Button en cualquier lugar dentro de Form llama useContext(ThemeContext), recibirá "dark" como valor.

Atención

useContext() siempre busca al proveedor más cercano arriba del componente que lo llama. Busca hacia arriba y no toma en cuenta a los proveedores en el componente desde el cual estás llamando useContext().

import { createContext, useContext } from 'react';

const ThemeContext = createContext(null);

export default function MyApp() {
  return (
    <ThemeContext.Provider value="dark">
      <Form />
    </ThemeContext.Provider>
  )
}

function Form() {
  return (
    <Panel title="Welcome">
      <Button>Registrarse</Button>
      <Button>Iniciar sesión</Button>
    </Panel>
  );
}

function Panel({ title, children }) {
  const theme = useContext(ThemeContext);
  const className = 'panel-' + theme;
  return (
    <section className={className}>
      <h1>{title}</h1>
      {children}
    </section>
  )
}

function Button({ children }) {
  const theme = useContext(ThemeContext);
  const className = 'button-' + theme;
  return (
    <button className={className}>
      {children}
    </button>
  );
}


Actualizar los datos pasados a través del contexto

A menudo, querrás que el contexto cambie a través del tiempo. Para actualizar el contexto, necesitas combinarlo con el estado. Declara una variable de estado en el componente padre, y pasa el estado actual como el valor de contexto al proveedor.

function MyPage() {
const [theme, setTheme] = useState('dark');
return (
<ThemeContext.Provider value={theme}>
<Form />
<Button onClick={() => {
setTheme('light');
}}>
Cambiar a tema claro
</Button>
</ThemeContext.Provider>
);
}

Ahora cualquier Button dentro del proveedor recibirá el valor actual de theme. Si llamas setTheme para actualizar el valor de theme que pasaste al proveedor, todos los componentes Button se rerenderizarán con el nuevo valor 'light'.

Ejemplos de actualizar el contexto

Ejemplo 1 de 5:
Actualizar un valor a través del contexto

En este ejemplo, el componente MyApp guarda una variable de estado la cual es luego pasada al proveedor de ThemeContext. Marcar la casilla «Dark mode» actualiza el estado. Cambiar el valor proporcionado rerenderiza todos los componentes que utilizan ese contexto.

import { createContext, useContext, useState } from 'react';

const ThemeContext = createContext(null);

export default function MyApp() {
  const [theme, setTheme] = useState('light');
  return (
    <ThemeContext.Provider value={theme}>
      <Form />
      <label>
        <input
          type="checkbox"
          checked={theme === 'dark'}
          onChange={(e) => {
            setTheme(e.target.checked ? 'dark' : 'light')
          }}
        />
        Usar modo oscuro
      </label>
    </ThemeContext.Provider>
  )
}

function Form({ children }) {
  return (
    <Panel title="Bienvenido">
      <Button>Registrarse</Button>
      <Button>Iniciar sesión</Button>
    </Panel>
  );
}

function Panel({ title, children }) {
  const theme = useContext(ThemeContext);
  const className = 'panel-' + theme;
  return (
    <section className={className}>
      <h1>{title}</h1>
      {children}
    </section>
  )
}

function Button({ children }) {
  const theme = useContext(ThemeContext);
  const className = 'button-' + theme;
  return (
    <button className={className}>
      {children}
    </button>
  );
}

Fíjate que value="dark" pasa el string "dark", pero value={theme} pasa el valor de la variable JavaScript theme con llaves de JSX. Las llaves también te permiten pasar valores de contexto que no son strings.


Especificar un valor por defecto

Si React no puede encontrar ningún proveedor de ese contexto en particular en el árbol padre, el valor del contexto devuelto por useContext() será igual al valor por defecto que especificaste cuando creaste ese contexto:

const ThemeContext = createContext(null);

El valor por defecto nunca cambia. Si quieres actualizar el contexto, úsalo en conjunto con el estado como está descrito arriba.

A menudo, en lugar de null, hay algunos valores significativos más que puedes usar por defecto, por ejemplo:

const ThemeContext = createContext('light');

De esta manera, si accidentalmente renderizas algún componente sin su proveedor correspondiente, no se romperá. Esto también ayuda a que tus componentes funcionen bien en un ambiente de pruebas sin configurar un montón de proveedores en las pruebas.

En este ejemplo a continuación, el botón «Cambiar tema» siempre es claro, porque está afuera de cualquier proveedor de contexto del tema y el valor por defecto del contexto del tema es 'light'. Intenta editar el tema por defecto para que sea 'dark'.

import { createContext, useContext, useState } from 'react';

const ThemeContext = createContext('light');

export default function MyApp() {
  const [theme, setTheme] = useState('light');
  return (
    <>
      <ThemeContext.Provider value={theme}>
        <Form />
      </ThemeContext.Provider>
      <Button onClick={() => {
        setTheme(theme === 'dark' ? 'light' : 'dark');
      }}>
        Cambiar tema
      </Button>
    </>
  )
}

function Form({ children }) {
  return (
    <Panel title="Bienvenido">
      <Button>Registrarse</Button>
      <Button>Iniciar sesión</Button>
    </Panel>
  );
}

function Panel({ title, children }) {
  const theme = useContext(ThemeContext);
  const className = 'panel-' + theme;
  return (
    <section className={className}>
      <h1>{title}</h1>
      {children}
    </section>
  )
}

function Button({ children, onClick }) {
  const theme = useContext(ThemeContext);
  const className = 'button-' + theme;
  return (
    <button className={className} onClick={onClick}>
      {children}
    </button>
  );
}


Sobreescribir el contexto para una parte del árbol

Puedes sobreescribir el contexto para una parte del árbol al envolver esa parte en un proveedor con un valor diferente.

<ThemeContext.Provider value="dark">
...
<ThemeContext.Provider value="light">
<Footer />
</ThemeContext.Provider>
...
</ThemeContext.Provider>

Puedes anidar y sobreescribir proveedores tantas veces como necesites.

Prueba algunos ejemplos

Ejemplo 1 de 2:
Sobreescribir un tema

Aquí, el botón dentro del Footer recibe un valor del contexto diferente ("light") que los objetos fuera ("dark").

import { createContext, useContext } from 'react';

const ThemeContext = createContext(null);

export default function MyApp() {
  return (
    <ThemeContext.Provider value="dark">
      <Form />
    </ThemeContext.Provider>
  )
}

function Form() {
  return (
    <Panel title="Bienvenido">
      <Button>Registrarse</Button>
      <Button>Iniciar sesión</Button>
      <ThemeContext.Provider value="light">
        <Footer />
      </ThemeContext.Provider>
    </Panel>
  );
}

function Footer() {
  return (
    <footer>
      <Button>Ajustes</Button>
    </footer>
  );
}

function Panel({ title, children }) {
  const theme = useContext(ThemeContext);
  const className = 'panel-' + theme;
  return (
    <section className={className}>
      {title && <h1>{title}</h1>}
      {children}
    </section>
  )
}

function Button({ children }) {
  const theme = useContext(ThemeContext);
  const className = 'button-' + theme;
  return (
    <button className={className}>
      {children}
    </button>
  );
}


Optimizar rerenderizados al pasar objetos y funciones

Puedes pasar cualquier valor a través del contexto, incluyendo objetos y funciones.

function MyApp() {
const [currentUser, setCurrentUser] = useState(null);

function login(response) {
storeCredentials(response.credentials);
setCurrentUser(response.user);
}

return (
<AuthContext.Provider value={{ currentUser, login }}>
<Page />
</AuthContext.Provider>
);
}

Aquí, el valor del contexto es un objeto de JavaScript con dos propiedades, una de las cuales es una función. Siempre que MyApp se rerenderice (por ejemplo, en una actualización de ruta), este será un objeto diferente apuntando a una función diferente, así que React también tendrá que rerenderizar todos los componentes en lo profundo del árbol que llamen useContext(AuthContext).

En aplicaciones más pequeñas, esto no es un problema. Sin embargo, no hay necesidad de rerenderizarlas si los datos subyacentes, como currentUser, no han cambiado. Para ayudar a React a aprovechar esa información, puedes envolver la función login con useCallback y envolver la creación del objeto en un useMemo. Esta es una optimización del rendimiento:

import { useCallback, useMemo } from 'react';

function MyApp() {
const [currentUser, setCurrentUser] = useState(null);

const login = useCallback((response) => {
storeCredentials(response.credentials);
setCurrentUser(response.user);
}, []);

const contextValue = useMemo(() => ({
currentUser,
login
}), [currentUser, login]);

return (
<AuthContext.Provider value={contextValue}>
<Page />
</AuthContext.Provider>
);
}

Como resultado de este cambio, incluso si MyApp necesita rerenderizarse, los componentes que llaman useContext(AuthContext) no se rerenderizarán a menos que currentUser haya cambiado. Lee más sobre useMemo y useCallback.


Solución de problemas

Mi componente no ve el valor desde mi proveedor

Hay algunas maneras comunes en que esto puede ocurrir:

  1. Estás renderizando <SomeContext.Provider> en el mismo componente (o debajo de) donde estás llamando useContext(). Mueve <SomeContext.Provider> arriba y afuera del componente que llama useContext().
  2. Puede que hayas olvidado envolver tu componente con <SomeContext.Provider>, o quizás lo colocaste en una parte diferente del árbol de la que pensabas. Revisa si la jerarquía está correcta utilizando React DevTools.
  3. Puede que tengas un problema de compilación con tus herramientas que provoque que SomeContext como es visto desde el componente proveedor y que SomeContext como es visto desde el componente que lee sean dos objetos diferentes. Esto puede suceder si usas enlaces simbólicos, por ejemplo. Puedes verificar esto al asignarlos a variables globales como window.SomeContext1 y window.SomeContext2 y luego verificar si window.SomeContext1 === window.SomeContext2 en la consola. Si no son el mismo, necesitas arreglar ese problema a nivel de herramienta de compilación.

Siempre recibo undefined de mi contexto a pesar de que el valor por defecto es diferente

Puede que tengas un proveedor sin un value en el árbol:

// 🚩 No funciona: No hay prop value
<ThemeContext.Provider>
<Button />
</ThemeContext.Provider>

Si te olvidas de especificar un value, es como pasar value={undefined}.

Es posible que hayas utilizado un nombre de prop diferente por error:

// 🚩 No funciona: la prop debería llamarse "value"
<ThemeContext.Provider theme={theme}>
<Button />
</ThemeContext.Provider>

En ambos casos deberías ver una advertencia de React en la consola. Para solucionarlos llama a la prop value:

// ✅ Pasando la prop value
<ThemeContext.Provider value={theme}>
<Button />
</ThemeContext.Provider>

Fíjate que el valor por defecto de tu llamada createContext(defaultValue) solo es usado si no hay ningún proveedor que coincida arriba en absoluto. Si hay algún componente <SomeContext.Provider value={undefined}> en algún lugar del árbol, el componente llamando useContext(SomeContext) recibirá undefined como el valor del contexto.