Starchart Studio is a tool to create component stories, similar to Storybook, but designed with JavaScript islands and static site generators in mind and powered by Astro.
This is a preview release! It's enough to get going with but the UX and DX still needs some work. If you wanna help, please file an issue and we can chat!
Current features:
- MDX powered stories
- Static props and dynamic events
- Variants
- Isolated, resizable component preview area
- Minimalist styling
Start by installing Astro MDX , then install this module:
npm install starchart-studio
Next, determine what folder you want to keep your stories in. Stories are MDX files that export information for Starchart Studio to build your story from as well as your long-form documentation. src/stories
is a good option, but it can be anywhere. A story looks like this (using Astro's starter Card component):
import Card from '../components/Card.astro';
export const story = {
title: 'Card',
component: Card,
variants: [
{
props: [
{
name: 'title',
type: 'text',
default: 'Card Title',
description: 'The title of the card',
},
{
name: 'body',
type: 'text',
default: 'Card Body',
description: 'The body of the card',
},
{
name: 'href',
type: 'url',
default: 'https://example.com',
},
],
}
]
};
This is Astro's default card component! It's a list item that links to a URL and has a title and body text.
Your story needs to export a story
object with the following properties:
title
- The title of your storycomponent
- The imported component for your story.variants
- The different variants for this components, an array of objects containing the following:title
(optional) - The title of a variantprops
(optional) - The properties to pass to your component. Each property needs aname
(the property name) and atype
that corresponds to an HTML input type. It can also optionally have adefault
which will be passed in, adescription
of the property, and alabel
to describe the property. Properties get passed in at render time, so cannot be changed once rendered.state
(optional) - State that affects your component. State and props share a structure. Hooking up state is a little more complex and is covered below.
Next, create a page (in your Astro src/pages
directory) to generate the stories in. It can be in whatever folder structure or filename you want, as long as the filename ends in [story]
for each story to be properly generated (for instance style-guide/[story].astro
, stories-[story].astro
). The generated page will replace [story]
with the filename (minus .mdx
) of your story. Right now, we recommend a flat folder structure in your stories directory. Once that file is created, add the following to it:
---
import Starchart from 'starchart-studio/Starchart.astro';
import { StarchartStudio } from 'starchart-studio';
/**
* Builds static paths for stories
* @return {Promise<{params: Object, props: Object}[]>}
*/
export async function getStaticPaths() {
// Pass in the glob to your stories
return StarchartStudio.getStaticPaths(Astro.glob('../../stories/*.mdx'));
}
// Chart out your components
const starchart = StarchartStudio.chart(Astro);
---
<!-- Render your Starchart -->
<Starchart {...starchart} />
This will generate unique pages for each story, including stand-alone components for previewing. You can style the page with global CSS, or you can grab the individual sub-components and render them yourself.
If you need your component to load on the client, or you want to pass state, you're going to need to create a wrapper component as dynamic elements can't be hydrated. Doing so can be as simple as creating an Astro component like so:
---
import Counter from '../components/Counter.svelte';
export interface Props {
start: number;
}
const { start } = Astro.props;
---
<Counter start={start} client:load />
Here we've created a file src/stories/counter.astro
that takes in the same properties as our component (to let us pass properties down), imported our Counter component, and rendered it, passing in the prop and loading it. We then use this component instead of our counter component directly in our story.
If we want to control our component's state, we start by doing this, then we need to add a little script to let Starchart Studio know how to manage state:
<script>
import { Starchart } from '@starchart/Starchart';
// Import your state manager or store here to use with this component
import { count } from '../../counterStore';
// Create a new starchart instance for this component. Use the name of your story here (this would be for counter.mdx)
const starchart = new Starchart('counter');
// This will fire whenever any state update for counter comes in from Starchart, in the form of an object with name: value pairs
starchart.onUpdate((data) => {
console.log(data);
});
// This will fire only when an update to the 'count' state comes in from Starchart. You'll get the value
starchart.onUpdateTo('count', (data) => {
// Here you update state according to your state manager
count.set(Number(data));
});
// You can also go in the other direction. Here we're subscribing to state changes and telling Starchart to update the 'count' state with the provided value, letting state changes work in both directions
count.subscribe((value) => {
starchart.send('count', value);
});
</script>
This uses Astro's script bundling to let you listen for changes sent from Starchart so you can update your component's state, and even lets you update Starchart's state input when state in your component changes! Remember, sharing state in an islands architecture is different than in single-page apps, so you may need to reconsider how you manage state.
Starchart will automatically build out the menu of components for you, hook up properties and send and listen for state changes, and render your stories into a nice display. It'll also create an isolated environment for each component preview that's resizable.
Icons are from Google Material Symbols licensed under the Apache 2.0 license and modified to swap height and width for viewBox
.
Component resize area is heavily influenced by ish..