npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@alveole/storybook

v0.29.4

Published

Catalog screens and helpers for Alveole stories and theme tokens.

Downloads

1,038

Readme

@alveole/storybook

Composants réutilisables pour exposer un catalogue de stories et de tokens de thème.

Le package reprend les grandes parties du ui-kit de groove-application/app/ui-kit, mais sous une forme réutilisable et indépendante d'Expo Router.

Prérequis

Le package est prévu pour être utilisé avec :

  • @alveole/components
  • @alveole/theme
  • react
  • react-native

Il doit être rendu dans une app déjà enveloppée par ThemeProvider.

Exports disponibles

Le package exporte :

  • StoriesScreen
  • StoryDetailScreen
  • ThemeConstantsScreen
  • ThemeConstantDetailScreen
  • ThemePaletteScreen
  • ThemeTypographyScreen
  • JsonBlock
  • les types StorybookMeta, StorybookModule, StorybookFlag, StorybookFlagKey
  • les helpers de utils

Ecriture recommandee des stories

Le package @alveole/storybook est pense pour consommer directement les stories ecrites comme dans packages/components.

Le pattern recommande est :

  • un export default qui satisfait le type Story
  • des export const pour chaque exemple rendu

Exemple proche de l'ecriture actuelle de Box.stories.tsx :

import { Story } from '../../type';
import { Typography } from '../Typography';
import { Box } from './Box';

export default {
  title: 'Box',
  tags: ['Kit'],
  experimental: false,
  description: 'Description du composant',
  component: Box,
  styleFn: () => 'Aucun style applique',
} satisfies Story;

export const BoxDefault = () => (
  <Box tag="box">
    <Typography>Contenu par defaut</Typography>
  </Box>
);

export const BoxPadded = () => (
  <Box p={16}>
    <Typography>Box avec padding</Typography>
  </Box>
);

Le default porte les metadonnees de la story. Les autres exports sont les variantes affichees par StoryDetailScreen.

Concretement, @alveole/storybook n'introduit pas un nouveau format de story. Il reutilise celui que tu ecris deja dans packages/components/src/**/*.stories.tsx.

Structure attendue par le package

Quand tu fais :

import * as Stories from '@alveole/components/stories';

chaque module exporte se retrouve sous une forme equivalente a :

import type { StorybookModule } from '@alveole/storybook';

const ButtonStory = {
  default: {
    title: 'Button',
    tags: ['Kit'],
    experimental: false,
    description: 'Boutons de l application',
    figmaURL: 'https://figma.com/...',
    styleFn: () => ({}),
  },
  Primary: () => <MyButton label="Primary" />,
  Secondary: () => <MyButton label="Secondary" />,
} satisfies StorybookModule;

Autrement dit :

  • default doit contenir les metadonnees
  • les autres exports doivent etre des composants React affichables
  • il ne faut pas rendre StoriesScreen ou StoryDetailScreen a l'interieur d'une story unitaire
  • le bon point d'entree est un ecran catalogue qui consomme @alveole/components/stories

Exemple minimal pour les stories

import * as Stories from '@alveole/components/stories';
import { StoriesScreen, StoryDetailScreen, type StorybookModule } from '@alveole/storybook';
import React from 'react';

const storyList = Object.values(Stories) as StorybookModule[];

export const StoryCatalogPage = () => {
  const [selectedStory, setSelectedStory] = React.useState<StorybookModule | null>(null);

  if (selectedStory) {
    return <StoryDetailScreen story={selectedStory} />;
  }

  return (
    <StoriesScreen
      stories={storyList}
      title="UI Kit - Components"
      description="Catalogue des composants"
      onSelectStory={setSelectedStory}
    />
  );
};

Exemple avec Expo Router

Le package ne crée pas les routes. Il fournit seulement les écrans. À toi de brancher la navigation.

import * as Stories from '@alveole/components/stories';
import { StoriesScreen, type StorybookModule } from '@alveole/storybook';
import { router } from 'expo-router';

const storyList = Object.values(Stories) as StorybookModule[];

export default function UIKitStoriesRoute() {
  return (
    <StoriesScreen
      stories={storyList}
      onSelectStory={story => {
        router.push(`/ui-kit/components/${encodeURIComponent(story.default.title)}`);
      }}
    />
  );
}

Puis dans la route de détail :

import * as Stories from '@alveole/components/stories';
import { StoryDetailScreen, type StorybookModule } from '@alveole/storybook';
import { useLocalSearchParams } from 'expo-router';

export default function UIKitStoryDetailRoute() {
  const { component } = useLocalSearchParams<{ component: string }>();

  const story = (Object.values(Stories) as StorybookModule[]).find(entry => entry.default.title === component) ?? null;

  return <StoryDetailScreen story={story} />;
}

Exemple pour les constantes du thème

import * as ThemeConstants from '@alveole/theme';
import { ThemeConstantsScreen } from '@alveole/storybook';

export default function ThemeConstantsPage() {
  return (
    <ThemeConstantsScreen constants={ThemeConstants} title="UI Kit - Constants" description="Constantes du thème" />
  );
}

Exemple pour la palette

import { CustomPalette } from '@alveole/theme';
import { ThemePaletteScreen } from '@alveole/storybook';

export default function ThemePalettePage() {
  return <ThemePaletteScreen palette={CustomPalette} />;
}

Exemple pour la typographie

import { CustomTypography } from '@alveole/theme';
import { ThemeTypographyScreen } from '@alveole/storybook';

export default function ThemeTypographyPage() {
  return <ThemeTypographyScreen typography={CustomTypography} />;
}

Props principales

StoriesScreen

  • stories: liste des stories
  • title: titre de page
  • description: description de page
  • emptyMessage: message affiché si aucun résultat
  • createLabel: libellé du bouton d’action
  • onCreatePress: callback du bouton d’action
  • onSelectStory: callback au clic sur une story

StoryDetailScreen

  • story: story sélectionnée
  • notFoundMessage: message si la story est absente

ThemeConstantsScreen

  • constants: objet de constantes
  • title: titre de page
  • description: description de page
  • onSelectConstant: callback au clic sur une constante

ThemeConstantDetailScreen

  • name: nom de la constante
  • value: valeur de la constante

ThemePaletteScreen

  • palette: objet de palette
  • title: titre de page
  • description: description de page

ThemeTypographyScreen

  • typography: objet de tokens typo
  • title: titre de page
  • description: description de page

Notes

  • Le package n’embarque pas de moteur Storybook officiel.
  • Il ne dépend pas d’Expo Router, mais il est pensé pour s’y brancher facilement.
  • StoryDetailScreen ouvre le lien Figma via Linking.openURL.
  • Les vues de détail affichent les objets sous forme JSON via JsonBlock.