View Source Component stories

Basic component documentation is in PhoenixStorybook.Story.

Documentation

Component documentation is fetched from your component doc tags:

  • For a live_component, fetches @moduledoc content.
  • For a function component, fetches @doc content from the matching function.

If you are deploying phoenix_storybook in production with an Elixir release, make sure your doc chunks are not stripped out from the release.

releases: [
  my_app_web: [
    strip_beams: [
      keep: ["Docs"]
    ]
  ]
]

Variation groups

You may want to present different variations of a component in a single variation block. It is possible using PhoenixStorybook.VariationGroup.

Container

By default, each variation is rendered within a div in the storybook DOM. You can pass additional HTML attributes or extend the class attribute.

# storybook/my_component.story.exs
defmodule Storybook.MyComponent do
  use PhoenixStorybook.Story, :component
  def container, do: {:div, class: "block"}
end

If you need further sandboxing you can opt in for iframe rendering.

  • For function components, storybook will use the iframe srcdoc attribute (the whole iframe content is inlined as an HTML attribute).
  • For live components, storybook will use the typical iframe behavior, triggering an extra HTTP request to fetch the iframe content.
# storybook/my_component.story.exs
defmodule Storybook.MyComponent do
  use PhoenixStorybook.Story, :component
  def container, do: :iframe
  # or def container, do: {:iframe, style: "display: inline; ..."}
end

ℹ️ Learn more on this topic in the sandboxing guide.

Aliases & Imports

When using nested components or JS commands, you might need to reference other functions or components. Whilst it is possible to use fully qualified module names, you might want to provide custom aliases and imports.

Here is an example defining both:

defmodule NestedComponent do
  use PhoenixStorybook.Story, :component
  def function, do: &NestedComponent.nested_component/1

  def aliases, do: [MyStorybook.Helpers.JSHelpers]
  def imports, do: [{NestedComponent, nested: 1}]

  def variations do
    [
      %Variation{
        id: :default,
        slots: [
          """
          <.nested phx-click={JSHelpers.toggle()}>hello</.nested>
          <.nested phx-click={JSHelpers.toggle()}>world</.nested>
          """
        ]
      }
    ]
  end
end

Templates

You may want to render your components within some wrapping markup. For instance, when your component can only be used as a slot of another enclosing component.

Some components, such as modals, slideovers, and notifications, are not visible from the start: they first need user interaction. Such components can be accompanied by an outer template, that will for instance render a button next to the component, to toggle its visibility state.

Variation templates

You can define a template in your component story by defining a template/0 function. Every variation will be rendered within the defined template, the variation itself is injected in place of <.psb-variation/>.

def template do
  """
  <div class="my-custom-wrapper">
    <.psb-variation/>
  </div>
  """
end

You can also override the template, per variation or variation_group by setting the :template key to your variation. Setting it to a falsy value will disable templating for this variation.

Variation group templates

Variation groups can also leverage on templating:

  • either by wrapping every variation in their own template.
"""
<div class="one-wrapper-for-each-variation">
  <.psb-variation/>
</div>
"""
  • or by wrapping all variations as a whole, in a single template.
"""
<div class="a-single-wrapper-for-all">
  <.psb-variation-group/>
</div>
"""

If you want to get unique id, you can use :variation_id that will be replaced, at rendering time by the current variation (or variation group) id.

Placeholder attributes

In template, you can pass some extra attributes to your variation. Just add them to the .psb-variation or .psb-variation-group placeholder.

"""
<.form_for :let={f} for={%{}} as={:user}>
  <.psb-variation form={f}/>
</.form>
"""

JS-controlled visibility

Here is an example of templated component managing its visibility client-side, by toggling CSS classes through JS commands.

defmodule Storybook.Components.Modal do
  use PhoenixStorybook.Story, :component

  def function, do: &Components.Modal.modal/1

  def template do
    """
    <div>
      <button phx-click={Modal.show_modal()}>Open modal</button>
      <.psb-variation/>
    </div>
    """
  end

  def variations do
    [
      %Variation{
        id: :default_modal,
        slots: ["<:body>hello world</:body>"]
      }
    ]
  end
end

Elixir-controlled visibility

Some components don't rely on JS commands but need external assigns, like a modal that takes a show={true} or show={false} assign to manage its visibility state.

PhoenixStorybook handles special psb-assign and psb-toggle events that you can leverage on to update some properties that will be passed to your components as extra assigns.

defmodule Storybook.Components.Slideover do
  use PhoenixStorybook.Story, :component
  def function, do: &Components.Slideover.slideover/1

  def template do
    """
    <div>
      <button phx-click={JS.push("psb-assign", value: %{show: true})}>
        Open slideover
      </button>
      <.psb-variation/>
    </div>
    """
  end

  def variations do
    [
      %Variation{
        id: :default_slideover,
        attributes: %{
          close_event: JS.push("psb-assign", value: %{variation_id: :default_slideover, show: false})
        },
        slots: ["<:body>Hello world</:body>"]
      }
    ]
  end
end

Template code preview

By default, the code preview will render the variation and its template markup as well. You can choose to render only the variation markup, without its surrounding template by using the psb-code-hidden HTML attribute.

"""
<div psb-code-hidden>
  <button phx-click={Modal.show_modal()}>Open modal</button>
  <.psb-variation/>
</div>
"""

Block, slots & let

Liveview let you define blocks of HEEx content in your components, referred to as as slots. They can be passed in your variations with the :slots keys :

%Variation{
  id: :modal,
  slots: [
    """
    <:button>
      <button type="button">Cancel</button>
    </:button>
    """,
    """
    <:button>
      <button type="button">OK</button>
    </:button>
    """
  ]
}

You can also use LiveView let mechanism to pass data to your default block. You just need to declare the let attribute you are using in your variation.

%Variation{
  id: :list,
  attributes: %{stories: ~w(apple banana cherry)},
  let: :entry,
  slots: [
    "I like <%= entry %>"
  ]
}

let syntax can also be used with named slots, but requires no specific livebook setup.

%Variation{
  id: :table,
  attributes: %{
    rows: [
      %{first_name: "Jean", last_name: "Dupont"},
      %{first_name: "Sam", last_name: "Smith"}
    ]
  },
  slots: [
    """
    <:col :let={user} label="First name">
      <%= user.first_name %>
    </:col>
    """,
    """
    <:col :let={user} label="Last name">
      <%= user.last_name %>
    </:col>
    """
  ]
}

Late evaluation

In some cases, you want to pass to your variation attributes a complex value which should be evaluated at runtime but not in code preview (where you rather want to see the orignal expression).

For instance with the following variation of a Modal component.

%Variation{
  attributes: %{
    :"on-open": JS.push("open"),
    :"on-close": {:eval, ~s|JS.push("close")|}
  }
}

Both open & close events would work, but code would be rendered like this.

<.modal
  on-open="%Phoenix.LiveView.JS{ops: [["push", %{event: "open"}]]}"
  on-close={JS.push("close")}
/>

Layout

You can control how your story variations are rendered in the stories tab by defining an optional layout/0 function in any of your component or live_component stories.

By default a story will be rendered with the :two_columns layout. But you can use the alternate :one_column layout making the component preview taking the full container width.

defmodule Storybook.Components.Breadcrumb do
  use PhoenixStorybook.Story, :component

  def layout, do: :one_column

  # ...
end