Under the hood

Guillaume Chau

@Akryum

Vue.js Core Team

@Akryum@m.webtoo.ls

A story is a small number of components mounted in an isolated environment

Story:

Why stories?

Organize and document components for other developers

Showcase features and components

Develop components in isolation

Tests components

Visual regression

⚡ Dynamic source

🍱 Variant grids

📖 Markdown docs

🌔 Dark theme

📱 Responsive testing

🎹 Flexible controls

📷 Visual regression testing

🎨 Automatic design tokens

🔍 Fast fuzzy/full-text search

More to come...

Packed with features!

Writing stories

Writing stories
should look similar to writing component code

<!-- Cars.story.vue -->
<template>
  <Story title="Cars">
    <Variant title="default">
      🚗
    </Variant>
    <Variant title="Fast">
      🏎️
    </Variant>
    <Variant title="Slow">
      🚜
    </Variant>
  </Story>
</template>

How Vite powers Histoire

Vite Native

Reuse the same build pipeline

Less time and effort setting up

Fast boot and instant HMR

Virtual modules

histoire dev

Story FS watcher

.md FS watcher

Vite dev server

(HTTP server)

vite-node server

Generate docs-only stories

<!-- Cars.story.vue -->
<template>
  <Story title="Cars">
    <Variant title="default">
      🚗
    </Variant>
    <Variant title="Fast">
      🏎️
    </Variant>
    <Variant title="Slow">
      🚜
    </Variant>
  </Story>
</template>
  • Docs of Story 1 
  • Docs page 1
  • Docs page 2
  • Story 1
  • Story 2
  • Story 3
  • ...

Collect

Story FS watcher

.md FS watcher

Vite dev server

(HTTP server)

vite-node server

  • Story 1
  • Story 2
  • Story 3
  • ...
  • Docs of Story 1 
  • Docs page 1
  • Docs page 2

Vite dev server

(HTTP server)

Virtual modules

Browser

import {
  files, tree, onUpdate
} from 'virtual:$histoire-stories'
export let files = [{ ... }]
export let tree = { ... }
const handlers = []
export function onUpdate (cb) {
  handlers.push(cb)
}
if (import.meta.hot) {
  import.meta.hot.accept(newModule => {
    files = newModule.files
    tree = newModule.tree
    handlers.forEach(h => {
      h(newModule.files, newModule.tree)
      newModule.onUpdate(h)
    })
  })
}

HMR

async resolveId (id) {
  if (id === 'virtual:$histoire-stories') {
    return '\0virtual:$histoire-stories'
  }
}
async load (id) {
  if (id === '\0virtual:$histoire-stories') {
    return `export let files = [${JSON.stringify(files)}]
export let tree = ${JSON.stringify(tree)}
const handlers = []
export function onUpdate (cb) {
  handlers.push(cb)
}
if (import.meta.hot) {
  // Handle HMR...
}`
  }
}

Vite dev server

(HTTP server)

Virtual modules

Browser

HMR

stories data & tree

resolved config

theme CSS variables

setup code

support plugins

markdown files

full-text search indexes

Vite dev server

(HTTP server)

Browser

View: Story 1

stories data & tree

setup code

support plugin

markdown files

Mount: Story 1

Story1.story.vue

Render: Story 1: Content

Render: Story 1: Controls

support plugin

support plugin

Load

  • Props
  • Slots
  • Render functions

Browser

View: Story 1

stories data & tree

support plugin

Mount: Story 1

Story1Component.vue

Load

Render: Story 1: Content

Render: Story 1: Controls

support plugin

support plugin

Sync state

can be in an iframe

Browser

View: Story 1

stories data & tree

support plugin

Mount: Story 1

Story1Component.vue

Load

Render: Story 1: Content

support plugin

Auto-CodeGen: Story 1

support plugin

slot

<button
  color="primary"
  disabled
>
  Click me!
</button>

Sync state

Browser

Virtual modules

stories data & tree

resolved config

theme CSS variables

setup code

support plugins

markdown files

full-text search indexes

Static code

Vite build

HMR API

export const count = 1

if (import.meta.hot) {
  import.meta.hot.accept((newModule) => {
    if (newModule) {
      // newModule is undefined when SyntaxError happened
      console.log('updated: count is now ', newModule.count)
    }
  })
}

Accept an update for current module

import { foo } from './foo.js'

foo()

if (import.meta.hot) {
  import.meta.hot.accept('./foo.js', (newFoo) => {
    // the callback receives the updated './foo.js' module
    newFoo?.foo()
  })
}

Accept an update for a dependency

if (import.meta.hot) {
  // When current module is replaced
  import.meta.hot.dispose((data) => {
    // cleanup side effect
  })
  
  // when the module is no longer used in the page
  import.meta.hot.prune((data) => {
    // cleanup side effect
  })
}

Cleanup module side-effects

if (import.meta.hot) {
  import.meta.hot.accept((module) => {
    // You may use the new module instance
    // to decide whether to invalidate.
    if (cannotHandleUpdate(module)) {
      import.meta.hot.invalidate()
    }
  })
}

Bail out of HMR and propagate the update in parent modules

Client-Dev server communication

Using HMR API

vitePlugins.push({
  name: 'histoire:example',
  
  configureServer (server) {
    server.ws.on('some-event', (payload) => {
      server.ws.send('some-other-event', somePayload)
    })
  },
})
if (import.meta.hot) {
  import.meta.hot.send('some-event', { message: 'Hello Amsterdam' })
  
  import.meta.hot.on('some-other-event', (somePayload) => {
    console.log(somePayload)
  })
}

Browser

Vite server

Thank you!

Histoire: A deep dive (Vue Amsterdam 2023)

By Guillaume Chau

Histoire: A deep dive (Vue Amsterdam 2023)

A look at the internals of Histoire, a tool to generate component stories, and how Vite makes it possible.

  • 2,646