Portal is implemented with reagent so implementing a custom viewer is as simple as writing a reagent component, this should be relatively easy assuming you are familiar with reagent.
In this guide I will walk you through creating a slide deck viewer in Portal.
If you would like to interactively develop your custom viewer, and are using
nREPL, you can use the portal.nrepl/wrap-repl
middleware. With that middleware
enabled, you can launch a sub repl with the following:
(require '[portal.api :as p])
(def portal (p/open))
;; Turns the current nREPL session into a portal repl session.
(p/repl portal)
Now all subsequent eval invocations will run in the Portal UI ClojureScript runtime.
note
To quit the Portal repl, eval :cljs/quit
. This will automatically
occur if the portal session is no longer available.
For this particular viewer, we can enable any collection to act as a slide deck and allow each element in the sequence to decide how it wants to be rendered.
For example, the following should be considered a valid slide deck:
[^{:portal.viewer/default :portal.viewer/hiccup}
[:h1 "hello"]
^{:portal.viewer/default :portal.viewer/hiccup}
[:h1 "world"]]
Let's express this as a :predicate
for later use:
(defn deck? [value] (and (coll? value) (not (map? value))))
The only assumption that Portal makes about viewer components is that they take a value as their first argument, everything else is up to the component.
For example, the following is a valid Portal viewer:
(defn viewer [value] [:pre {:style {:color :pink}} (pr-str value)])
However, that's not very interesting. For the deck viewer, let's render the currently selected element in the sequence as a slide:
(ns portal-present.viewer
(:require [portal.ui.api :as p]
[portal.ui.inspector :as ins]
[reagent.core :as r]))
(defn view-presentation []
(let [slide (r/atom 0)]
(fn [slides]
[:<>
[ins/inspector (nth (seq slides) @slide :no-slide)]
[:button {:on-click #(swap! slide dec)} "prev"]
[:button {:on-click #(swap! slide inc)} "next"]])))
When you are satisfied with your component, you can register it with Portal via
portal.ui.api/register-viewer!
such as the following:
(portal.ui.api/register-viewer!
{:name ::slides
:predicate deck?
:component view-presentation})
note
Anytime a viewer is registered, it will cause the UI to re-render. This is very handy for interactive development.
And with just that, you now have a fully working custom viewer!
The next step, is getting the code loaded into the Portal UI runtime without
using portal.api/repl
. This can be done via portal.api/eval-str
which
enables evaluating arbitrary ClojureScript code in the Portal UI runtime from
the host runtime.
For example, the following will alert 1
when invoked at the host repl.
(require '[portal.api :as p])
(p/eval-str "(js/alert 1)")
To load the code for an extension, we can provide the string directly or load it from a file via slurp. For example:
(require '[portal.api :as p])
(p/eval-str (slurp (io/resource "portal_present/viewer.cljs")))
note
you can specify which Portal instance you want this code evaluated in to prevent evaluating it in every available Portal UI runtime.
Now that the viewer has been defined and loaded, you can select it like any viewer or enable it by default via metadata:
(require '[portal.api :as p]
'[portal.viewer :as-alias v])
(def slides
^{::v/default :portal-present.viewer/slides}
[^{::v/default ::v/hiccup} [:h1 "hello"]
^{::v/default ::v/hiccup} [:h1 "world"]])
(add-tap p/submit)
(tap> slides)
An issue with the current loading mechanism is that is won't persist across
browser reloads or portal.api/open
invocations. To address this issue, we can
auto load the viewer via the :on-load
hook. For example:
(require '[portal.api :as p])
(declare portal)
(defn on-load []
(p/eval-str portal (slurp (io/resource "portal_present/viewer.cljs"))))
(def portal
(p/open
{:value slides
:on-load on-load
:window-title "Portal Present"}))
With only the concepts above, you can now build a variety of custom viewers! For a more complete example that builds on the above with improved styles and UX, look at portal-present.viewer.
Can you improve this documentation? These fine people already did:
Chris Badahdah & Martin ClausenEdit on GitHub
cljdoc is a website building & hosting documentation for Clojure/Script libraries
× close