This project provides a reader for a clojure dialect similar to the dialect of the Racket language used in Pollen.
Deps coords:
#:fr.jeremyschoffen{textp-reader-alpha #:mvn{:version "0"}}
Lein coords:
[fr.jeremyschoffen/textp-reader-alpha "0"]
The idea is to have a dialect of clojure that inverses the priority of code over text in a file. Text is primary, code is secondary and introduced with special syntactic constructs.
Since the reader considers everything plain text unless specified otherwise, we could say that the
reader is in text mode by default. In this mode the character ◊
is the only special character.
It announces that a special syntactic construct is going to be used, putting the reader in clojure mode.
The first kind of syntax is embedded clojure:
Clojure code (list expressions) can be embedded inside the text:
◊(def x "some value")◊
A symbol can be embeded this way ◊|x|◊. It's basically sugar for ◊(do x)◊
reads as:
["Clojure code (list expressions) can be embedded inside the text:\n"
(def x "some value")
"\nA symbol can be embeded this way "
x
". It's basically sugar for "
(do x)]
We can have ◊!verbatim!◊ and ◊/commented/◊ text
reads as:
["We can have " "verbatim" " and " "" " text"]
Like in Pollen we have a tag construct. We use tags similarly to what is found in Pollen and Scribble. A tag starts with the '◊' character followed by the tag's name and then the any number of tag arguments.
There are 2 types of arguments:
For instance we could have html links.
◊html/a[:href "www.url.com"]{link}.
reads as:
[(html/a
{:tag :tag-args-clj, :content [:href "www.url.com"]}
{:tag :tag-args-txt, :content ["link"]})
"."]
The way the reader is implemented allows for recursion inside syntactic constructs. When in text mode we can use embedded code and tags. In clojure mode we can use tags giving us access again to text mode via text arguments.
This is valid for the reader.
◊(defn template [x]
◊div [:class ◊text{class1 class2} ]
{
Some text and crazy recursion: ◊(str (+ 2 x) ◊text{times.})◊
})◊
reads as:
[(defn
template
[x]
(div
{:tag :tag-args-clj,
:content
[:class (text {:tag :tag-args-txt, :content ["class1 class2"]})]}
{:tag :tag-args-txt,
:content
["\n Some text and crazy recursion: "
(str (+ 2 x) (text {:tag :tag-args-txt, :content ["times."]}))
"\n "]}))]
Assuming you have a div
and a text
function that can deal with this data, you can eval that
in a repl. I personnaly find that pretty neat...
Note that the previous example shows that we can have any manner of spaces between tag arguments.
This creates the case where one might want to use a opening bracket directly after a tag. This bracket might be considered as the beginning of clojure arguments to the tag.
There are several corner case like this one. We then allow for escaped characters in the syntax.
◊br \[
the [ character won't count as the opening of clojure arguments. Also \◊ doesn't start any syntactic rule.
reads as:
[(br)
" "
"["
"\n\nthe [ character won't count as the opening of clojure arguments. Also "
"◊"
" doesn't start any syntactic rule."]
Copyright © 2020 Jeremy Schoffen.
Distributed under the Eclipse Public License 2.0.
Can you improve this documentation?Edit on GitHub
cljdoc is a website building & hosting documentation for Clojure/Script libraries
× close