Skip to content

Commit d144ac5

Browse files
Implement nbShow and highlight.js support for nim and other langs' code in markdown (#179)
* add nbShow * add highlight.js * add nohighlight class to nbCodeSource partial * take back nim class * take back hljs * disable highlight.js for Show source * fix tests * update readme * add option to disable highlight.js with `nb.disableHighlightJs` * hide warnings in index and readme * Update docsrc/penguins.nim Co-authored-by: Pietro Peterlongo <[email protected]> * mention dynamic highlighting in readme --------- Co-authored-by: Pietro Peterlongo <[email protected]>
1 parent d72664c commit d144ac5

File tree

7 files changed

+43
-11
lines changed

7 files changed

+43
-11
lines changed

README.md

+10-4
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,7 @@ Currently most of the documentation on customization is given by the examples.
156156

157157
* `nbImage`: image command to show images (see `penguins.nim` example linked above)
158158
* `nbFile`: content (string or untyped) is saved to file (see example document [files](https://pietroppeter.github.io/nimib/files.html))
159+
* `nbShow`: show a variable that has a `toHtml` proc defined. For example to pretty print a dataframe.
159160
* `nbRawHtml`: called with string content, it will add the raw content to document (html backend)
160161
* `nbTextWithCode`: a variant of `nbText` that also reads nim source. See example of usage
161162
at the end of the source in `numerical.nim` linked above.
@@ -190,6 +191,14 @@ and [counter](https://pietroppeter.github.io/nimib/counters.html) for examples o
190191
In [caesar](https://pietroppeter.github.io/nimib/caesar.html) we have an example of a karax app
191192
that implements [Caesar cipher](https://en.wikipedia.org/wiki/Caesar_cipher).
192193

194+
### highlighting
195+
Code blocks produced by `nbCode` are statically highlighted, but code in markdown code blocks are dynamically highlighted using
196+
[highlightjs](https://highlightjs.org/). The dynamic highlighting can be disabled by running `nb.disableHighlightJs()`.
197+
The supported languages are the ones listed as "common" [here](https://highlightjs.org/download/) plus Nim, Julia and Latex.
198+
199+
Highlight styling classes are the same of [highlightjs](https://highlightjs.org/)
200+
and you can pick a different styling (`atom-one-light` is default for light mode, `androidstudio` is default for dark mode).
201+
193202
### latex
194203

195204
See [numerical](https://pietroppeter.github.io/nimib/numerical.html) for an example of latex usage.
@@ -235,7 +244,7 @@ All the options available can be seen by running any nimib file with option `nbH
235244
```nim
236245
import osproc
237246
withDir nb.srcDir:
238-
echo execProcess(&quot;nim r --verbosity:0 --hints:off hello --nbHelp&quot;)
247+
echo execProcess(&quot;nim r --verbosity:0 --hints:off --warnings:off hello --nbHelp&quot;)
239248
```
240249

241250

@@ -368,6 +377,3 @@ because I made a [package](https://github.com/pietroppeter/nimoji) for that and
368377
because [someone made it into an art form](https://github.com/oakes/vim_cubed#q--a)
369378
and they tell me [imitation is the sincerest form of flattery](https://www.goodreads.com/quotes/558084-imitation-is-the-sincerest-form-of-flattery-that-mediocrity-can)
370379

371-
372-
373-

docsrc/index.nim

+10-1
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,7 @@ Currently most of the documentation on customization is given by the examples.
134134
135135
* `nbImage`: image command to show images (see `penguins.nim` example linked above)
136136
* `nbFile`: content (string or untyped) is saved to file (see example document [files]({docs}/files.html))
137+
* `nbShow`: show a variable that has a `toHtml` proc defined. For example to pretty print a dataframe.
137138
* `nbRawHtml`: called with string content, it will add the raw content to document (html backend)
138139
* `nbTextWithCode`: a variant of `nbText` that also reads nim source. See example of usage
139140
at the end of the source in `numerical.nim` linked above.
@@ -168,6 +169,14 @@ and [counter]({docs}/counters.html) for examples of how to create widgets using
168169
In [caesar]({docs}/caesar.html) we have an example of a karax app
169170
that implements [Caesar cipher](https://en.wikipedia.org/wiki/Caesar_cipher).
170171
172+
### highlighting
173+
Code blocks produced by `nbCode` are statically highlighted, but code in markdown code blocks are dynamically highlighted using
174+
[highlightjs](https://highlightjs.org/). The dynamic highlighting can be disabled by running `nb.disableHighlightJs()`.
175+
The supported languages are the ones listed as "common" [here](https://highlightjs.org/download/) plus Nim, Julia and Latex.
176+
177+
Highlight styling classes are the same of [highlightjs](https://highlightjs.org/)
178+
and you can pick a different styling (`atom-one-light` is default for light mode, `androidstudio` is default for dark mode).
179+
171180
### latex
172181
173182
See [numerical]({docs}/numerical.html) for an example of latex usage.
@@ -212,7 +221,7 @@ All the options available can be seen by running any nimib file with option `nbH
212221
nbCode:
213222
import osproc
214223
withDir nb.srcDir:
215-
echo execProcess("nim r --verbosity:0 --hints:off hello --nbHelp")
224+
echo execProcess("nim r --verbosity:0 --hints:off --warnings:off hello --nbHelp")
216225

217226
let renderProcType = "type NbRenderProc = proc (doc: var NbDoc, blk: var NbBlock) {. nimcall .}"
218227

docsrc/penguins.nim

+4-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,10 @@ nbCode:
1717
let df = readCsv("data/penguins.csv")
1818
nbText: "let us see how it looks"
1919
nbCode:
20-
echo df # why so wide the first column?
20+
echo df
21+
nbText: "or even nicer using `nbShow(df.head(10))`:"
22+
nbShow(df.head(10))
23+
2124
nbText: """Note that among the 7 columns of the dataframe, the first 2 and the last one have datatype string.
2225
The remaining 4 are numeric but they have datatype object. Why?
2326

src/nimib.nim

+4-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ export types, blocks, docs, boost, sugar, jsutils
55
# types exports mustache, tables, paths
66

77
from nimib / themes import nil
8-
export themes.useLatex, themes.darkMode, themes.`title=`
8+
export themes.useLatex, themes.darkMode, themes.`title=`, themes.disableHighlightJs
99

1010
from nimib / renders import nil
1111

@@ -127,6 +127,9 @@ when moduleAvailable(nimpy):
127127
captureStdout(nb.blk.output):
128128
discard nbPythonBuiltins.exec(pythonStr)
129129

130+
template nbShow*(obj: untyped) =
131+
nbRawHtml(obj.toHtml())
132+
130133
template nbRawOutput*(content: string) {.deprecated: "Use nbRawHtml instead".} =
131134
nbRawHtml(content)
132135

src/nimib/renders.nim

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ proc useHtmlBackend*(doc: var NbDoc) =
1616
doc.partials["nbCode"] = """
1717
{{>nbCodeSource}}
1818
{{>nbCodeOutput}}"""
19-
doc.partials["nbCodeSource"] = "<pre><code class=\"nim hljs\">{{&codeHighlighted}}</code></pre>"
19+
doc.partials["nbCodeSource"] = "<pre><code class=\"nohighlight hljs nim\">{{&codeHighlighted}}</code></pre>"
2020
doc.partials["nbCodeOutput"] = """{{#output}}<pre class="nb-output"><samp>{{output}}</samp></pre>{{/output}}"""
2121
doc.partials["nimibCode"] = doc.partials["nbCode"]
2222
doc.partials["nbImage"] = """<figure>

src/nimib/themes.nim

+12-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,9 @@ const head* = """
2323
{{^no_normalize}}<link rel='stylesheet' href='https://unpkg.com/normalize.css'>{{/no_normalize}}
2424
{{{stylesheet}}}
2525
{{{highlight}}}
26+
{{^disableHighlightJs}}
27+
{{{highlightJs}}}
28+
{{/disableHighlightJs}}
2629
{{{nb_style}}}
2730
{{{latex}}}
2831
{{> head_other }}
@@ -43,6 +46,10 @@ const waterLight* = """<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/
4346
const waterDark* = """<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/kognise/water.css@latest/dist/dark.min.css">"""
4447
const atomOneLight* = """<link rel='stylesheet' href='https://cdn.jsdelivr.net/gh/pietroppeter/nimib/assets/atom-one-light.css'>"""
4548
const androidStudio* = """<link rel='stylesheet' href='https://cdn.jsdelivr.net/gh/pietroppeter/nimib/assets/androidstudio.css'>"""
49+
const highlightJsTags* = """
50+
<script src="https://cdn.jsdelivr.net/gh/pietroppeter/nimib@main/assets/highlight.min.js"></script>
51+
<script>hljs.highlightAll();</script>
52+
"""
4653
const latex* = """<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/katex.min.css" integrity="sha384-AfEj0r4/OFrOo5t7NnNe46zW/tFgW6x/bCJG8FqQCEo3+Aro6EYUG4+cU+KJWu/X" crossorigin="anonymous">
4754
<script defer src="https://cdn.jsdelivr.net/npm/[email protected]/dist/katex.min.js" integrity="sha384-g7c+Jr9ZivxKLnZTDUhnkOnsh30B4H0rpLUpJ4jAIKs4fnJI+sEnkvrMWph2EDg4" crossorigin="anonymous"></script>
4855
<script defer src="https://cdn.jsdelivr.net/npm/[email protected]/dist/contrib/auto-render.min.js" integrity="sha384-mll67QQFJfxn0IYznZYonOWZ644AWYC+Pt2cHqMaRhXVrursRwvLnLaebdGIlYNa" crossorigin="anonymous" onload="renderMathInElement(document.body,{delimiters:[{left: '$$', right: '$$', display: true},{left: '$', right: '$', display: false}]});"></script>"""
@@ -100,7 +107,7 @@ const footer* = """
100107
const madeWithNimib* = """<span class="nb-small">made with <a href="https://pietroppeter.github.io/nimib/">nimib 🐳</a></span>"""
101108
const showSourceButton* = """<button class="nb-small" id="show" onclick="toggleSourceDisplay()">Show Source</button>"""
102109
const sourceSection* = """<section id="source">
103-
<pre><code class="nim hljs">{{{source_highlighted}}}</code></pre>
110+
<pre><code class="nohighlight nim hljs">{{{source_highlighted}}}</code></pre>
104111
</section>"""
105112
const showSourceScript* = """<script>
106113
function toggleSourceDisplay() {
@@ -131,6 +138,7 @@ proc useDefault*(doc: var NbDoc) =
131138
doc.context["favicon"] = faviconWhale
132139
doc.context["stylesheet"] = waterLight
133140
doc.context["highlight"] = atomOneLight
141+
doc.context["highlightJs"] = highlightJsTags
134142
doc.context["nb_style"] = nbStyle
135143
# header
136144
doc.partials["header"] = header
@@ -157,6 +165,9 @@ proc darkMode*(doc: var NbDoc) =
157165
proc useLatex*(doc: var NbDoc) =
158166
doc.context["latex"] = latex
159167

168+
proc disableHighlightJs*(doc: var NbDoc) =
169+
doc.context["disableHighlightJs"] = true
170+
160171
proc `title=`*(doc: var NbDoc, text: string) =
161172
# to deprecate?
162173
doc.context["title"] = text

tests/trenders.nim

+2-2
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,9 @@ suite "render (block), html default backend":
1111

1212
test "nbCode without output":
1313
nbCode: discard
14-
check nb.render(nb.blk).strip == """<pre><code class="nim hljs"><span class="hljs-keyword">discard</span></code></pre>"""
14+
check nb.render(nb.blk).strip == """<pre><code class="nohighlight hljs nim"><span class="hljs-keyword">discard</span></code></pre>"""
1515

1616
test "nbCode with output":
1717
nbCode: echo "hi"
1818
check nb.render(nb.blk).strip == """
19-
<pre><code class="nim hljs"><span class="hljs-keyword">echo</span> <span class="hljs-string">&quot;hi&quot;</span></code></pre><pre class="nb-output"><samp>hi</samp></pre>"""
19+
<pre><code class="nohighlight hljs nim"><span class="hljs-keyword">echo</span> <span class="hljs-string">&quot;hi&quot;</span></code></pre><pre class="nb-output"><samp>hi</samp></pre>"""

0 commit comments

Comments
 (0)