Lookbook v1.0.8

Inspector Panels

The Lookbook preview inspector is where each preview is rendered alongside its source code, notes and more.

The inspector is made up of a number of tabbed panels, grouped into two areas - the preview area at the top (with Preview and HTML tabs) and the drawer area at the bottom (with Source, Notes, and Params tabs).

It is possible to add custom tab panels into the drawer area, as well as to hide, reposition or reorder the system-provided panels.

Adding a custom panel

You can add a custom inspector panel using the Lookbook.define_panel method when you configure your Lookbook installation:

Lookbook.define_panel(<name>, <opts>)
<name>

A unique name for the panel

Type: String
<opts>

A Hash of panel options (see below for details)

Type: Hash

For example, a very simple ‘info’ panel could be created as follows:

# config/application.rb
Lookbook.define_panel(:info, {
  label: "Extra Info",
  partial: "panels/info"
})
<!-- views/panels/_info.html.erb -->
<div class="lookbook-panel">
  <h2>Some information</h2>
  <ul>
    <li>You are looking at the '<%= preview.label %>' preview</li>
    <li>The preview file path is: '<%= preview.full_path %>'<li>
    <li>There are <%= examples.size %> examples in this preview<li>
  </ul>
</div>

Panel partial templates can be located anywhere within your app’s views directory.

Panel options

There are a number of available options when defining a panel.

label

The text to be displayed in the tab for the panel

partial

The path to the view template partial used to render the panel

Type: String
hotkey

keyboard shortcut to make panel become the active tab

Type: String
disabled

Disabled tabs are still accessible but are greyed out in the UI

Type: Boolean
show

Whether or not to display the tab/panel

Type: Boolean,Default: true
copy

If present, the panel will display a copy button that copies the value of this property to the clipboard when clicked

Type: String
position

Position of the tab in the tab list

Type: Integer
locals

A Hash of local variables that will be passed to the panel when it is rendered

Type: Hash
Lookbook.define_panel(:info, {
  label: "New Panel",
  position: 1,
  partial: "path/to/view_partial",
  hotkey: "ctrl.n",
  disabled: false,
  show: true,
  copy: "Content to be copied",
  locals: {
    last_updated: '2022-03-02'
  }
})

All panel option values can be provided either as a simple static value or as a lambda function. Lambdas receive a single object with data relating to the currently active preview/example. For example:

{
  label: "Params",
  disabled: ->(data) { data.preview.params.none? } # grey out the Params tab if no params are set for the current preview
}

The data hash provided as the single argument to any lambda functions contains the same set of objects that are provided to the panel partial template (see below for full details):

data.preview

Preview object representing the current preview

data.examples

Array of examples that are included in the current preview

data.context

Data about the request context

data.app

The main Lookbook app instance

Panel templates

Panel template files are just regular ERB partials. Unlike pages they are not additionally parsed as markdown.

To match the padding and styles of the system panels you should ensure your panel’s content is in an element with the .lookbook-panel class applied to it.

<div class="lookbook-panel">
  <!-- your panel content here -->
</div>

Each panel is has access to a number of variables and helpers that can be used to build dynamic content.

Variables

Any locals defined in the panel options will be available, plus the following:

panel

The resolved panel options object (see above for details)

<p>
  This <%= panel.label %> panel can be switched
  to using the keyboard shortcut <%= panel.hotkey %>
</p>

preview

A preview object representing the current preview.

<h2>This is the <%= preview.label %> preview</h2>

target

An object representing the example or group of examples currently being rendered

<h2>Currently rendering '<%= target.label %>' which is <%= "not" unless target.type == :group %> a group.</h2>

examples

An array of preview example objects representing the individual examples rendered in the current preview. For non-grouped (i.e. regular) previews, this array will always only have one item.

<p>This preview is showing <%= examples.size %> examples rendered together.</p>
<p>Here is the source code for each example:</p>
<% examples.each do |example| %>
  <div>
    <%= example.source %>
  <div>
<% end %>

