Develop ClojureScript UI components in isolation in a "visual REPL". Increase your visual bandwidth by seeing your components in various states, screen resolutions, and other configurations simultaneously.
Portfolio brings some of the best features of Storybook.js to ClojureScript, and adds a few of its own. While Storybook.js was its starting point, Portfolio does not aspire to feature-parity with it, and instead caters to the REPL-oriented ClojureScript development process.
Portfolio is stable and ready to use. APIs documented in this document are final and will not change. APIs not explicitly documented in this document, especially those pertaining to extending and customizing the UI, may still be subject to change.
Is Portfolio good enough that you can port over from Storybook? Probably, yes. If you're using Storybook extensions not covered by Portfolio, open an issue.
I recently presented Portfolio at London Clojurians, you can watch it on YouTube.

There is a live sample to check out. The source code for the sample is also available.
With tools.deps:
no.cjohansen/portfolio {:mvn/version "2024.03.18"}
With Leiningen:
[no.cjohansen/portfolio "2024.03.18"]
Portfolio displays your components in "scenes". A scene is a component in a specific state. At its most minimal, a scene is just a named instance of a component:
(defscene button
  [:button.button "I am a button"])
After you have created your scenes, start the UI:
(require '[portfolio.ui :as ui])
(ui/start!)
You can add custom CSS and custom HTML to render your scenes with.
To use Portfolio with shadow-cljs, you must ensure that Portfolio's resources
are served by the development HTTP server. Include "classpath:public" in your
:dev-http sources:
:dev-http {8080 ["public" "classpath:public"]}
This will serve files from public in your project (where presumably your
index.html and CSS files are), and resources in public on the classpath (e.g.
Portfolio's resources). Adjust as necessary.
Check out this repo for a sample setup with an app target and a portfolio target with shadow-cljs.
If you are using shadow-cljs to build the front-end and leiningen to run the dev
server separately you need to make sure that you add classpath:public to the
:resource-paths and the dependency to Portfolio to the :dependencies in
project.clj. This will then serve the necessary assets to make Portfolio work.
Portfolio comes with a bunch of handy (and optional) tools to help you develop components:
Portfolio automatically organizes your scenes by component and application layer, and offers many knobs and tweaks to give you full control. Read about organization.
Search your scenes and collections. This feature is not enabled by default, as it's assumed not to be very useful until you have enough content. To enable it, create an index and pass it when you start the UI:
(require '[portfolio.ui.search :as search]
         '[portfolio.ui :as ui])
(ui/start!
 {:config
  {:css-paths ["/styles/app.css"]}
  :index (search/create-index)})
create-index returns an implementation of
portfolio.ui.search.protocols/Index. You can provide custom implementations of
this protocol to completely customize the search. More documentation on this
will follow.
If you just want to see what a specific component looks like with some data you
caught in your REPL, but don't necessarily want to commit a new scene, you can
tap> components, and Portfolio will render it under a dedicated folder.
;; Evaluate this expression
(tap> (MyComponent {:text "Test"}))
You can take the Portfolio UI for a spin by cloning this repo, starting figwheel, and then checking out http://localhost:5995, which will show you all the scenes defined in the sample project. There are also some scenes in the "mirror" build, which demonstrate some of Portfolio's own UI components, available at http://localhost:5995/mirror.html.
The defscene macro can be placed anywhere you like - in separate files, or
inline in production code alongside the implementation being demonstrated. In
the latter case, you probably want to strip the macros from you production
builds. It is assumed that most people will put Portfolio scenes in a separate
directory that can easily be excluded from production builds, so Portfolio is
enabled by default. To disable it in your build, use any of the following two
options.
Add :portfolio/enabled? false to your ClojureScript compiler options:
{:main "myns.prod"
 :optimizations :advanced
 :source-map true
 :portfolio/enabled? false}
Your second option is to set the portfolio.core/enabled Closure
define to
false. Closure defines can be set several ways, see the link.
Yes please! Feel free to contribute more framework adapters, UI extensions or whatever. Please open an issue or a draft PR to discuss larger changes before pouring too much work into them, so we're sure we're one the same page.
Portfolio is proudly sponsored by Clojurists Together.
Some features and fixes that are likely to be explored in the close future:
:on-mountdefscene:background/document-data:background/document-classtap> components from your code or the REPL to render them in Portfolio.When running Portfolio in split mode, you can select specific scenes for either of the panes. This way you can not only compare different versions of the same scene, but compare different scenes to each other.
Scenes and collections can now have Markdown docstrings. These render above the scenes, and there is a new toolbar button to toggle their display on or off.
Portfolio can now also optionally display the scene code. This is toggled off by default, and can be enabled by clicking the brackets button in the toolbar.
There are some improvements to Portfolio's default organization into packages and folders, and particularly their interaction with your custom collection configuration.
You can now specify default scene icons for collections.
Portfolio now watches all atoms in :param. This means you can set :param to
e.g. a map or a vector, put several atoms inside (arbitrarily nested), and
Portfolio will re-render the scene whenever any of them change. To reflect that
:param is no longer necessarily just one thing, it has also been aliased as
:params.
Add search, including APIs for customizing indexing, searching, and result preparation.
Added new APIs for organizing scenes in the sidebar.
Copyright © 2022-2023 Christian Johansen
Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.
Can you improve this documentation? These fine people already did:
Christian Johansen, Brandon Stubbs, Michael Salihi, Teodor Heggelund & Boris KourtoukovEdit on GitHub
cljdoc builds & hosts documentation for Clojure/Script libraries
| Ctrl+k | Jump to recent docs | 
| ← | Move to previous article | 
| → | Move to next article | 
| Ctrl+/ | Jump to the search field |