Skip to content

Commit

Permalink
Refactored highlightjs and mathjax into a separate plugins.
Browse files Browse the repository at this point in the history
  • Loading branch information
svetlyak40wt committed May 13, 2024
1 parent 4ebdb55 commit 99e630c
Show file tree
Hide file tree
Showing 10 changed files with 305 additions and 60 deletions.
69 changes: 66 additions & 3 deletions full/highlight.lisp → full/plugins/highlightjs.lisp
Original file line number Diff line number Diff line change
@@ -1,5 +1,12 @@
(uiop:define-package #:40ants-doc-full/highlight
(uiop:define-package #:40ants-doc-full/plugins/highlightjs
(:use #:cl)
(:import-from #:40ants-doc-full/themes/api
#:copy-static
#:inject-into-page-header)
(:import-from #:spinneret
#:with-html-string)
(:import-from #:40ants-doc-full/utils
#:make-relative-path)
(:import-from #:dexador)
(:import-from #:log)
(:import-from #:alexandria
Expand All @@ -14,9 +21,59 @@
;; (:import-from #:trivial-extract
;; #:extract-zip)
(:import-from #:which)
(:import-from #:jonathan))
(:import-from #:jonathan)
(:export #:highlightjs))
(in-package #:40ants-doc-full/plugins/highlightjs)

(in-package #:40ants-doc-full/highlight)

(defvar *default-languages*
'("lisp" "bash" "css" "json" "yaml" "plaintext" "xml" "markdown"))


(defvar *default-theme*
"atom-one-dark")


(defclass highlightjs ()
((languages :initform *default-languages*
:initarg :languages
:reader highlight-languages)
(theme :initform *default-theme*
:initarg :theme
:reader highlight-theme))
(:documentation "Injects a necessary scripts to use Highlightjs for rendering math formulas in your documentation."))


(defun highlightjs (&key (languages *default-languages*)
(theme *default-theme*))
"Creates a Highlightjs plugin.
You can redefine languages list and color theme like this:
```
(make-instance '40ants-doc-full/themes/light:light-theme
:plugins (list
(highlightjs :theme \"magula\"
:languages '(\"lisp\" \"python\"))))
```
"
(make-instance 'highlightjs
:languages languages
:theme theme))


(defmethod inject-into-page-header ((plugin highlightjs) uri)
(with-html-string
(let ((highlight-css-uri (make-relative-path uri "highlight.min.css"))
(highlight-js-uri (make-relative-path uri "highlight.min.js")))
(:link :rel "stylesheet"
:type "text/css"
:href highlight-css-uri)
(:script :type "text/javascript"
:src highlight-js-uri)
(:script :type "text/javascript"
"hljs.highlightAll();"))))


(defvar *supported-languages*
Expand Down Expand Up @@ -311,3 +368,9 @@
(write-string-into-file metadata metadata-path
:if-exists :supersede))))))
(values))


(defmethod copy-static ((plugin highlightjs) target-dir)
(download-highlight-js (highlight-languages plugin)
:to target-dir
:theme (highlight-theme plugin)))
46 changes: 46 additions & 0 deletions full/plugins/mathjax.lisp
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
(uiop:define-package #:40ants-doc-full/plugins/mathjax
(:use #:cl)
(:import-from #:40ants-doc-full/themes/api
#:inject-after-content)
(:import-from #:spinneret
#:with-html-string)
(:export #:mathjax))
(in-package #:40ants-doc-full/plugins/mathjax)


(defclass mathjax ()
()
(:documentation "Injects a necessary scripts to use MathJax for rendering math formulas in your documentation."))


(defun mathjax ()
"Creates a MathJax plugin."
(make-instance 'mathjax))


(defmethod inject-after-content ((plugin mathjax) uri)
(with-html-string
;; MathJax configuration to display inline formulas
(:script :type "text/javascript"
;; Here we need this :RAW
;; because of the bug in the Spinneret
;; https://github.com/ruricolist/spinneret/issues/59
(:raw "
MathJax = {
options: {
ignoreHtmlClass: 'page',
processHtmlClass: 'content'
},
tex: {
inlineMath: [['$','$']]
},
svg: {
fontCache: 'global'
}
};
"))
(:script :type "text/javascript"
:src "https://polyfill.io/v3/polyfill.min.js?features=es6")
(:script :type "text/javascript"
:async t
:src "https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-svg.js")))
86 changes: 73 additions & 13 deletions full/themes/api.lisp
Original file line number Diff line number Diff line change
@@ -1,7 +1,5 @@
(uiop:define-package #:40ants-doc-full/themes/api
(:use #:cl)
(:import-from #:40ants-doc-full/highlight
#:download-highlight-js)
(:import-from #:alexandria
#:once-only)
(:export #:render-css
Expand All @@ -17,7 +15,12 @@
#:render-page-header
#:render-page-footer
#:highlight-languages
#:highlight-theme))
#:highlight-theme
#:inject-into-page-header
#:theme-plugins
#:inject-before-content
#:inject-after-content
#:copy-static))
(in-package #:40ants-doc-full/themes/api)

(defvar *theme*)
Expand All @@ -29,19 +32,43 @@
(t ,theme))))
,@body)))


(defgeneric theme-plugins (theme)
(:documentation "Returns a list of plugin objects which will be used to inject additional information into the pages.")
(:method ((theme t))
nil))


(defgeneric highlight-languages (theme)
(:documentation "Returns a list of languages to highlight in snippets. Each language should be supported by Highlight.js.")
(:documentation "Returns a list of languages to highlight in snippets. Each language should be supported by Highlight.js.
**Deprecated!** will be removed after 2024-11-13.
Pass languages and highlight theme as arguments to highlighjs plugin.")
(:method ((theme t))
(list :lisp
:bash)))

(defgeneric highlight-theme (theme)
(:documentation "Returns a string with the name of the Highlight.js color theme for highlighted snippets.
To preview themes, use this site: <https://highlightjs.org/static/demo/>")
To preview themes, use this site: <https://highlightjs.org/static/demo/>
**Deprecated!** Will be removed after 2024-11-13.
Pass languages and highlight theme as arguments to highlighjs plugin.")
(:method ((theme t))
"magula"))


(defgeneric copy-static (plugin target-dir)
(:documentation "Define a method for this function if your plugin need to some static assets to work.
TARGET-DIR argument is an absolute directory pathname pointing to the root of the site.
By default does nothing.")
(:method ((theme t) (target-dir t))
(values)))


(defgeneric render-css (theme)
(:documentation "Returns a string with CSS."))

Expand All @@ -51,6 +78,36 @@
(defgeneric render-page-header (theme uri title)
(:documentation "Renders whole page header. Does nothing by default."))

(defgeneric inject-into-page-header (plugin uri)
(:documentation "Plugins can define a method for this generic-function to add some code to the end of a page header.
Each method should return a string which will be inserted without \"escaping\" so
the plugin's responsibility to escape all user input's if necessary.
Does nothing by default.")
(:method ((plugin t) uri)
nil))

(defgeneric inject-before-content (plugin uri)
(:documentation "Plugins can define a method for this generic-function to add some HTML before the main content of the page.
Each method should return a string which will be inserted without \"escaping\" so
the plugin's responsibility to escape all user input's if necessary.
Does nothing by default.")
(:method ((plugin t) uri)
nil))

(defgeneric inject-after-content (plugin uri)
(:documentation "Plugins can define a method for this generic-function to add some HTML after the main content of the page.
Each method should return a string which will be inserted without \"escaping\" so
the plugin's responsibility to escape all user input's if necessary.
Does nothing by default.")
(:method ((plugin t) uri)
nil))

(defgeneric render-page-footer (theme uri)
(:documentation "Renders whole page footer. Does nothing by default."))

Expand Down Expand Up @@ -91,14 +148,13 @@
(write-string (render-css *theme*)
stream)
(terpri stream))

(when (or highlight-languages
highlight-theme)
(warn "Deprecated, will be removed after 2024-11-13.
Pass languages and highlight theme as arguments to highlighjs plugin of the theme ~A"
*theme*))

(download-highlight-js (or highlight-languages
(highlight-languages *theme*))
:to absolute-dir
:theme (or highlight-theme
(highlight-theme *theme*)))

;; TODO: Probably let to override these files too
(loop with paths = '(("toc.js" "toc.js")
("search/searchtools.js" "searchtools.js")
("search/language_data.js" "language_data.js")
Expand All @@ -109,7 +165,11 @@
do (uiop:copy-file (asdf:system-relative-pathname :40ants-doc
(concatenate 'string
"static/" from))
(uiop:merge-pathnames* to absolute-dir)))))
(uiop:merge-pathnames* to absolute-dir)))

;; Plugins can override or remove some of the files written above:
(loop for plugin in (theme-plugins *theme*)
do (copy-static plugin absolute-dir))))

(defun call-with-page-template (func uri title toc)
(check-type uri string)
Expand Down
3 changes: 0 additions & 3 deletions full/themes/dark.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,3 @@
:background ,font-color)
((.toc-active > a)
:color ,background)))))))

(defmethod 40ants-doc-full/themes/api:highlight-theme ((theme dark-theme))
"atom-one-dark")
60 changes: 27 additions & 33 deletions full/themes/default.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@
(:use #:cl)
(:import-from #:common-html.emitter)
(:import-from #:40ants-doc-full/themes/api
#:inject-after-content
#:inject-before-content
#:inject-into-page-header
#:theme-plugins
#:render-css)
(:import-from #:lass)
(:import-from #:40ants-doc-full/commondoc/html
Expand All @@ -12,12 +16,25 @@
(:import-from #:40ants-doc-full/commondoc/changelog)
(:import-from #:alexandria
#:read-file-into-string)
(:import-from #:40ants-doc-full/plugins/highlightjs
#:highlightjs)
(:export #:default-theme))
(in-package #:40ants-doc-full/themes/default)


(defun %inject (theme func uri)
(loop for plugin in (theme-plugins theme)
for html = (funcall func plugin uri)
when html
do (write-string html spinneret:*html*)))


(defclass default-theme ()
())
((plugins :initarg :plugins
:initform nil
:reader theme-plugins))
(:default-initargs
:plugins (list (highlightjs))))


(defmethod render-css ((theme default-theme))
Expand Down Expand Up @@ -304,11 +321,6 @@
(.unresolved-reference
:color magenta))))

(defmethod 40ants-doc-full/themes/api:highlight-languages ((theme default-theme))
'("lisp" "bash" "css" "json" "yaml" "plaintext" "xml" "markdown"))

(defmethod 40ants-doc-full/themes/api:highlight-theme ((theme default-theme))
"atom-one-dark")

(defmethod 40ants-doc-full/themes/api:render-page ((theme default-theme) uri title
&key toc content)
Expand All @@ -333,8 +345,6 @@

(defmethod 40ants-doc-full/themes/api:render-html-head ((theme default-theme) uri title)
(let ((theme-uri (make-relative-path uri "theme.css"))
(highlight-css-uri (make-relative-path uri "highlight.min.css"))
(highlight-js-uri (make-relative-path uri "highlight.min.js"))
(jquery-uri (make-relative-path uri "jquery.js"))
(toc-js-uri (make-relative-path uri "toc.js"))
(rss-url (40ants-doc-full/commondoc/changelog::get-changelog-rss-url)))
Expand All @@ -353,44 +363,28 @@
:src jquery-uri)
(:script :type "text/javascript"
:src toc-js-uri)
(:link :rel "stylesheet"
:type "text/css"
:href highlight-css-uri)
(:script :type "text/javascript"
:src highlight-js-uri)
(:script :type "text/javascript"
"hljs.highlightAll();")
;; MathJax configuration to display inline formulas
(:script :type "text/javascript"
;; Here we need this :RAW
;; because of the bug in the Spinneret
;; https://github.com/ruricolist/spinneret/issues/59
(:raw "
MathJax = {
tex: {
inlineMath: [['$','$']],
processEscapes: true
}
};
"))
(:script :type "text/javascript"
:src "https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-svg.js")

;; This hack is required, because :HAS CSS selector is not supported by
;; browsers yet: https://caniuse.com/css-has
(:script :type "text/javascript"
(:raw "$(document).ready(function() {$('a:has(img)').css('border-bottom', 'none')})")))))
(:raw "$(document).ready(function() {$('a:has(img)').css('border-bottom', 'none')})"))

(%inject theme #'inject-into-page-header uri))))


(defmethod 40ants-doc-full/themes/api:render-content ((theme default-theme) uri toc content-func)
(declare (ignore uri toc))
(with-html
(:div :class "content"
;; This role is required for Sphinx Doc's
;; Javascript code. It searches texts inside
;; the role[main] block
:role "main"
(when content-func
(funcall content-func)))))
(%inject theme #'inject-before-content uri)

(funcall content-func)

(%inject theme #'inject-after-content uri)))))


(defmethod 40ants-doc-full/themes/api:render-sidebar ((theme default-theme) uri toc)
Expand Down
Loading

0 comments on commit 99e630c

Please sign in to comment.