context

Request context data

context.preview_params

Hash of supplied preview parameters, if any

context.path

URL path of the current page

<p>Current URL path: <%= context.path %></p>

data

Shortcut for accessing the Lookbook.data store.

<p>My custom data variable value: <%= data.my_variable %></p>

app

The main Lookbook app instance.

app.previews

Array of all preview objects

app.pages

Array of all page objects

app.logger

Lookbook logger instance

<p>There are <%= app.previews.size %> previews available.</p>
<% app.logger.info("Successfully output the number of previews") %>

Helpers

The following helpers are available in panel templates:

code

Adds syntax highlighting to blocks of code.

The code helper is an alternative to using markdown code blocks for templates that have markdown parsing disabled, or for when more control is required.

code(<lang?>) { <block_content> }
<lang?>

The language the code is written in. Defaults to :ruby

Type: Symbol
<!-- Highlight Ruby code -->
<%= code do %>
  # Ruby code to highlight goes here
<% end %>

<!-- Highlight a different language (HTML) -->
<%= code :html do %>
  <!-- HTML code to highlight goes here -->
<% end %>

icon

Displays an icon.

Lookbook uses icons from the Feather Icons set and a full list of available icon names can be found on that site.

icon(<icon_name>, <opts?>)
<icon_name>

The name of the icon to display

Type: Symbol
<opts?>

Hash of options

Type: Hash
<%= icon :trash %>
<%= icon :camera, size: 6, style: "color: red;" %>

Styling panel content

If you wish to specify custom CSS rules to style the contents of the panel, just include a <style> element in the panel partial template:

<style>
  h3 {
    color: green;
  }
</style>

<div>
  <h3>My Custom Panel</h3> <!-- will be green -->
  <!-- ... -->
</div>

The <style> element will be removed when the panel is rendered and any styles will be automagically scoped to the panel that they are defined in, so they will not affect other panels or leak out to affect the styling of the UI in general.

Adding JavaScript

Lookbook uses Alpine JS for its UI, and it is available for use in panel templates too. It’s a great way to add interactivity without ever having to touch a script tag.

For example, to add a button to show/hide a piece of content in the panel, it’s as simple as this:

<div x-data="{ open: false }">
  <button @click="open = !open">Show/hide content</button>
  <div x-show="open">
    Some content here
  </div>
</div>

Check out the Alpine JS documentation for more information.

Because of the way that new content is loaded in to the UI when navigation occurs, adding arbitrary JavaScript in a script tag within the panel template may not always work as expected, so it is recommended to use Alpine for adding interactivity where required.

Utility classes

There are a number of utility classes available for use in panels to make it easier to match the look and feel of the rest of the Lookbook UI.

.lookbook-panel

Apply to the panel’s root element

.prose

Apply to text content containers. Adds default prose styles to child elements.

Editing system panels

It is possible to edit, reorder or remove any of the system-provided panels, if required. The following panels are provided by Lookbook:

:preview

Renders the preview in a resizable container

:output

The rendered preview HTML source code

:source

The preview example method source code (or preview template source, if using)

:notes

Preview notes panel

:params

Inputs and controls for dynamic preview parameters

Removing a panel

To remove a panel from the UI entirely you can use the Lookbook.remove_panel method.

Lookbook.remove_panel(<name>)
<name>

The name of the panel to remove (see above for a list of valid panel names)

Type: Symbol
Lookbook.remove_panel(:notes) # Notes panel will not appear in the UI

Editing panel properties

If you want to tweak any properties of a system-defined panel, you can do so using the Lookbook.amend_panel method.

Lookbook.amend_panel(<name>, <opts>)
<name>

The name of the panel to edit (see above for a list of valid panel names)

Type: Symbol
<opts>

A Hash of panel options to override (see above for full option details)

Type: Hash
Lookbook.amend_panel(:params, {
  label: "Knobs", # change the tab text
  hotkey: "ctrl.k", # override hotkey
  position: 1 # move to first position in the tab list
})