Literate is a Clojure & ClojureScript application which you can use to create interactive web documents from a Clojure REPL.
See the welcome example document.
Literate is essentially a collection of Widgets, and a Widget is a visual component, or renderer, for a particular type of data.
It's a tricky thing to define 'type' of data, but I will show you some examples, and hopefully, it will make things more straightforward.
Our example is a collection of maps; it's easy to think about this data in a table, let's say, like this one:
A | B |
---|---|
X | 1 |
Y | 3 |
In Literate, what you could do is view this same data but in a Vega-Lite Bar chart, for instance. And this is accomplished by Widgets, a Vega-Lite Widget in particular:
(literate/vega-lite
{"$schema" "https://vega.github.io/schema/vega-lite/v4.json"
:description "A simple bar chart with embedded data."
:data {:values
[{:a "A" :b 28}
{:a "B" :b 55}
{:a "C" :b 43}
{:a "D" :b 91}
{:a "E" :b 81}
{:a "F" :b 53}
{:a "G" :b 19}
{:a "H" :b 87}
{:a "I" :b 52}]}
:mark "bar"
:encoding {:x {:field "a"
:type "ordinal"}
:y {:field "b"
:type "quantitative"}}})
We are not too bad at making sense of tabular data. Still, if you happen to work with spatial data, it isn't easy to make sense without a graphical representation. It's similar to talking about geometry without drawing the shapes on a piece of paper. Very hard.
For spatial data, you could view it in a Vega-Lite Widget too (Vega is capable of many great things), but there's also the Leaflet Widget:
(literate/leaflet
{:style {:height "600px"}
:center center
:zoom 10
:geojson geojson})
You see, it's tricky to define 'type' of data because it can take many different shapes and forms. But the idea is the same: Widgets take data in and present it visually.
The Widget API is embarrassingly dummy; whenever you call a Widget function, you are merely creating an entity:
(literate/vega-lite
{"$schema" "https://vega.github.io/schema/vega-lite/v4.json"
:description "A simple bar chart with embedded data."
:data {:values
[{:a "A" :b 28}
{:a "B" :b 55}
{:a "C" :b 43}
{:a "D" :b 91}
{:a "E" :b 81}
{:a "F" :b 53}
{:a "G" :b 19}
{:a "H" :b 87}
{:a "I" :b 52}]}
:mark "bar"
:encoding {:x {:field "a"
:type "ordinal"}
:y {:field "b"
:type "quantitative"}}})
;; ==>
#:widget{:uuid "7385f5da-f835-49b5-8d8c-005f92b77d11",
:type :widget.type/vega-embed,
:vega-lite-spec {"$schema" "https://vega.github.io/schema/vega-lite/v4.json",
:description "A simple bar chart with embedded data.",
:data {:values [{:a "A", :b 28}
{:a "B", :b 55}
{:a "C", :b 43}
{:a "D", :b 91}
{:a "E", :b 81}
{:a "F", :b 53}
{:a "G", :b 19}
{:a "H", :b 87}
{:a "I", :b 52}]},
:mark "bar",
:encoding {:x {:field "a", :type "ordinal"}, :y {:field "b", :type "quantitative"}}}}
Literate ClojureScript app is a (DataScript) database of Widgets. Everything you see in the app is stored in DataScript, and it's a Widget entity. No special case.
Every Widget entity has an attribute to describe its type (I'm sorry for the overloaded use of this word.), and for each known type, there is a Widget renderer.
Add literate
and literate.client
aliases to you user or project deps.edn
:
{:aliases
{:literate
{:replace-paths []
:replace-deps {pedrorgirardi/literate {:git/url "https://github.com/pedrorgirardi/literate"
:sha "29d9263d6ffa6805873033c5e1405e5ebdde3081"}}
:main-opts ["-m" "literate.core" "--port" "8118"]}
:literate.client
{:extra-deps {pedrorgirardi/literate.client {:git/url "https://github.com/pedrorgirardi/literate.client"
:sha "0136b091c621f261d038c28ab5451d1073464a46"}}}}}
You can add the literate.client
library to an existing alias if you prefer, e.g. dev,
but I recommend setting
a literate
alias as shown above so you can start Literate on a separate process and isolate from your
project's classpath.
clojure -M:literate
You should see the output:
Welcome to Literate
Starting server...
Up and running on port: 8118
Happy coding!
All set and ready to go:
(require '[literate.client.core :as literate])
(def l (partial literate/view {:url "http://localhost:8118"}))
(l (literate/vega-lite {"$schema" "https://vega.github.io/schema/vega-lite/v4.json"
:description "A simple bar chart with embedded data."
:data {:values
[{:a "A" :b 28}
{:a "B" :b 55}
{:a "C" :b 43}
{:a "D" :b 91}
{:a "E" :b 81}
{:a "F" :b 53}
{:a "G" :b 19}
{:a "H" :b 87}
{:a "I" :b 52}]}
:mark "bar"
:encoding {:x {:field "a"
:type "ordinal"}
:y {:field "b"
:type "quantitative"}}}))
(l (literate/hiccup [:span "Hiccup Snippet"]))
npm install
npm run watch
(require '[literate.server :as server])
(server/run-server {:port 8090})