New
Storybook 7 DocsAutomate with Chromatic
Storybook Day 2023
Star77,071
Back to Create an Addon
React
Chapters
  • Introducción
  • Configuración
  • Registrar complemento
  • Seguimiento de Estado
  • Decoradores
  • Preajuste
  • Agregar al catálogo
  • Conclusión

Decoradores

Interactuando con las historias

Ya casi. Hasta ahora, creamos una herramienta, la agregamos a la barra de herramientas e incluso rastrea el estado. Ahora debemos responder a ese estado y mostrar u ocultar los contornos.

Los Decoradores envuelven historias y agregan funcionalidad adicional de renderizado. Crearemos un decorador que responda al esquema global y maneje la inyección de CSS. Lo que a su vez, dibuja contornos alrededor de todos los elementos HTML.

En el paso anterior definimos el outlineActive (esquema activo) global, ¡ahora conectémoslo! Podemos consumir globales en un decorador usando el hook useGlobals.

Copy
src/withGlobals.js
/* eslint-env browser */
import { useEffect, useGlobals } from '@storybook/addons';

export const withGlobals = (StoryFn, context) => {
  const [{ outlineActive }, updateGlobals] = useGlobals();
  // ¿Se está utilizando el complemento en el panel de documentación?
  const isInDocs = context.viewMode === 'docs';

  useEffect(() => {
    // Ejecuta tu efecto secundario aquí
    // Por ejemplo, para manipular el contenido de la vista previa.
    const selectorId = isInDocs ? `#anchor--${context.id} .docs-story` : `root`;

    displayToolState(selectorId, { outlineActive, isInDocs });
  }, [outlineActive]);

  return StoryFn();
};

function displayToolState(selector, state) {
  const rootElement = document.getElementById(selector);
  let preElement = rootElement.querySelector('pre');

  if (!preElement) {
    preElement = document.createElement('pre');
    preElement.style.setProperty('margin-top', '2rem');
    preElement.style.setProperty('padding', '1rem');
    preElement.style.setProperty('background-color', '#eee');
    preElement.style.setProperty('border-radius', '3px');
    preElement.style.setProperty('max-width', '600px');
    rootElement.appendChild(preElement);
  }

  preElement.innerText = `Este snippet es inyectado por el decorador withGlobals.
Se actualiza a medida que el usuario interactúa con la herramienta ⚡ en la barra de herramientas de arriba.
${JSON.stringify(state, null, 2)}
`;
}

Inyectando el CSS del contorno

Agregar y borrar estilos es un efecto secundario, por lo tanto, necesitamos envolver esa operación usando useEffect; que a su vez es activado por el outlineActive global. El código del kit viene con un ejemplo, pero actualicémoslo para manejar la inyección CSS del contorno.

Copy
src/withGlobals.js
/* eslint-env browser */
import { useEffect, useMemo, useGlobals } from '@storybook/addons';

import { clearStyles, addOutlineStyles } from './helpers';
import outlineCSS from './outlineCSS';

export const withGlobals = (StoryFn, context) => {
  const [{ outlineActive }, updateGlobals] = useGlobals();
  // ¿Se está utilizando el complemento en el panel de documentación?
  const isInDocs = context.viewMode === 'docs';

  const outlineStyles = useMemo(() => {
    const selector = isInDocs ? `#anchor--${context.id} .docs-story` : '.sb-show-main';

    return outlineCSS(selector);
  }, [context.id]);

  useEffect(() => {
    const selectorId = isInDocs ? `my-addon-outline-docs-${context.id}` : `my-addon-outline`;

    if (!outlineActive) {
      clearStyles(selectorId);
      return;
    }

    addOutlineStyles(selectorId, outlineStyles);

    return () => {
      clearStyles(selectorId);
    };
  }, [outlineActive, outlineStyles, context.id]);

  return StoryFn();
};

Ok, parece un gran salto. Repasemos todos los cambios.

El complemento puede estar activo en los modos de vista de documentación y de historia. El nodo DOM real para el iframe de vista previa es diferente en estos dos modos. De hecho, el modo de documentación muestra varias vistas previas de historias en una página. Por lo tanto, debemos elegir el selector apropiado para el nodo DOM donde se inyectarán los estilos. Además, el CSS debe ajustarse a ese selector en particular.

Nota: El useMemo y useEffect aquí provienen de @storybook/addons y no de React. Esto se debe a que el código del decorador se ejecuta en la parte de vista previa de Storybook. Ahí es donde se carga el código del usuario que puede no contener React. Por lo tanto, para ser independiente del framework (marco), Storybook implementa una biblioteca de hooks similar a React que podemos utilizar.

A continuación, a medida que inyectamos los estilos en el DOM, debemos realizarles un seguimiento para borrarlos cuando el usuario lo desactiva o cuando cambia el modo de visualización.

Para administrar toda esta lógica de CSS, necesitamos algunos ayudantes. Éstos utilizan APIs de DOM para inyectar y eliminar hojas de estilo.

Copy
src/helpers.js
/* eslint-env browser */
export const clearStyles = selector => {
  const selectors = Array.isArray(selector) ? selector : [selector];
  selectors.forEach(clearStyle);
};

const clearStyle = selector => {
  const element = document.getElementById(selector);
  if (element && element.parentElement) {
    element.parentElement.removeChild(element);
  }
};

export const addOutlineStyles = (selector, css) => {
  const existingStyle = document.getElementById(selector);
  if (existingStyle) {
    if (existingStyle.innerHTML !== css) {
      existingStyle.innerHTML = css;
    }
  } else {
    const style = document.createElement('style');
    style.setAttribute('id', selector);
    style.innerHTML = css;
    document.head.appendChild(style);
  }
};

El CSS del contorno en sí mismo se basa en lo que usa Pesticide. Tómalo del archivo outlineCSS.js.

En conjunto, esto nos permite trazar contornos alrededor de los elementos UI.

toggling the tool toggles the outlines

Keep your code in sync with this chapter. View 0e7246a on GitHub.
Is this free guide helping you? Tweet to give kudos and help other devs find it.
Next Chapter
Preajuste
Habilita Outline (contorno) para cada historia
✍️ Edit on GitHub – PRs welcome!
Join the community
5,959 developers and counting
WhyWhy StorybookComponent-driven UI
Open source software
Storybook

Maintained by
Chromatic
Special thanks to Netlify and CircleCI