texlive[55347] Master/texmf-dist: nodetree (30may20)

commits+karl at tug.org commits+karl at tug.org
Sat May 30 22:59:30 CEST 2020


Revision: 55347
          http://tug.org/svn/texlive?view=revision&revision=55347
Author:   karl
Date:     2020-05-30 22:59:30 +0200 (Sat, 30 May 2020)
Log Message:
-----------
nodetree (30may20)

Modified Paths:
--------------
    trunk/Master/texmf-dist/doc/luatex/nodetree/README.md
    trunk/Master/texmf-dist/doc/luatex/nodetree/nodetree.pdf
    trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.dtx
    trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.ins
    trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.lua
    trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.sty
    trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.tex

Added Paths:
-----------
    trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree-embed.sty

Modified: trunk/Master/texmf-dist/doc/luatex/nodetree/README.md
===================================================================
--- trunk/Master/texmf-dist/doc/luatex/nodetree/README.md	2020-05-30 20:58:42 UTC (rev 55346)
+++ trunk/Master/texmf-dist/doc/luatex/nodetree/README.md	2020-05-30 20:59:30 UTC (rev 55347)
@@ -1,4 +1,3 @@
-
 ![nodetree](https://raw.githubusercontent.com/Josef-Friedrich/nodetree/master/graphics/packagename.png)
 
 # Abstract
@@ -19,7 +18,7 @@
 
 # License
 
-Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
+Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
 ------------------------------------------------------------------------
 This work may be distributed and/or modified under the conditions of
 the LaTeX Project Public License, either version 1.3 of this license
@@ -102,3 +101,63 @@
 
 ![nodetree](https://raw.githubusercontent.com/Josef-Friedrich/nodetree/master/graphics/ligatures.png)
 
+# Development
+
+First delete the stable version installed by TeX Live. Because the
+package `nodetree` belongs to the collection `collection-latexextra`, the
+option  `--force` must be used to delete the package.
+
+    tlmgr remove --force nodetree
+
+## Deploying a new version
+
+Update the version number in the file `nodetree.dtx` on this locations:
+
+### In the markup for the file `nodetree.sty` (approximately at the line number 30)
+
+    %<*package>
+    [2016/07/18 v1.2 Visualize node lists in a tree view]
+    %<*package>
+
+### In the markup for the file `nodetree-embed.sty` (approximately at the line number 220)
+
+    %<*package>
+    [2016/07/18 v1.2 Visualize node lists in a tree view]
+    %<*package>
+
+### In the markup for the package documentation (approximately at the line number 50)
+
+Add a changes entry:
+
+```latex
+\changes{v1.2}{2020/05/20}{...}
+```
+
+### In documentation (documentation.tex) (approximately at the line number 70)
+
+```latex
+\date{v2.0 from 2020/05/29}
+```
+
+### In the file `nodetree.lua` (approximately at the line number 20)
+
+```lua
+if not modules then modules = { } end modules ['nodetree'] = {
+  version   = '1.2'
+}
+```
+
+### Update the copyright year:
+
+```
+sed -i 's/(C) 2016-2020/(C) 2016-2021/g' nodetree.ins
+sed -i 's/(C) 2016-2020/(C) 2016-2021/g' nodetree.dtx
+```
+
+### Command line tasks:
+
+```
+git tag -a v1.4
+make
+make ctan
+```

Modified: trunk/Master/texmf-dist/doc/luatex/nodetree/nodetree.pdf
===================================================================
(Binary files differ)

Modified: trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.dtx
===================================================================
--- trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.dtx	2020-05-30 20:58:42 UTC (rev 55346)
+++ trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.dtx	2020-05-30 20:59:30 UTC (rev 55347)
@@ -1,6 +1,6 @@
 % \iffalse meta-comment
 %
-% Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
+% Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
 % ----------------------------------------------------------------------
 % This work may be distributed and/or modified under the conditions of
 % the LaTeX Project Public License, either version 1.3 of this license
@@ -28,648 +28,8 @@
 %<package>\NeedsTeXFormat{LaTeX2e}[1999/12/01]
 %<package>\ProvidesPackage{nodetree}
 %<*package>
-    [2016/07/18 v1.2 Visualize node lists in a tree view]
+    [2020/05/29 v2.0 Visualize node lists in a tree view]
 %</package>
-%<*driver>
-\documentclass{ltxdoc}
-\usepackage{paralist,fontspec,graphicx,fancyvrb}
-\usepackage[
-  colorlinks=true,
-  linkcolor=red,
-  filecolor=red,
-  urlcolor=red,
-]{hyperref}
-%\usepackage{nodetree}
-\EnableCrossrefs
-\CodelineIndex
-\RecordChanges
-
-\setmonofont{DejaVu Sans Mono}
-
-\def\nodetreelua#1{\texttt{\scantokens{\catcode`\_=12\relax#1}}}
-
-\def\secref#1{(\rightarrow\ \ref{#1})}
-
-\newcommand{\tmpgraphics}[1]{
-  \noindent
-  \includegraphics[scale=0.4]{graphics/#1}
-}
-
-\DefineVerbatimEnvironment{code}{Verbatim}
-{
-  frame=single,
-  fontsize=\footnotesize,
-}
-
-\begin{document}
-
-\providecommand*{\url}{\texttt}
-\GetFileInfo{nodetree.dtx}
-\title{The \textsf{nodetree} package}
-\author{%
-  Josef Friedrich\\%
-  \url{josef at friedrich.rocks}\\%
-  \href{https://github.com/Josef-Friedrich/nodetree}{github.com/Josef-Friedrich/nodetree}%
-}
-\date{\fileversion~from \filedate}
-
-\maketitle
-
-\noindent
-\includegraphics[width=\linewidth]{graphics/packagename}
-
-\newpage
-
-\tableofcontents
-
-\newpage
-
-%-----------------------------------------------------------------------
-% Abstract
-%-----------------------------------------------------------------------
-
-\section{Abstract}
-
-|nodetree| is a development package that visualizes the structure of
-node lists. |nodetree| shows its debug informations in the consoles’
-output when you compile a Lua\TeX{} file. It uses a similar visual
-representation for node lists as the UNIX |tree| command uses for a
-folder structure.
-
-Node lists are the main building blocks of each document generated by
-the \TeX{} engine \emph{Lua\TeX}. The package |nodetree| doesn‘t change
-the rendered document. The tree view can only be seen when using a
-terminal to generate the document.
-
-|nodetree| is inspired by a
-\href{https://gist.github.com/pgundlach/556247}
-{gist from Patrick Gundlach}.
-
-%-----------------------------------------------------------------------
-% Usage
-%-----------------------------------------------------------------------
-
-\section{Usage}
-
-The package |nodetree| can be used both with Lua\TeX{} and Lua\LaTeX{}.
-You have to use both engines in a text console. Run for example
-|luatex luatex-test.tex| to list the nodes using Lua\TeX{}.
-
-\begin{code}
-\input{nodetree.tex}
-\nodetreeregister{postline}
-
-Lorem ipsum dolor.
-\bye
-\end{code}
-
-Or run |lualatex lualatex-test.tex| to show a node tree using
-Lua\LaTeX{}. In Lua\LaTeX{} you can omit |\nodetreeregister{postline}|.
-|\usepackage{nodetree}| registers automatically the
-|post_linebreak_filter|. If you don’t want debug the
-|post_linebreak_filter| use |\nodetreeunregister{postline}|.
-
-\begin{code}
-\documentclass{article}
-\usepackage{nodetree}
-
-\begin{document}
-Lorem ipsum dolor.
-\end{document}
-\end{code}
-
-%%
-% inside Lua code
-%%
-
-\subsection{Debug nodes inside Lua code}
-
-Use the Lua function |nodetree.analyze(head)| to debug nodes inside your
-Lua code. The following code snippet demonstrates the usage in Lua\TeX{}.
-|head| is the current node.
-
-\begin{code}
-\input{nodetree.tex}
-
-\directlua{
-  local test = function (head)
-    nodetree.analyze(head)
-  end
-  callback.register('post_linebreak_filter', test)
-}
-
-Lorem ipsum dolor.
-\bye
-\end{code}
-
-This example illustrates how the function has to be applied in
-Lua\LaTeX{}.
-
-\begin{code}
-\documentclass{article}
-\usepackage{nodetree}
-
-\begin{document}
-
-\directlua{
-  local test = function (head)
-    nodetree.analyze(head)
-  end
-  luatexbase.add_to_callback('post_linebreak_filter', test, 'test')
-}
-
-Lorem ipsum dolor.
-\end{document}
-\end{code}
-
-%-----------------------------------------------------------------------
-% Macros
-%-----------------------------------------------------------------------
-
-\section{Macros}
-
-%%
-% \nodetreeregister
-%%
-
-\subsection{\cmd{\nodetreeregister}}
-
-\DescribeMacro{\nodetreeregister}
-\cmd{\nodetreeregister}\marg{callbacks}: The argument \marg{callbacks}
-takes a comma separated list of callback aliases as described in
-\secref{sec:option-callback}.
-
-%%
-% \nodetreeunregister
-%%
-
-\subsection{\cmd{\nodetreeunregister}}
-
-\DescribeMacro{\nodetreeunregister}
-\cmd{\nodetreeunregister}\marg{callbacks}: The argument \marg{callbacks}
-takes a comma separated list of callback aliases as described in
-\secref{sec:option-callback}.
-
-%%
-% \nodetreeoption
-%%
-
-\subsection{\cmd{\nodetreeoption}}
-
-\DescribeMacro{\nodetreeoption}
-\cmd{\nodetreeoption}\oarg{option}\marg{value}: \secref{sec:options}
-This macro sets the option \oarg{option} to the value \marg{value}.
-
-%%
-% \nodetreeset
-%%
-
-\subsection{\cmd{\nodetreeset}}
-
-\DescribeMacro{\nodetreeset}
-\cmd{\nodetreeset}\marg{kv-options}:
-This macro can only be used in Lua\LaTeX{}. \marg{kv-options} are key
-value pairs.
-
-\begin{code}
-\nodetreeset{color=no,callbacks={hpack,vpack},verbosity=2}
-\end{code}
-
-%-----------------------------------------------------------------------
-% Options
-%-----------------------------------------------------------------------
-
-\section{Options}
-\label{sec:options}
-
-%%
-% callback
-%%
-
-\subsection{Option \texttt{callback}}
-\label{sec:option-callback}
-
-The option |callback| is the most important setting of the package. You
-have to specify one alias to select the |callback|. Because of the
-underscores the callback name contains it can not set by its technical
-name (\rightarrow{} Figure \ref{fig:callback}).
-
-This macros process callback options:
-\cmd{\nodetreeregister}\marg{callbacks},
-\cmd{\nodetreeunregister}\marg{callbacks},
-\cmd{\nodetreeset}\marg{callback=<callbacks>} and
-\cmd{\usepackage}\oarg{callback=<callbacks>}\marg{nodetree}.
-
-Use commas to specify mulitple callbacks. Avoid using whitespaces:
-
-\begin{code}
-\nodetreeregister{preline,line,postline}
-\end{code}
-
-Wrap your callback aliases in curly braces for the macro |\nodetreeset|:
-
-\begin{code}
-\nodetreeset{callback={preline,line,postline}}
-\end{code}
-
-The same applies for the macro |\usepackage|:
-
-\begin{code}
-\usepackage{callback={preline,line,postline}}
-\end{code}
-
-%%
-% Tabular callbacks
-%%
-
-\newcommand{\nodetreecallback}[3]{
-  \nodetreelua{#1} & \nodetreelua{#2} & \nodetreelua{#3} \\
-}
-
-\begin{figure}
-
-\noindent
-\begin{tabular}{lll}
-\textbf{Alias (short)} & \textbf{Alias (longer)} & \textbf{Callback} \\
-\nodetreecallback{contribute}{contributefilter}{contribute_filter}
-\nodetreecallback{buildpage}{buildpagefilter}{buildpage_filter}
-\nodetreecallback{preline}{prelinebreakfilter}{pre_linebreak_filter}
-\nodetreecallback{line}{linebreakfilter}{linebreak_filter}
-\nodetreecallback{append}{appendtovlistfilter}{append_to_vlist_filter}
-\nodetreecallback{postline}{postlinebreakfilter}{post_linebreak_filter}
-\nodetreecallback{hpack}{hpackfilter}{hpack_filter}
-\nodetreecallback{vpack}{vpackfilter}{vpack_filter}
-\nodetreecallback{hpackq}{hpackquality}{hpack_quality}
-\nodetreecallback{vpackq}{vpackquality}{vpack_quality}
-\nodetreecallback{process}{processrule}{process_rule}
-\nodetreecallback{preout}{preoutputfilter}{pre_output_filter}
-\nodetreecallback{hyph}{hyphenate}{hyphenate}
-\nodetreecallback{liga}{ligaturing}{ligaturing}
-\nodetreecallback{kern}{kerning}{kerning}
-\nodetreecallback{insert}{insertlocalpar}{insert_local_par}
-\nodetreecallback{mhlist}{mlisttohlist}{mlist_to_hlist}
-\end{tabular}
-
-\caption{The callback aliases}
-\label{fig:callback}
-\end{figure}
-
-%%
-% verbosity
-%%
-
-\subsection{Option \texttt{verbosity}}
-
-Higher integer values result in a more verbose output. The default value
-for this options is |1|. At the moment only verbosity level |2| is
-implemented.
-
-%%
-% color
-%%
-
-\subsection{Option \texttt{color}}
-
-The default option for |color| is |colored|. Use any other string (for
-example |none| or |no|) to disable the colored terminal output of the
-package.
-
-\begin{code}
-\usepackage[color=no]{nodetree}
-\end{code}
-
-%%
-% unit
-%%
-
-\subsection{Option \texttt{unit}}
-
-The option |unit| sets the length unit to display all length values of
-the nodes. The default option for |unit| is |pt|. See figure
-\ref{fig:fixed-units} and \ref{fig:relative-units} for possible values.
-
-\begin{figure}
-\begin{tabular}{lp{10cm}}
-\textbf{Unit} &
-\textbf{Description} \\
-
-pt &
-Point 1/72.27 inch. The conversion to metric units, to two decimal
-places, is 1 point = 2.85 mm = 28.45 cm. \\
-
-pc &
-Pica, 12 pt \\
-
-in &
-Inch, 72.27 pt \\
-
-bp &
-Big point, 1/72 inch. This length is the definition of a point in
-PostScript and many desktop publishing systems. \\
-
-cm &
-Centimeter \\
-
-mm &
-Millimeter \\
-
-dd &
-Didot point, 1.07 pt \\
-
-cc &
-Cicero, 12 dd \\
-
-sp &
-Scaled point, 1/65536 pt \\
-\end{tabular}
-\caption{Fixed units}
-\label{fig:fixed-units}
-\end{figure}
-
-\begin{figure}
-\begin{tabular}{lp{10cm}}
-\textbf{Unit} &
-\textbf{Description} \\
-
-ex &
-x-height of the current font \\
-
-em &
-Width of the capital letter M \\
-\end{tabular}
-\caption{Relative units}
-\label{fig:relative-units}
-\end{figure}
-
-%%
-% decimalplaces
-%%
-
-\subsection{Option \texttt{decimalplaces}}
-
-The options |decimalplaces| sets the number of decimal places for some
-node fields.
-
-\begin{code}
-\nodetreeoption[decimalplaces]{4}
-\end{code}
-
-gets
-
-\begin{code}
-├─GLYPH char: "a"; width: 5pt; height: 4.3055pt;
-\end{code}
-
-If |decimalplaces| is set to |0| only integer values are shown.
-
-\begin{code}
-├─GLYPH char: "a"; width: 5pt; height: 4pt;
-\end{code}
-
-%-----------------------------------------------------------------------
-% Visual tree structure
-%-----------------------------------------------------------------------
-
-\section{Visual tree structure}
-
-%%
-% Two different connections
-%%
-
-\subsection{Two different connections}
-
-Nodes in Lua\TeX{} are connected. The |nodetree| package distinguishs
-between the |list| and |field| connections.
-
-\begin{itemize}
- \item |list|: Nodes, which are double connected by |next| and
-       |previous| fields.
- \item |field|: Connections to nodes by other fields than |next| and
-       |previous| fields, e. g. |head|, |pre|.
-\end{itemize}
-
-%%
-% Unicode characters
-%%
-
-\subsection{Unicode characters to show the tree view}
-
-\renewcommand{\arraystretch}{1.5}
-
-The package |nodetree| uses the unicode box drawing symbols. Your
-default terminal font should contain this characters to obtain the tree
-view. Eight box drawing characters are necessary.
-
-\noindent
-\begin{tabular}{lcl}
-\textbf{Code} & \textbf{Character} & \textbf{Name} \\
-U+2500 & |─| & BOX DRAWINGS LIGHT HORIZONTAL \\
-U+2502 & |│| & BOX DRAWINGS LIGHT VERTICAL \\
-U+2514 & |└| & BOX DRAWINGS LIGHT UP AND RIGHT \\
-U+251C & |├| & BOX DRAWINGS LIGHT VERTICAL AND RIGHT \\
-U+2550 & |═| & BOX DRAWINGS DOUBLE HORIZONTAL \\
-U+2551 & |║| & BOX DRAWINGS DOUBLE VERTICAL \\
-U+255A & |╚| & BOX DRAWINGS DOUBLE UP AND RIGHT \\
-U+2560 & |╠| & BOX DRAWINGS DOUBLE VERTICAL AND RIGHT \\
-\end{tabular}
-
-For |list| connections \emph{light} characters are shown.
-
-\begin{code}
-│ │
-│ ├─list1
-│ └─list2
-└─list3
-\end{code}
-
-|field| connections are visialized by \emph{Double} characters.
-
-\begin{code}
-║ ║
-║ ╠═field1
-║ ╚═field2
-╚═field3
-\end{code}
-
-%-----------------------------------------------------------------------
-% Examples
-%-----------------------------------------------------------------------
-
-\newpage
-
-\section{Examples}
-
-%%
-% packagename
-%%
-
-\subsection{The node list of the package name}
-
-\begin{code}
-\documentclass{article}
-\usepackage{nodetree}
-\begin{document}
-nodetree
-\end{document}
-\end{code}
-
-\tmpgraphics{packagename}
-
-%%
-% math
-%%
-
-\newpage
-
-\subsection{The node list of a mathematical formula}
-
-\begin{code}
-\documentclass{article}
-\usepackage[callback={mhlist}]{nodetree}
-\begin{document}
-\[\left(a\right)\left[\frac{b}{a}\right]=a\,\]
-\end{document}
-\end{code}
-
-\tmpgraphics{math}
-
-%%
-% ligatures
-%%
-
-\newpage
-
-\subsection{The node list of the word \emph{Office}}
-
-The characters \emph{ffi} are deeply nested in a discretionary node.
-
-\begin{code}
-\documentclass{article}
-\usepackage{nodetree}
-\begin{document}
-Office
-\end{document}
-\end{code}
-
-\tmpgraphics{ligatures}
-
-%-----------------------------------------------------------------------
-% Index
-%-----------------------------------------------------------------------
-
-  \DocInput{nodetree.dtx}
-  \pagebreak
-  \PrintChanges
-  \pagebreak
-  \PrintIndex
-\end{document}
-%</driver>
-%<*readme>
-
-![nodetree](https://raw.githubusercontent.com/Josef-Friedrich/nodetree/master/graphics/packagename.png)
-
-# Abstract
-
-`nodetree` is a development package that visualizes the structure of
-node lists. `nodetree` shows its debug informations in the consoles’
-output when you compile a LuaTeX file. It uses a similar visual
-representation for node lists as the UNIX `tree` command uses for a
-folder structure.
-
-Node lists are the main building blocks of each document generated by
-the TeX engine LuaTeX. The package `nodetree` doesn‘t change
-the rendered document. The tree view can only be seen when using a
-terminal to generate the document.
-
-`nodetree` is inspired by a
-[gist from Patrick Gundlach](https://gist.github.com/pgundlach/556247).
-
-# License
-
-Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
-------------------------------------------------------------------------
-This work may be distributed and/or modified under the conditions of
-the LaTeX Project Public License, either version 1.3 of this license
-or (at your option) any later version.  The latest version of this
-license is in:
-
-  http://www.latex-project.org/lppl.txt
-
-and version 1.3 or later is part of all distributions of LaTeX
-version 2005/12/01 or later.
-
-# CTAN
-
-Since July 2016 the cloze package is included in the Comprehensive TeX
-Archive Network (CTAN).
-
-* TeX archive: http://mirror.ctan.org/tex-archive/macros/luatex/generic/nodetree
-* Package page: http://www.ctan.org/pkg/nodetree
-
-# Repository
-
-https://github.com/Josef-Friedrich/nodetree
-
-# Installation
-
-Get source:
-
-    git clone git at github.com:Josef-Friedrich/nodetree.git
-    cd nodetree
-
-Compile:
-
-    make
-
-or manually:
-
-    luatex nodetree.ins
-    lualatex nodetree.dtx
-    makeindex -s gglo.ist -o nodetree.gls nodetree.glo
-    makeindex -s gind.ist -o nodetree.ind nodetree.idx
-    lualatex nodetree.dtx
-
-# Examples
-
-## The node list of the package name
-
-```latex
-\documentclass{article}
-\usepackage{nodetree}
-\begin{document}
-nodetree
-\end{document}
-```
-
-![nodetree](graphics/packagename.png)
-
-## The node list of a mathematical formula
-
-```latex
-\documentclass{article}
-\usepackage[callback={mhlist}]{nodetree}
-\begin{document}
-\[\left(a\right)\left[\frac{b}{a}\right]=a\,\]
-\end{document}
-```
-
-![nodetree](https://raw.githubusercontent.com/Josef-Friedrich/nodetree/master/graphics/math.png)
-
-## The node list of the word 'Office'
-
-The characters 'ffi' are deeply nested in a discretionary node.
-
-```latex
-\documentclass{article}
-\usepackage{nodetree}
-\begin{document}
-Office
-\end{document}
-```
-
-![nodetree](https://raw.githubusercontent.com/Josef-Friedrich/nodetree/master/graphics/ligatures.png)
-
-%</readme>
 % \fi
 %
 % \CheckSum{0}
@@ -695,6 +55,16 @@
 % \changes{v1.0}{2016/07/07}{Inital release}
 % \changes{v1.1}{2016/07/13}{Fix the registration of same callbacks}
 % \changes{v1.2}{2016/07/18}{Fix difference between README.md in the upload and that from nodetree.dtx}
+% \changes{v2.0}{2020/05/29}{
+%  * Switch from lowercase macro names to PascalCase names for better readability.
+%  * The Lua code is no longer developed inside the DTX file, instead in a separate file named nodetree.lua.
+%  * Add a sub package named nodetree-embed.sty for embedding nodetree views into a \LaTeX{} document.
+%  * Add support for new node subtype names.
+%  * Add support for a new Lua\TeX{} node callback.
+%  * Add support for node properties.
+%  * Less verbose representation of node attributes.
+%  * Minor tree output adjustments.
+% }
 %
 % \DoNotIndex{\newcommand,\newenvironment,\def,\directlua}
 %
@@ -716,40 +86,68 @@
 %    \begin{macrocode}
 \directlua{
   nodetree = require('nodetree')
-  nodetree.set_option('engine', 'luatex')
-  nodetree.set_default_options()
 }
 %    \end{macrocode}
 %
-% \begin{macro}{\nodetreeoption}
+% \begin{macro}{\NodetreeSetOption}
 %    \begin{macrocode}
-\def\nodetreeoption[#1]#2{
+\def\NodetreeSetOption[#1]#2{
   \directlua{
     nodetree.set_option('#1', '#2')
   }
 }
+\let\nodetreeoption\NodetreeSetOption
 %    \end{macrocode}
 % \end{macro}
 %
-% \begin{macro}{\nodetreeregister}
+% \begin{macro}{\NodetreeResetOption}
 %    \begin{macrocode}
-\def\nodetreeregister#1{
+\def\NodetreeResetOption#1{
+  \NodetreeSetOption[#1]{%
+    \directlua{
+      tex.print(nodetree.get_default_option('#1'))
+    }%
+  }%
+}
+%    \end{macrocode}
+% \end{macro}
+%
+% \begin{macro}{\NodetreeReset}
+%    \begin{macrocode}
+\def\NodetreeReset{
+  \NodetreeResetOption{callback}
+  \NodetreeResetOption{channel}
+  \NodetreeResetOption{color}
+  \NodetreeResetOption{decimalplaces}
+  \NodetreeResetOption{engine}
+  \NodetreeResetOption{unit}
+  \NodetreeResetOption{verbosity}
+}
+\let\nodetreereset\NodetreeReset
+%    \end{macrocode}
+% \end{macro}
+%
+% \begin{macro}{\NodetreeRegisterCallback}
+%    \begin{macrocode}
+\def\NodetreeRegisterCallback#1{
   \directlua{
     nodetree.set_option('callback', '#1')
     nodetree.register_callbacks()
   }
 }
+\let\nodetreeregister\NodetreeRegisterCallback
 %    \end{macrocode}
 % \end{macro}
 %
-% \begin{macro}{\nodetreeunregister}
+% \begin{macro}{\NodetreeUnregisterCallback}
 %    \begin{macrocode}
-\def\nodetreeunregister#1{
+\def\NodetreeUnregisterCallback#1{
   \directlua{
     nodetree.set_option('callback', '#1')
     nodetree.unregister_callbacks()
   }
 }
+\let\nodetreeunregister\NodetreeUnregisterCallback
 %    \end{macrocode}
 % \end{macro}
 %
@@ -765,9 +163,6 @@
 %
 %    \begin{macrocode}
 \input{nodetree}
-\directlua{
-  nodetree.set_option('engine', 'lualatex')
-}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
@@ -777,1233 +172,303 @@
 %    \begin{macrocode}
 \SetupKeyvalOptions{
   family=NT,
-  prefix=NT@
+  prefix=NTK@
 }
 %    \end{macrocode}
 %
 %    \begin{macrocode}
 \DeclareStringOption[term]{channel}
-\define at key{NT}{channel}[]{\nodetreeoption[channel]{#1}}
+\define at key{NT}{channel}[]{\NodetreeSetOption[channel]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
 \DeclareStringOption[postlinebreak]{callback}
-\define at key{NT}{callback}[]{\nodetreeoption[callback]{#1}}
+\define at key{NT}{callback}[]{\NodetreeSetOption[callback]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
 \DeclareStringOption[1]{verbosity}
-\define at key{NT}{verbosity}[]{\nodetreeoption[verbosity]{#1}}
+\define at key{NT}{verbosity}[]{\NodetreeSetOption[verbosity]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
 \DeclareStringOption[colored]{color}
-\define at key{NT}{color}[]{\nodetreeoption[color]{#1}}
+\define at key{NT}{color}[]{\NodetreeSetOption[color]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
 \DeclareStringOption[1]{unit}
-\define at key{NT}{unit}[]{\nodetreeoption[unit]{#1}}
+\define at key{NT}{unit}[]{\NodetreeSetOption[unit]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
 \DeclareStringOption[1]{decimalplaces}
-\define at key{NT}{decimalplaces}[]{\nodetreeoption[decimalplaces]{#1}}
+\define at key{NT}{decimalplaces}[]{\NodetreeSetOption[decimalplaces]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-\ProcessKeyvalOptions*
+\ProcessKeyvalOptions{NT}
 \directlua{
-  nodetree.set_default_options()
   nodetree.register_callbacks()
 }
 %    \end{macrocode}
 %
-% \begin{macro}{\nodetreeset}
+% \begin{macro}{\NodetreeSet}
 %    \begin{macrocode}
-\newcommand{\nodetreeset}[1]{\setkeys{nodetree}{#1}}
+\newcommand{\NodetreeSet}[1]{%
+  \setkeys{NT}{#1}%
+}
+\let\nodetreeset\NodetreeSet
 %    \end{macrocode}
 % \end{macro}
 %
 % \iffalse
 %</package>
-%<*luamain>
+%<*packageembed>
 % \fi
-%
-% \makeatletter
-% \c at CodelineNo 0 \relax
-% \makeatother
-%
-% \subsection{The file \tt{nodetree.lua}}
-%
 %    \begin{macrocode}
-local nodex = {}
+\NeedsTeXFormat{LaTeX2e}[1994/06/01]
+\ProvidesPackage{nodetree-embed}
+  [2020/05/29 v2.0 Embed node trees into a LaTeX document]
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-local tpl = {}
+\RequirePackage{xcolor,mdframed,expl3,xparse,fontspec}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-local tree = {}
+\input{nodetree}
 %    \end{macrocode}
 %
-% Nodes in Lua\TeX{} are connected. The nodetree view distinguishs
-% between the |list| and |field| connections.
-%
-% \begin{itemize}
-%  \item |list|: Nodes, which are double connected by |next| and
-%        |previous| fields.
-%  \item |field|: Connections to nodes by other fields than |next| and
-%        |previous| fields, e. g. |head|, |pre|.
-% \end{itemize}
-%
-% The lua table named |tree.state| holds state values for the current
-% tree item.
-%
-% \begin{code}
-%  tree.state:
-%    - 1:
-%      - list: continue
-%      - field: stop
-%    - 2:
-%      - list: continue
-%      - field: stop
-% \end{code}
 %    \begin{macrocode}
-tree.state = {}
+\RequirePackage{kvoptions}
+\SetupKeyvalOptions{
+  family=NTE,
+  prefix=NTEK@
+}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-local callbacks = {}
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-local base = {}
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-local options = {}
-%    \end{macrocode}
-%
-% \subsubsection{nodex --- Extend the node library}
-%
-% Get the node id form, e. g.:
-% \begin{code}
-% <node    nil <    172 >    nil : hlist 2>
-% \end{code}
-%    \begin{macrocode}
-function nodex.node_id(n)
-  return string.gsub(tostring(n), '^<node%s+%S+%s+<%s+(%d+).*', '%1')
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function nodex.subtype(n)
-  local typ = node.type(n.id)
-  local subtypes = {
-%    \end{macrocode}
-% \paragraph{hlist (0)}
-%    \begin{macrocode}
-    hlist = {
-      [0] = 'unknown',
-      [1] = 'line',
-      [2] = 'box',
-      [3] = 'indent',
-      [4] = 'alignment',
-      [5] = 'cell',
-      [6] = 'equation',
-      [7] = 'equationnumber',
-    },
-%    \end{macrocode}
-% \paragraph{vlist (1)}
-%    \begin{macrocode}
-    vlist = {
-      [0] = 'unknown',
-      [4] = 'alignment',
-      [5] = 'cell',
-    },
-%    \end{macrocode}
-% \paragraph{rule (2)}
-%    \begin{macrocode}
-    rule = {
-      [0] = 'unknown',
-      [1] = 'box',
-      [2] = 'image',
-      [3] = 'empty',
-      [4] = 'user',
-    },
-%    \end{macrocode}
-%
-% \noindent
-% Nodes without subtypes:
-% \begin{compactitem}
-% \item ins (3)
-% \item mark (4)
-% \end{compactitem}
-%    \begin{macrocode}
-%    \end{macrocode}
-% \paragraph{adjust (5)}
-%    \begin{macrocode}
-    adjust = {
-      [0] = 'normal',
-      [1] = 'pre',
-    },
-%    \end{macrocode}
-% \paragraph{boundary (6)}
-%    \begin{macrocode}
-    boundary = {
-      [0] = 'cancel',
-      [1] = 'user',
-      [2] = 'protrusion',
-      [3] = 'word',
-    },
-%    \end{macrocode}
-% \paragraph{disc (7)}
-%    \begin{macrocode}
-    disc  = {
-      [0] = 'discretionary',
-      [1] = 'explicit',
-      [2] = 'automatic',
-      [3] = 'regular',
-      [4] = 'first',
-      [5] = 'second',
-    },
-%    \end{macrocode}
-%
-% \noindent
-% Nodes without subtypes:
-% \begin{compactitem}
-% \item whatsit (8)
-% \item local\_par (9)
-% \item dir (10)
-% \end{compactitem}
-%
-% \paragraph{math (11)}
-%    \begin{macrocode}
-    math = {
-      [0] = 'beginmath',
-      [1] = 'endmath',
-    },
-%    \end{macrocode}
-% \paragraph{glue (12)}
-%    \begin{macrocode}
-    glue = {
-      [0]   = 'userskip',
-      [1]   = 'lineskip',
-      [2]   = 'baselineskip',
-      [3]   = 'parskip',
-      [4]   = 'abovedisplayskip',
-      [5]   = 'belowdisplayskip',
-      [6]   = 'abovedisplayshortskip',
-      [7]   = 'belowdisplayshortskip',
-      [8]   = 'leftskip',
-      [9]   = 'rightskip',
-      [10]  = 'topskip',
-      [11]  = 'splittopskip',
-      [12]  = 'tabskip',
-      [13]  = 'spaceskip',
-      [14]  = 'xspaceskip',
-      [15]  = 'parfillskip',
-      [16]  = 'mathskip',
-      [17]  = 'thinmuskip',
-      [18]  = 'medmuskip',
-      [19]  = 'thickmuskip',
-      [98]  = 'conditionalmathskip',
-      [99]  = 'muglue',
-      [100] = 'leaders',
-      [101] = 'cleaders',
-      [102] = 'xleaders',
-      [103] = 'gleaders',
-    },
-%    \end{macrocode}
-% \paragraph{kern (13)}
-%    \begin{macrocode}
-    kern = {
-      [0] = 'fontkern',
-      [1] = 'userkern',
-      [2] = 'accentkern',
-      [3] = 'italiccorrection',
-    },
-%    \end{macrocode}
-%
-% \noindent
-% Nodes without subtypes:
-% \begin{compactitem}
-% \item penalty (14)
-% \item unset (15)
-% \item style (16)
-% \item choice (17)
-% \end{compactitem}
-%
-% \paragraph{noad (18)}
-%    \begin{macrocode}
-    noad = {
-      [0] = 'ord',
-      [1] = 'opdisplaylimits',
-      [2] = 'oplimits',
-      [3] = 'opnolimits',
-      [4] = 'bin',
-      [5] = 'rel',
-      [6] = 'open',
-      [7] = 'close',
-      [8] = 'punct',
-      [9] = 'inner',
-      [10] = 'under',
-      [11] = 'over',
-      [12] = 'vcenter',
-    },
-%    \end{macrocode}
-% \paragraph{radical (19)}
-%    \begin{macrocode}
-    radical = {
-      [0] = 'radical',
-      [1] = 'uradical',
-      [2] = 'uroot',
-      [3] = 'uunderdelimiter',
-      [4] = 'uoverdelimiter',
-      [5] = 'udelimiterunder',
-      [6] = 'udelimiterover',
-    },
-%    \end{macrocode}
-%
-% \noindent
-% Nodes without subtypes:
-% \begin{compactitem}
-% \item fraction (20)
-% \end{compactitem}
-%
-% \paragraph{accent (21)}
-%    \begin{macrocode}
-    accent = {
-      [0] = 'bothflexible',
-      [1] = 'fixedtop',
-      [2] = 'fixedbottom',
-      [3] = 'fixedboth',
-    },
-%    \end{macrocode}
-% \paragraph{fence (22)}
-%    \begin{macrocode}
-    fence = {
-      [0] = 'unset',
-      [1] = 'left',
-      [2] = 'middle',
-      [3] = 'right',
-    },
-%    \end{macrocode}
-%
-% \noindent
-% Nodes without subtypes:
-% \begin{compactitem}
-% \item math\_char (23)
-% \item sub\_box (24)
-% \item sub\_mlist (25)
-% \item math\_text\_char (26)
-% \item delim (27)
-% \item margin\_kern (28)
-% \end{compactitem}
-%
-% \paragraph{glyph (29)}
-%    \begin{macrocode}
-    glyph = {
-      [0] = 'character',
-      [1] = 'ligature',
-      [2] = 'ghost',
-      [3] = 'left',
-      [4] = 'right',
-    },
-%    \end{macrocode}
-%
-% \noindent
-% Nodes without subtypes:
-% \begin{compactitem}
-% \item align\_record (30)
-% \item pseudo\_file (31)
-% \item pseudo\_line (32)
-% \item page\_insert (33)
-% \item split\_insert (34)
-% \item expr\_stack (35)
-% \item nested\_list (36)
-% \item span (37)
-% \item attribute (38)
-% \item glue\_spec (39)
-% \item attribute\_list (40)
-% \item temp (41)
-% \item align\_stack (42)
-% \item movement\_stack (43)
-% \item if\_stack (44)
-% \item unhyphenated (45)
-% \item hyphenated (46)
-% \item delta (47)
-% \item passive (48)
-% \item shape (49)
-% \end{compactitem}
-%    \begin{macrocode}
-  }
-  subtypes.whatsit = node.whatsits()
-  local out = ''
-  if subtypes[typ] and subtypes[typ][n.subtype] then
-    out = subtypes[typ][n.subtype]
-    if options.verbosity > 1 then
-      out = out .. tpl.type_id(n.subtype)
-    end
-    return out
-  else
-    return tostring(n.subtype)
-  end
-  assert(false)
-end
-%    \end{macrocode}
-%
-% \subsubsection{tpl --- Template function}
-%
-%    \begin{macrocode}
-function tpl.round(number)
-  local mult = 10^(options.decimalplaces or 0)
-  return math.floor(number * mult + 0.5) / mult
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tpl.length(input)
-  input = tonumber(input)
-  input = input / tex.sp('1' .. options.unit)
-  return string.format('%g%s', tpl.round(input), options.unit)
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tpl.fill(number, order, field)
-  if order ~= nil and order ~= 0 then
-    if field == 'stretch' then
-      out = '+'
-    else
-      out = '-'
-    end
-    return out .. string.format(
-      '%gfi%s', number / 2^16,
-      string.rep('l', order - 1)
-    )
-  else
-    return tpl.length(number)
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-tpl.node_colors = {
-  hlist = {'red', 'bright'},
-  vlist = {'green', 'bright'},
-  rule = {'blue', 'bright'},
-  ins = {'blue'},
-  mark = {'magenta'},
-  adjust = {'cyan'},
-  boundary = {'red', 'bright'},
-  disc = {'green', 'bright'},
-  whatsit = {'yellow', 'bright'},
-  local_par = {'blue', 'bright'},
-  dir = {'magenta', 'bright'},
-  math = {'cyan', 'bright'},
-  glue = {'magenta', 'bright'},
-  kern = {'green', 'bright'},
-  penalty = {'yellow', 'bright'},
-  unset = {'blue'},
-  style = {'magenta'},
-  choice = {'cyan'},
-  noad = {'red'},
-  radical = {'green'},
-  fraction = {'yellow'},
-  accent = {'blue'},
-  fence = {'magenta'},
-  math_char = {'cyan'},
-  sub_box = {'red', 'bright'},
-  sub_mlist = {'green', 'bright'},
-  math_text_char = {'yellow', 'bright'},
-  delim = {'blue', 'bright'},
-  margin_kern = {'magenta', 'bright'},
-  glyph = {'cyan', 'bright'},
-  align_record = {'red'},
-  pseudo_file = {'green'},
-  pseudo_line = {'yellow'},
-  page_insert = {'blue'},
-  split_insert = {'magenta'},
-  expr_stack = {'cyan'},
-  nested_list = {'red'},
-  span = {'green'},
-  attribute = {'yellow'},
-  glue_spec = {'magenta'},
-  attribute_list = {'cyan'},
-  temp = {'magenta'},
-  align_stack = {'red', 'bright'},
-  movement_stack = {'green', 'bright'},
-  if_stack = {'yellow', 'bright'},
-  unhyphenated = {'magenta', 'bright'},
-  hyphenated = {'cyan', 'bright'},
-  delta = {'red'},
-  passive = {'green'},
-  shape = {'yellow'},
+\directlua{
+  nodetree = require('nodetree')
+  nodetree.check_shell_escape()
 }
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.color_code(code)
-  return string.char(27) .. '[' .. tostring(code) .. 'm'
-end
+\define at key{NTE}{callback}[]{\NodetreeSetOption[callback]{#1}}
 %    \end{macrocode}
 %
-% \begin{code}
-% local colors = {
-%     -- attributes
-%     reset = 0,
-%     clear = 0,
-%     bright = 1,
-%     dim = 2,
-%     underscore = 4,
-%     blink = 5,
-%     reverse = 7,
-%     hidden = 8,
-%
-%     -- foreground
-%     black = 30,
-%     red = 31,
-%     green = 32,
-%     yellow = 33,
-%     blue = 34,
-%     magenta = 35,
-%     cyan = 36,
-%     white = 37,
-%
-%     -- background
-%     onblack = 40,
-%     onred = 41,
-%     ongreen = 42,
-%     onyellow = 43,
-%     onblue = 44,
-%     onmagenta = 45,
-%     oncyan = 46,
-%     onwhite = 47,
-% }
-% \end{code}
 %    \begin{macrocode}
-function tpl.color(color, mode, background)
-  if options.color ~= 'colored' then
-    return ''
-  end
+\DeclareStringOption[1]{verbosity}
+\define at key{NTE}{verbosity}[]{\NodetreeSetOption[verbosity]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-  local out = ''
-  local code = ''
+\DeclareStringOption[colored]{color}
+\define at key{NTE}{color}[]{\NodetreeSetOption[color]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-  if mode == 'bright' then
-    out = tpl.color_code(1)
-  elseif mode == 'dim' then
-    out = tpl.color_code(2)
-  end
+\DeclareStringOption[1]{unit}
+\define at key{NTE}{unit}[]{\NodetreeSetOption[unit]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-  if not background then
-    if color == 'reset' then code = 0
-    elseif color == 'red' then code = 31
-    elseif color == 'green' then code = 32
-    elseif color == 'yellow' then code = 33
-    elseif color == 'blue' then code = 34
-    elseif color == 'magenta' then code = 35
-    elseif color == 'cyan' then code = 36
-    else code = 37 end
-  else
-    if color == 'black' then code = 40
-    elseif color == 'red' then code = 41
-    elseif color == 'green' then code = 42
-    elseif color == 'yellow' then code = 43
-    elseif color == 'blue' then code = 44
-    elseif color == 'magenta' then code = 45
-    elseif color == 'cyan' then code = 46
-    elseif color == 'white' then code = 47
-    else code = 40 end
-  end
-  return out .. tpl.color_code(code)
-end
+\DeclareStringOption[1]{decimalplaces}
+\define at key{NTE}{decimalplaces}[]{\NodetreeSetOption[decimalplaces]{#1}}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.key_value(key, value)
-  local out = tpl.color('yellow') .. key .. ': '
-  if value then
-    out = out .. tpl.color('white') .. value .. '; '
-  end
-  return out .. tpl.color('reset')
-end
+\DeclareStringOption[monokaisoda]{theme}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.char(input)
-  return string.format('%q', unicode.utf8.char(input))
-end
+\DeclareStringOption[dark]{thememode}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.type(type, id)
-  local out = tpl.color(
-    tpl.node_colors[type][1],
-    tpl.node_colors[type][2]
-    )
-    .. string.upper(type)
-  if options.verbosity > 1 then
-    out = out .. tpl.type_id(id)
-  end
-  return out .. tpl.color('reset')  .. ' '
-end
+\DeclareStringOption[Ubuntu Mono]{font}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.callback_variable(variable_name, variable)
-  if variable ~= nil and variable ~= '' then
-    tpl.print(variable_name .. ': ' .. tostring(variable))
-  end
-end
+\DeclareStringOption[\footnotesize]{fontsize}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.line(length)
-  if length == 'long' then
-    return '------------------------------------------'
-  else
-    return '-----------------------'
-  end
-end
+\DeclareBoolOption{showmarkup}
 %    \end{macrocode}
 %
 %    \begin{macrocode}
-function tpl.callback(callback_name, variables)
-  tpl.print('\n\n')
-  tpl.print('Callback: ' .. tpl.color('red', '', true) ..
-    callback_name .. tpl.color('reset')
-  )
-  if variables then
-    for name, value in pairs(variables) do
-      if value ~= nil and value ~= '' then
-        tpl.print('  - ' .. name .. ': ' .. tostring(value))
-      end
-    end
-  end
-  tpl.print(tpl.line('long'))
-end
+\ProcessKeyvalOptions{NTE}
 %    \end{macrocode}
 %
+% \begin{macro}{\NTE at colors}
 %    \begin{macrocode}
-function tpl.type_id(id)
-  return '[' .. tostring(id) .. ']'
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tpl.branch(connection_type, connection_state, last)
-  local c = connection_type
-  local s = connection_state
-  local l = last
-  if c == 'list' and s == 'stop' and l == false then
-    return '  '
-  elseif c == 'field' and s == 'stop' and l == false then
-    return '  '
-  elseif c == 'list' and s == 'continue' and l == false then
-    return '│ '
-  elseif c == 'field' and s == 'continue' and l == false then
-    return '║ '
-  elseif c == 'list' and s == 'continue' and l == true then
-    return '├─'
-  elseif c == 'field' and s == 'continue' and l == true then
-    return '╠═'
-  elseif c == 'list' and s == 'stop' and l == true then
-    return '└─'
-  elseif c == 'field' and s == 'stop' and l == true then
-    return '╚═'
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tpl.branches(level, connection_type)
-  local out = ''
-  for i = 1, level - 1  do
-    out = out .. tpl.branch('list', tree.state[i]['list'], false)
-    out = out .. tpl.branch('field', tree.state[i]['field'], false)
-  end
-%    \end{macrocode}
-% Format the last branches
-%    \begin{macrocode}
-  if connection_type == 'list' then
-    out = out .. tpl.branch('list', tree.state[level]['list'], true)
-  else
-    out = out .. tpl.branch('list', tree.state[level]['list'], false)
-    out = out .. tpl.branch('field', tree.state[level]['field'], true)
-  end
-  return out
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tpl.print(text)
-
-  if options.channel == 'log' then
-    if not log then
-      log = io.open(tex.jobname .. '_nodetree.log', 'a')
-    end
-    log:write(text, '\n')
-  else
-    print('  ' .. text)
-  end
-end
-%    \end{macrocode}
-%
-% \subsubsection{tree --- Build the node tree}
-%
-%    \begin{macrocode}
-function tree.format_field(head, field)
-  local out = ''
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  if not head[field] or head[field] == 0 then
-    return ''
-  end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  if options.verbosity < 2 and
-    -- glyph
-    field == 'font' or
-    field == 'left' or
-    field == 'right' or
-    field == 'uchyph' or
-    -- hlist
-    field == 'dir' or
-    field == 'glue_order' or
-    field == 'glue_sign' or
-    field == 'glue_set' or
-    -- glue
-    field == 'stretch_order' then
-    return ''
-  elseif options.verbosity < 3 and
-    field == 'prev' or
-    field == 'next' or
-    field == 'id'
-  then
-    return ''
-  end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  if field == 'prev' or field == 'next' then
-    out = nodex.node_id(head[field])
-  elseif field == 'subtype' then
-    out = nodex.subtype(head)
-  elseif
-    field == 'width' or
-    field == 'height' or
-    field == 'depth' or
-    field == 'kern' or
-    field == 'shift' then
-    out = tpl.length(head[field])
-  elseif field == 'char' then
-    out = tpl.char(head[field])
-  elseif field == 'glue_set' then
-    out = tpl.round(head[field])
-  elseif field == 'stretch' or field == 'shrink' then
-    out = tpl.fill(head[field], head[field .. '_order'], field)
-  else
-    out = tostring(head[field])
-  end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  return tpl.key_value(field, out)
-end
-%    \end{macrocode}
-%
-% |level| is a integer beginning with 1. The variable |connection_type|
-% is a string, which can be either |list| or |field|. The variable
-% |connection_state| is a string, which can be either |continue| or
-% |stop|.
-%    \begin{macrocode}
-function tree.set_state(level, connection_type, connection_state)
-  if not tree.state[level] then
-    tree.state[level] = {}
-  end
-  tree.state[level][connection_type] = connection_state
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tree.analyze_fields(fields, level)
-  local max = 0
-  local connection_state = ''
-  for _ in pairs(fields) do
-    max = max + 1
-  end
-  local count = 0
-  for field_name, recursion_node in pairs(fields) do
-    count = count + 1
-    if count == max then
-      connection_state = 'stop'
-    else
-      connection_state = 'continue'
-    end
-    tree.set_state(level, 'field', connection_state)
-    tpl.print(tpl.branches(level, 'field') .. tpl.key_value(field_name))
-    tree.analyze_list(recursion_node, level + 1)
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tree.analyze_node(head, level)
-  local connection_state
-  local out = ''
-  if head.next then
-    connection_state = 'continue'
-  else
-    connection_state = 'stop'
-  end
-  tree.set_state(level, 'list', connection_state)
-  out = tpl.branches(level, 'list')
-    .. tpl.type(node.type(head.id), head.id)
-  if options.verbosity > 1 then
-    out = out .. tpl.key_value('no', nodex.node_id(head))
-  end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  local fields = {}
-  for field_id, field_name in pairs(node.fields(head.id, head.subtype)) do
-    if field_name ~= 'next' and
-      field_name ~= 'prev' and
-      node.is_node(head[field_name]) then
-      fields[field_name] = head[field_name]
-    else
-      out = out .. tree.format_field(head, field_name)
-    end
-  end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  tpl.print(out)
-  tree.analyze_fields(fields, level)
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tree.analyze_list(head, level)
-  while head do
-    tree.analyze_node(head, level)
-    head = head.next
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function tree.analyze_callback(head)
-  tree.analyze_list(head, 1)
-  tpl.print(tpl.line('short') .. '\n')
-end
-%    \end{macrocode}
-%
-% \subsubsection{callbacks --- Callback wrapper}
-%
-%    \begin{macrocode}
-function callbacks.contribute_filter(extrainfo)
-  tpl.callback('contribute_filter', {extrainfo = extrainfo})
-  return true
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.buildpage_filter(extrainfo)
-  tpl.callback('buildpage_filter', {extrainfo = extrainfo})
-  return true
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.pre_linebreak_filter(head, groupcode)
-  tpl.callback('pre_linebreak_filter', {groupcode = groupcode})
-  tree.analyze_callback(head)
-  return true
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.linebreak_filter(head, is_display)
-  tpl.callback('linebreak_filter', {is_display = is_display})
-  tree.analyze_callback(head)
-  return true
-end
-%    \end{macrocode}
-%
-% TODO: Fix return values, page output
-%    \begin{macrocode}
-function callbacks.append_to_vlist_filter(head, locationcode, prevdepth, mirrored)
-  local variables = {
-    locationcode = locationcode,
-    prevdepth = prevdepth,
-    mirrored = mirrored,
+\ExplSyntaxOn
+\def\NTE at colors{
+  \str_case_e:nn{\NTEK at theme}{
+    {bwdark}{
+      \definecolor{NTEblack}{gray}{0}
+      \definecolor{NTEred}{gray}{1}
+      \definecolor{NTEgreen}{gray}{1}
+      \definecolor{NTEyellow}{gray}{1}
+      \definecolor{NTEblue}{gray}{1}
+      \definecolor{NTEmagenta}{gray}{1}
+      \definecolor{NTEcyan}{gray}{1}
+      \definecolor{NTEwhite}{gray}{1}
+      \definecolor{NTEblackbright}{gray}{0}
+      \definecolor{NTEredbright}{gray}{1}
+      \definecolor{NTEgreenbright}{gray}{1}
+      \definecolor{NTEyellowbright}{gray}{1}
+      \definecolor{NTEbluebright}{gray}{1}
+      \definecolor{NTEmagentabright}{gray}{1}
+      \definecolor{NTEcyanbright}{gray}{1}
+      \definecolor{NTEwhitebright}{gray}{1}
+    }
+    {bwlight}{
+      \definecolor{NTEblack}{gray}{0}
+      \definecolor{NTEred}{gray}{0}
+      \definecolor{NTEgreen}{gray}{0}
+      \definecolor{NTEyellow}{gray}{0}
+      \definecolor{NTEblue}{gray}{0}
+      \definecolor{NTEmagenta}{gray}{0}
+      \definecolor{NTEcyan}{gray}{0}
+      \definecolor{NTEwhite}{gray}{1}
+      \definecolor{NTEblackbright}{gray}{0}
+      \definecolor{NTEredbright}{gray}{0}
+      \definecolor{NTEgreenbright}{gray}{0}
+      \definecolor{NTEyellowbright}{gray}{0}
+      \definecolor{NTEbluebright}{gray}{0}
+      \definecolor{NTEmagentabright}{gray}{0}
+      \definecolor{NTEcyanbright}{gray}{0}
+      \definecolor{NTEwhitebright}{gray}{1}
+    }
+    {monokaisoda}{
+      \definecolor{NTEblack}{HTML}{1a1a1a}
+      \definecolor{NTEred}{HTML}{f4005f}
+      \definecolor{NTEgreen}{HTML}{98e024}
+      \definecolor{NTEyellow}{HTML}{fa8419}
+      \definecolor{NTEblue}{HTML}{9d65ff}
+      \definecolor{NTEmagenta}{HTML}{f4005f}
+      \definecolor{NTEcyan}{HTML}{58d1eb}
+      \definecolor{NTEwhite}{HTML}{c4c5b5}
+      \definecolor{NTEblackbright}{HTML}{625e4c}
+      \definecolor{NTEredbright}{HTML}{f4005f}
+      \definecolor{NTEgreenbright}{HTML}{98e024}
+      \definecolor{NTEyellowbright}{HTML}{e0d561}
+      \definecolor{NTEbluebright}{HTML}{9d65ff}
+      \definecolor{NTEmagentabright}{HTML}{f4005f}
+      \definecolor{NTEcyanbright}{HTML}{58d1eb}
+      \definecolor{NTEwhitebright}{HTML}{f6f6ef}
+    }
   }
-  tpl.callback('append_to_vlist_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.post_linebreak_filter(head, groupcode)
-  tpl.callback('post_linebreak_filter', {groupcode = groupcode})
-  tree.analyze_callback(head)
-  return true
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.hpack_filter(head, groupcode, size, packtype, direction, attributelist)
-  local variables = {
-    groupcode = groupcode,
-    size = size,
-    packtype = packtype,
-    direction = direction,
-    attributelist = attributelist,
+  \str_case_e:nn{\NTEK at thememode}{
+    {dark}{
+      \definecolor{NTEbackground}{named}{NTEblack}
+      \definecolor{NTEfont}{named}{NTEwhitebright}
+    }
+    {light}{
+      \definecolor{NTEbackground}{named}{NTEwhitebright}
+      \definecolor{NTEfont}{named}{NTEblack}
+    }
   }
-  tpl.callback('hpack_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
+}
+\ExplSyntaxOff
 %    \end{macrocode}
+% \end{macro}
 %
+% \begin{macro}{\NTE at fonts}
 %    \begin{macrocode}
-function callbacks.vpack_filter(head, groupcode, size, packtype, maxdepth, direction, attributelist)
-  local variables = {
-    groupcode = groupcode,
-    size = size,
-    packtype = packtype,
-    maxdepth = tpl.length(maxdepth),
-    direction = direction,
-    attributelist = attributelist,
-  }
-  tpl.callback('vpack_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
+\def\NTE at fonts{
+  \bfseries%
+  \NTEK at fontsize%
+  \setmonofont{\NTEK at font}%
+  \ttfamily%
+  \setlength{\parindent}{0pt}%
+  \setlength{\parskip}{-0.9pt}%
+}
 %    \end{macrocode}
+% \end{macro}
 %
+% \begin{macro}{\NodetreeSet}
+% Same definition as in nodetree.sty. Only implement this command
+% if not already registers.
 %    \begin{macrocode}
-function callbacks.hpack_quality(incident, detail, head, first, last)
-  local variables = {
-    incident = incident,
-    detail = detail,
-    first = first,
-    last = last,
-  }
-  tpl.callback('hpack_quality', variables)
-  tree.analyze_callback(head)
-end
+\providecommand{\NodetreeSet}[1]{%
+  \setkeys{NTE}{#1}%
+}
 %    \end{macrocode}
+% \end{macro}
 %
 %    \begin{macrocode}
-function callbacks.vpack_quality(incident, detail, head, first, last)
-  local variables = {
-    incident = incident,
-    detail = detail,
-    first = first,
-    last = last,
-  }
-  tpl.callback('vpack_quality', variables)
-  tree.analyze_callback(head)
-end
+\newenvironment{NodetreeEmbedView}[1][]{
+  \setkeys{NTE}{#1}
+  \NTE at colors
+  \begin{mdframed}[
+    linecolor=black,
+    backgroundcolor=NTEbackground,
+    fontcolor=NTEfont,
+  ]%
+  \NTE at fonts
+}{
+  \end{mdframed}%
+}
 %    \end{macrocode}
 %
+% \begin{environment}{NodetreeEmbedEnv}
 %    \begin{macrocode}
-function callbacks.process_rule(head, width, height)
-  local variables = {
-    width = width,
-    height = height,
-  }
-  tpl.callback('process_rule', variables)
-  tree.analyze_callback(head)
-  return true
-end
+\NewDocumentEnvironment { NodetreeEmbedEnv } { O{} +b } {
+  \setkeys{NTE}{#1}
+  \ifNTEK at showmarkup
+    \noindent
+    \texttt{\detokenize{#2}}
+  \else
+  \fi
+  \NTE at colors
+  \begin{NodetreeEmbedView}
+    \directlua{
+      nodetree.compile_include('\luaescapestring{\unexpanded{#2}}')
+    }
+  \end{NodetreeEmbedView}
+}{}
 %    \end{macrocode}
-%
+% \end{environment}
+
+% \begin{macro}{\NodetreeEmbedCmd}
 %    \begin{macrocode}
-function callbacks.pre_output_filter(head, groupcode, size, packtype, maxdepth, direction)
-  local variables = {
-    groupcode = groupcode,
-    size = size,
-    packtype = packtype,
-    maxdepth = maxdepth,
-    direction = direction,
-  }
-  tpl.callback('pre_output_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
+\NewDocumentCommand { \NodetreeEmbedCmd } { O{} +v } {
+  \setkeys{NTE}{#1}
+  \ifNTEK at showmarkup
+    \noindent
+    \texttt{#2}
+  \else
+  \fi
+  \NTE at colors
+  \begin{NodetreeEmbedView}
+    \directlua{
+      nodetree.compile_include('\luaescapestring{\unexpanded{#2}}')
+    }
+  \end{NodetreeEmbedView}
+}
 %    \end{macrocode}
+% \end{macro}
 %
+% \begin{macro}{\NodetreeEmbedInput}
 %    \begin{macrocode}
-function callbacks.hyphenate(head, tail)
-  tpl.callback('hyphenate')
-  tpl.print('head:')
-  tree.analyze_callback(head)
-  tpl.print('tail:')
-  tree.analyze_callback(tail)
-end
+\newcommand{\NodetreeEmbedInput}[2][]{
+  \setkeys{NTE}{#1}
+  \begin{NodetreeEmbedView}
+  \input{#2.nttex}
+  \end{NodetreeEmbedView}
+}
+\let\nodetreeterminalemulator\NodetreeEmbedInput
 %    \end{macrocode}
+% \end{macro}
 %
-%    \begin{macrocode}
-function callbacks.ligaturing(head, tail)
-  tpl.callback('ligaturing')
-  tpl.print('head:')
-  tree.analyze_callback(head)
-  tpl.print('tail:')
-  tree.analyze_callback(tail)
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.kerning(head, tail)
-  tpl.callback('kerning')
-  tpl.print('head:')
-  tree.analyze_callback(head)
-  tpl.print('tail:')
-  tree.analyze_callback(tail)
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.insert_local_par(local_par, location)
-  tpl.callback('insert_local_par', {location = location})
-  tree.analyze_callback(local_par)
-  return true
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function callbacks.mlist_to_hlist(head, display_type, need_penalties)
-  local variables = {
-    display_type = display_type,
-    need_penalties = need_penalties,
-  }
-  tpl.callback('mlist_to_hlist', variables)
-  tree.analyze_callback(head)
-  return node.mlist_to_hlist(head, display_type, need_penalties)
-end
-%    \end{macrocode}
-%
-% \subsubsection{base --- Exported base functions}
-%
-%    \begin{macrocode}
-function base.normalize_options()
-  options.verbosity = tonumber(options.verbosity)
-  options.decimalplaces = tonumber(options.decimalplaces)
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.set_default_options()
-  local defaults = {
-    verbosity = 1,
-    callback = 'postlinebreak',
-    engine = 'luatex',
-    color = 'colored',
-    decimalplaces = 2,
-    unit = 'pt',
-    channel = 'term',
-  }
-  if not options then
-    options = {}
-  end
-  for key, value in pairs(defaults) do
-    if not options[key] then
-      options[key] = value
-    end
-  end
-  base.normalize_options()
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.set_option(key, value)
-  if not options then
-    options = {}
-  end
-  options[key] = value
-  base.normalize_options()
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.get_option(key)
-  if not options then
-    options = {}
-  end
-  if options[key] then
-    return options[key]
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.get_callback_name(alias)
-  if alias == 'contribute' or alias == 'contributefilter' then
-    return 'contribute_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'buildpage' or alias == 'buildpagefilter' then
-    return 'buildpage_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'preline' or alias == 'prelinebreakfilter' then
-    return 'pre_linebreak_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'line' or alias == 'linebreakfilter' then
-    return 'linebreak_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'append' or alias == 'appendtovlistfilter' then
-    return 'append_to_vlist_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'postline' or alias == 'postlinebreakfilter' then
-    return 'post_linebreak_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'hpack' or alias == 'hpackfilter' then
-    return 'hpack_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'vpack' or alias == 'vpackfilter' then
-    return 'vpack_filter'
-%    \end{macrocode}
-% TODO: Fix: Unable to register callback
-%    \begin{macrocode}
-  elseif alias == 'hpackq' or alias == 'hpackquality' then
-    return 'hpack_quality'
-%    \end{macrocode}
-% TODO: Fix: Unable to register callback
-%    \begin{macrocode}
-  elseif alias == 'vpackq' or alias == 'vpackquality' then
-    return 'vpack_quality'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'process' or alias == 'processrule' then
-    return 'process_rule'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'preout' or alias == 'preoutputfilter' then
-    return 'pre_output_filter'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'hyph' or alias == 'hyphenate' then
-    return 'hyphenate'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'liga' or alias == 'ligaturing' then
-    return 'ligaturing'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'kern' or alias == 'kerning' then
-   return 'kerning'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'insert' or alias == 'insertlocalpar' then
-    return 'insert_local_par'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  elseif alias == 'mhlist' or alias == 'mlisttohlist' then
-    return 'mlist_to_hlist'
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-  else
-    return 'post_linebreak_filter'
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.register(cb)
-  if options.engine == 'lualatex' then
-    luatexbase.add_to_callback(cb, callbacks[cb], 'nodetree')
-  else
-    id, error = callback.register(cb, callbacks[cb])
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.register_callbacks()
-  for alias in string.gmatch(options.callback, '([^,]+)') do
-    base.register(base.get_callback_name(alias))
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.unregister(cb)
-  if options.engine == 'lualatex' then
-    luatexbase.remove_from_callback(cb, 'nodetree')
-  else
-    id, error = callback.register(cb, nil)
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.unregister_callbacks()
-  for alias in string.gmatch(options.callback, '([^,]+)') do
-    base.unregister(base.get_callback_name(alias))
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.execute()
-  local c = base.get_callback()
-  if options.engine == 'lualatex' then
-    luatexbase.add_to_callback(c, callbacks.post_linebreak_filter, 'nodetree')
-  else
-    id, error = callback.register(c, callbacks.post_linebreak_filter)
-  end
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-function base.analyze(head)
-  tpl.print('\n')
-  tree.analyze_list(head, 1)
-end
-%    \end{macrocode}
-%
-%    \begin{macrocode}
-return base
-%    \end{macrocode}
+\endinput
 % \iffalse
-%</luamain>
+%</packageembed>
 % \fi
 %
 % \Finale

Modified: trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.ins
===================================================================
--- trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.ins	2020-05-30 20:58:42 UTC (rev 55346)
+++ trunk/Master/texmf-dist/source/luatex/nodetree/nodetree.ins	2020-05-30 20:59:30 UTC (rev 55347)
@@ -1,4 +1,4 @@
-% Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
+% Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
 % ----------------------------------------------------------------------
 % This work may be distributed and/or modified under the conditions of
 % the LaTeX Project Public License, either version 1.3c of this license
@@ -21,7 +21,7 @@
 
 This is a generated file.
 
-Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
+Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
 ----------------------------------------------------------------------
 This work may be distributed and/or modified under the conditions of
 the LaTeX Project Public License, either version 1.3c of this license
@@ -37,12 +37,8 @@
 
 \generate{\file{nodetree.tex}{\from{nodetree.dtx}{tex}}}
 \generate{\file{nodetree.sty}{\from{nodetree.dtx}{package}}}
+\generate{\file{nodetree-embed.sty}{\from{nodetree.dtx}{packageembed}}}
 
-\nopreamble
-\nopostamble
-\generate{\file{nodetree.lua}{\from{nodetree.dtx}{luamain}}}
-\generate{\file{README.md}{\from{nodetree.dtx}{readme}}}
-
 \obeyspaces
 \Msg{*************************************************************}
 \Msg{*                                                           *}

Added: trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree-embed.sty
===================================================================
--- trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree-embed.sty	                        (rev 0)
+++ trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree-embed.sty	2020-05-30 20:59:30 UTC (rev 55347)
@@ -0,0 +1,183 @@
+%%
+%% This is file `nodetree-embed.sty',
+%% generated with the docstrip utility.
+%%
+%% The original source files were:
+%%
+%% nodetree.dtx  (with options: `packageembed')
+%% 
+%% This is a generated file.
+%% 
+%% Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
+%% ----------------------------------------------------------------------
+%% This work may be distributed and/or modified under the conditions of
+%% the LaTeX Project Public License, either version 1.3c of this license
+%% or (at your option) any later version. The latest version of this
+%% license is in:
+%% 
+%%   http://www.latex-project.org/lppl.txt
+%% 
+%% and version 1.3c or later is part of all distributions of LaTeX
+%% version 2008/05/05 or later.
+%% 
+\NeedsTeXFormat{LaTeX2e}[1994/06/01]
+\ProvidesPackage{nodetree-embed}
+  [2020/05/29 v2.0 Embed node trees into a LaTeX document]
+\RequirePackage{xcolor,mdframed,expl3,xparse,fontspec}
+\input{nodetree}
+\RequirePackage{kvoptions}
+\SetupKeyvalOptions{
+  family=NTE,
+  prefix=NTEK@
+}
+\directlua{
+  nodetree = require('nodetree')
+  nodetree.check_shell_escape()
+}
+\define at key{NTE}{callback}[]{\NodetreeSetOption[callback]{#1}}
+\DeclareStringOption[1]{verbosity}
+\define at key{NTE}{verbosity}[]{\NodetreeSetOption[verbosity]{#1}}
+\DeclareStringOption[colored]{color}
+\define at key{NTE}{color}[]{\NodetreeSetOption[color]{#1}}
+\DeclareStringOption[1]{unit}
+\define at key{NTE}{unit}[]{\NodetreeSetOption[unit]{#1}}
+\DeclareStringOption[1]{decimalplaces}
+\define at key{NTE}{decimalplaces}[]{\NodetreeSetOption[decimalplaces]{#1}}
+\DeclareStringOption[monokaisoda]{theme}
+\DeclareStringOption[dark]{thememode}
+\DeclareStringOption[Ubuntu Mono]{font}
+\DeclareStringOption[\footnotesize]{fontsize}
+\DeclareBoolOption{showmarkup}
+\ProcessKeyvalOptions{NTE}
+\ExplSyntaxOn
+\def\NTE at colors{
+  \str_case_e:nn{\NTEK at theme}{
+    {bwdark}{
+      \definecolor{NTEblack}{gray}{0}
+      \definecolor{NTEred}{gray}{1}
+      \definecolor{NTEgreen}{gray}{1}
+      \definecolor{NTEyellow}{gray}{1}
+      \definecolor{NTEblue}{gray}{1}
+      \definecolor{NTEmagenta}{gray}{1}
+      \definecolor{NTEcyan}{gray}{1}
+      \definecolor{NTEwhite}{gray}{1}
+      \definecolor{NTEblackbright}{gray}{0}
+      \definecolor{NTEredbright}{gray}{1}
+      \definecolor{NTEgreenbright}{gray}{1}
+      \definecolor{NTEyellowbright}{gray}{1}
+      \definecolor{NTEbluebright}{gray}{1}
+      \definecolor{NTEmagentabright}{gray}{1}
+      \definecolor{NTEcyanbright}{gray}{1}
+      \definecolor{NTEwhitebright}{gray}{1}
+    }
+    {bwlight}{
+      \definecolor{NTEblack}{gray}{0}
+      \definecolor{NTEred}{gray}{0}
+      \definecolor{NTEgreen}{gray}{0}
+      \definecolor{NTEyellow}{gray}{0}
+      \definecolor{NTEblue}{gray}{0}
+      \definecolor{NTEmagenta}{gray}{0}
+      \definecolor{NTEcyan}{gray}{0}
+      \definecolor{NTEwhite}{gray}{1}
+      \definecolor{NTEblackbright}{gray}{0}
+      \definecolor{NTEredbright}{gray}{0}
+      \definecolor{NTEgreenbright}{gray}{0}
+      \definecolor{NTEyellowbright}{gray}{0}
+      \definecolor{NTEbluebright}{gray}{0}
+      \definecolor{NTEmagentabright}{gray}{0}
+      \definecolor{NTEcyanbright}{gray}{0}
+      \definecolor{NTEwhitebright}{gray}{1}
+    }
+    {monokaisoda}{
+      \definecolor{NTEblack}{HTML}{1a1a1a}
+      \definecolor{NTEred}{HTML}{f4005f}
+      \definecolor{NTEgreen}{HTML}{98e024}
+      \definecolor{NTEyellow}{HTML}{fa8419}
+      \definecolor{NTEblue}{HTML}{9d65ff}
+      \definecolor{NTEmagenta}{HTML}{f4005f}
+      \definecolor{NTEcyan}{HTML}{58d1eb}
+      \definecolor{NTEwhite}{HTML}{c4c5b5}
+      \definecolor{NTEblackbright}{HTML}{625e4c}
+      \definecolor{NTEredbright}{HTML}{f4005f}
+      \definecolor{NTEgreenbright}{HTML}{98e024}
+      \definecolor{NTEyellowbright}{HTML}{e0d561}
+      \definecolor{NTEbluebright}{HTML}{9d65ff}
+      \definecolor{NTEmagentabright}{HTML}{f4005f}
+      \definecolor{NTEcyanbright}{HTML}{58d1eb}
+      \definecolor{NTEwhitebright}{HTML}{f6f6ef}
+    }
+  }
+  \str_case_e:nn{\NTEK at thememode}{
+    {dark}{
+      \definecolor{NTEbackground}{named}{NTEblack}
+      \definecolor{NTEfont}{named}{NTEwhitebright}
+    }
+    {light}{
+      \definecolor{NTEbackground}{named}{NTEwhitebright}
+      \definecolor{NTEfont}{named}{NTEblack}
+    }
+  }
+}
+\ExplSyntaxOff
+\def\NTE at fonts{
+  \bfseries%
+  \NTEK at fontsize%
+  \setmonofont{\NTEK at font}%
+  \ttfamily%
+  \setlength{\parindent}{0pt}%
+  \setlength{\parskip}{-0.9pt}%
+}
+\providecommand{\NodetreeSet}[1]{%
+  \setkeys{NTE}{#1}%
+}
+\newenvironment{NodetreeEmbedView}[1][]{
+  \setkeys{NTE}{#1}
+  \NTE at colors
+  \begin{mdframed}[
+    linecolor=black,
+    backgroundcolor=NTEbackground,
+    fontcolor=NTEfont,
+  ]%
+  \NTE at fonts
+}{
+  \end{mdframed}%
+}
+\NewDocumentEnvironment { NodetreeEmbedEnv } { O{} +b } {
+  \setkeys{NTE}{#1}
+  \ifNTEK at showmarkup
+    \noindent
+    \texttt{\detokenize{#2}}
+  \else
+  \fi
+  \NTE at colors
+  \begin{NodetreeEmbedView}
+    \directlua{
+      nodetree.compile_include('\luaescapestring{\unexpanded{#2}}')
+    }
+  \end{NodetreeEmbedView}
+}{}
+
+\NewDocumentCommand { \NodetreeEmbedCmd } { O{} +v } {
+  \setkeys{NTE}{#1}
+  \ifNTEK at showmarkup
+    \noindent
+    \texttt{#2}
+  \else
+  \fi
+  \NTE at colors
+  \begin{NodetreeEmbedView}
+    \directlua{
+      nodetree.compile_include('\luaescapestring{\unexpanded{#2}}')
+    }
+  \end{NodetreeEmbedView}
+}
+\newcommand{\NodetreeEmbedInput}[2][]{
+  \setkeys{NTE}{#1}
+  \begin{NodetreeEmbedView}
+  \input{#2.nttex}
+  \end{NodetreeEmbedView}
+}
+\let\nodetreeterminalemulator\NodetreeEmbedInput
+\endinput
+%%
+%% End of file `nodetree-embed.sty'.


Property changes on: trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree-embed.sty
___________________________________________________________________
Added: svn:eol-style
## -0,0 +1 ##
+native
\ No newline at end of property
Modified: trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.lua
===================================================================
--- trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.lua	2020-05-30 20:58:42 UTC (rev 55346)
+++ trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.lua	2020-05-30 20:59:30 UTC (rev 55347)
@@ -1,16 +1,632 @@
-local nodex = {}
-local tpl = {}
-local tree = {}
-tree.state = {}
-local callbacks = {}
-local base = {}
+--- The nodetree package.
+--
+-- Nodetree uses [LDoc](https://github.com/stevedonovan/ldoc) for the
+--  source code documentation. The supported tags are described on in
+--  the [wiki](https://github.com/stevedonovan/LDoc/wiki).
+--
+-- Nodes in LuaTeX are connected. The nodetree view distinguishs
+-- between the `list` and `field` connections.
+--
+-- * `list`: Nodes, which are double connected by `next` and
+--   `previous` fields.
+-- * `field`: Connections to nodes by other fields than `next` and
+--   `previous` fields, e. g. `head`, `pre`.
+-- @module nodetree
+
+-- luacheck: globals node tex luatexbase lfs callback os unicode status modules
+
+if not modules then modules = { } end modules ['nodetree'] = {
+  version   = '2.0',
+  comment   = 'nodetree',
+  author    = 'Josef Friedrich',
+  copyright = 'Josef Friedrich',
+  license   = 'The LaTeX Project Public License Version 1.3c 2008-05-04'
+}
+
+--- A counter for the compiled TeX examples. Some TeX code snippets
+-- a written into file, wrapped with some TeX boilerplate code.
+-- This written files are compiled.
+local example_counter = 0
+
+--- The default options
+local default_options = {
+  callback = 'post_linebreak_filter',
+  channel = 'term',
+  color = 'colored',
+  decimalplaces = 2,
+  engine = 'luatex', -- Required for the callback registration
+  unit = 'pt',
+  verbosity = 1,
+}
+
+--- The current options
+-- They are changed very often.
 local options = {}
-function nodex.node_id(n)
+for key, value in pairs(default_options) do
+  options[key] = value
+end
+
+if arg[0] == 'lualatex' then
+  options.engine = 'lualatex'
+end
+
+--- File descriptor
+local output_file
+
+--- The lua table named `tree_state` holds state values of the current
+-- tree item.
+--
+-- `tree_state`:
+--
+-- * `1` (level):
+--   * `list`: `continue`
+--   * `field`: `stop`
+-- * `2`:
+--   * `list`: `continue`
+--   * `field`: `stop`
+-- @table
+local tree_state = {}
+
+--- Format functions.
+--
+-- Low level template functions.
+--
+-- @section format
+
+local format = {
+  ---
+  -- @treturn string
+  underscore = function(string)
+    if options.channel == 'tex' then
+      return string.gsub(string, '_', '\\_')
+    else
+      return string
+    end
+  end,
+
+  ---
+  -- @treturn string
+  escape = function(string)
+    if options.channel == 'tex' then
+      return string.gsub(string, [[\]], [[\string\]])
+    else
+      return string
+    end
+  end,
+
+  -- @treturn number
+  number = function(number)
+    local mult = 10^(options.decimalplaces or 0)
+    return math.floor(number * mult + 0.5) / mult
+  end,
+
+  ---
+  -- @treturn string
+  whitespace = function(count)
+    local whitespace
+    local output = ''
+    if options.channel == 'tex' then
+      whitespace = '\\hspace{0.5em}'
+    else
+      whitespace = ' '
+    end
+    if not count then
+      count = 1
+    end
+    for _ = 1, count do
+      output = output .. whitespace
+    end
+    return output
+  end,
+
+  ---
+  -- @treturn string
+  color_code = function(code)
+    return string.char(27) .. '[' .. tostring(code) .. 'm'
+  end,
+
+  ---
+  -- @treturn string
+  color_tex = function(color, mode)
+    if not mode then mode = '' end
+    return 'NTE' .. color .. mode
+  end,
+
+  ---
+  -- @treturn string
+  node_begin = function()
+    if options.channel == 'tex' then
+      return '\\mbox{'
+    else
+      return ''
+    end
+  end,
+
+  ---
+  -- @treturn string
+  node_end = function()
+    if options.channel == 'tex' then
+      return '}'
+    else
+      return ''
+    end
+  end,
+
+  ---
+  -- @treturn string
+  new_line = function(count)
+    local output = ''
+    if not count then
+      count = 1
+    end
+    local new_line
+    if options.channel == 'tex' then
+      new_line = '\\par{}'
+    else
+      new_line = '\n'
+    end
+
+    for _ = 1, count do
+      output = output .. new_line
+    end
+    return output
+  end,
+
+  ---
+  -- @treturn string
+  type_id = function(id)
+    return '[' .. tostring(id) .. ']'
+  end
+}
+
+--- Print the output to stdout or write it into a file (`output_file`).
+-- New text is appended.
+--
+-- @tparam string text A text string.
+--
+local function nodetree_print(text)
+  if options.channel == 'log' or options.channel == 'tex' then
+    output_file:write(text)
+  else
+    io.write(text)
+  end
+end
+
+--- Template functions.
+-- @section template
+
+local template = {
+  node_colors = {
+    hlist = {'red', 'bright'},
+    vlist = {'green', 'bright'},
+    rule = {'blue', 'bright'},
+    ins = {'blue'},
+    mark = {'magenta'},
+    adjust = {'cyan'},
+    boundary = {'red', 'bright'},
+    disc = {'green', 'bright'},
+    whatsit = {'yellow', 'bright'},
+    local_par = {'blue', 'bright'},
+    dir = {'magenta', 'bright'},
+    math = {'cyan', 'bright'},
+    glue = {'magenta', 'bright'},
+    kern = {'green', 'bright'},
+    penalty = {'yellow', 'bright'},
+    unset = {'blue'},
+    style = {'magenta'},
+    choice = {'cyan'},
+    noad = {'red'},
+    radical = {'green'},
+    fraction = {'yellow'},
+    accent = {'blue'},
+    fence = {'magenta'},
+    math_char = {'cyan'},
+    sub_box = {'red', 'bright'},
+    sub_mlist = {'green', 'bright'},
+    math_text_char = {'yellow', 'bright'},
+    delim = {'blue', 'bright'},
+    margin_kern = {'magenta', 'bright'},
+    glyph = {'cyan', 'bright'},
+    align_record = {'red'},
+    pseudo_file = {'green'},
+    pseudo_line = {'yellow'},
+    page_insert = {'blue'},
+    split_insert = {'magenta'},
+    expr_stack = {'cyan'},
+    nested_list = {'red'},
+    span = {'green'},
+    attribute = {'yellow'},
+    glue_spec = {'magenta'},
+    attribute_list = {'cyan'},
+    temp = {'magenta'},
+    align_stack = {'red', 'bright'},
+    movement_stack = {'green', 'bright'},
+    if_stack = {'yellow', 'bright'},
+    unhyphenated = {'magenta', 'bright'},
+    hyphenated = {'cyan', 'bright'},
+    delta = {'red'},
+    passive = {'green'},
+    shape = {'yellow'},
+  },
+
+  ---
+  -- [SGR (Select Graphic Rendition) Parameters](https://en.wikipedia.org/wiki/ANSI_escape_code#SGR_parameters)
+  --
+  -- __attributes__
+  --
+  -- | color      |code|
+  -- |------------|----|
+  -- | reset      |  0 |
+  -- | clear      |  0 |
+  -- | bright     |  1 |
+  -- | dim        |  2 |
+  -- | underscore |  4 |
+  -- | blink      |  5 |
+  -- | reverse    |  7 |
+  -- | hidden     |  8 |
+  --
+  -- __foreground__
+  --
+  -- | color      |code|
+  -- |------------|----|
+  -- | black      | 30 |
+  -- | red        | 31 |
+  -- | green      | 32 |
+  -- | yellow     | 33 |
+  -- | blue       | 34 |
+  -- | magenta    | 35 |
+  -- | cyan       | 36 |
+  -- | white      | 37 |
+  --
+  -- __background__
+  --
+  -- | color      |code|
+  -- |------------|----|
+  -- | onblack    | 40 |
+  -- | onred      | 41 |
+  -- | ongreen    | 42 |
+  -- | onyellow   | 43 |
+  -- | onblue     | 44 |
+  -- | onmagenta  | 45 |
+  -- | oncyan     | 46 |
+  -- | onwhite    | 47 |
+  --
+  -- @tparam string color A color name (`black`, `red`, `green`,
+  --   `yellow`, `blue`, `magenta`, `cyan`, `white`).
+  -- @tparam string mode `bright` or `dim`.
+  -- @tparam boolean background Colorize the background not the text.
+  --
+  -- @treturn string
+  color = function(color, mode, background)
+    if options.color ~= 'colored' then
+      return ''
+    end
+
+    local output = ''
+    local code
+
+    if mode == 'bright' then
+      output = format.color_code(1)
+    elseif mode == 'dim' then
+      output = format.color_code(2)
+    end
+
+    if not background then
+      if color == 'reset' then code = 0
+      elseif color == 'red' then code = 31
+      elseif color == 'green' then code = 32
+      elseif color == 'yellow' then code = 33
+      elseif color == 'blue' then code = 34
+      elseif color == 'magenta' then code = 35
+      elseif color == 'cyan' then code = 36
+      else code = 37 end
+    else
+      if color == 'black' then code = 40
+      elseif color == 'red' then code = 41
+      elseif color == 'green' then code = 42
+      elseif color == 'yellow' then code = 43
+      elseif color == 'blue' then code = 44
+      elseif color == 'magenta' then code = 45
+      elseif color == 'cyan' then code = 46
+      elseif color == 'white' then code = 47
+      else code = 40 end
+    end
+    return output .. format.color_code(code)
+  end,
+
+  --- Format a single unicode character.
+  --
+  -- @tparam string char A single input character.
+  --
+  -- @treturn string
+  char = function(char)
+    char = string.format('%s', unicode.utf8.char(char))
+    char = '\'' .. char .. '\''
+    if options.channel == 'tex' then
+      char = format.escape(char)
+    end
+    return char
+  end,
+
+  ---
+  -- @treturn string
+  line = function(length)
+    local output
+    if length == 'long' then
+      output = '------------------------------------------'
+    else
+      output = '-----------------------'
+    end
+      return output .. format.new_line()
+  end,
+
+  ---
+  -- @treturn string
+  branch = function(connection_type, connection_state, last)
+    local c = connection_type
+    local s = connection_state
+    local l = last
+    if c == 'list' and s == 'stop' and l == false then
+      return format.whitespace(2)
+    elseif c == 'field' and s == 'stop' and l == false then
+      return format.whitespace(2)
+    elseif c == 'list' and s == 'continue' and l == false then
+      return '│' .. format.whitespace()
+    elseif c == 'field' and s == 'continue' and l == false then
+      return '║' .. format.whitespace()
+    elseif c == 'list' and s == 'continue' and l == true then
+      return '├─'
+    elseif c == 'field' and s == 'continue' and l == true then
+      return '╠═'
+    elseif c == 'list' and s == 'stop' and l == true then
+      return '└─'
+    elseif c == 'field' and s == 'stop' and l == true then
+      return '╚═'
+    end
+    return ''
+  end,
+}
+
+---
+-- @treturn string
+function template.fill(number, order, field)
+  local output
+  if order ~= nil and order ~= 0 then
+    if field == 'stretch' then
+      output = '+'
+    else
+      output = '-'
+    end
+    return output .. string.format(
+      '%g%s', number / 2^16,
+      template.colored_string(
+        'fi' .. string.rep('l', order - 1),
+        'white',
+        'dim'
+      )
+    )
+  else
+    return template.length(number)
+  end
+end
+
+--- Colorize a text string.
+--
+-- @tparam string text A text string.
+-- @tparam string color A color name (`black`, `red`, `green`,
+--   `yellow`, `blue`, `magenta`, `cyan`, `white`).
+-- @tparam string mode `bright` or `dim`.
+-- @tparam boolean background Colorize the background not the text.
+--
+-- @treturn string
+function template.colored_string(text, color, mode, background)
+  if options.channel == 'tex' then
+    if mode == 'dim' then
+      mode = ''
+    end
+    return '\\textcolor{' ..
+      format.color_tex(color, mode) ..
+      '}{' ..
+      text ..
+      '}'
+  else
+   return template.color(color, mode, background) .. text .. template.color('reset')
+  end
+end
+
+--- Format a scaled point input value into dimension string (`12pt`,
+--  `1cm`)
+--
+-- @tparam number input
+--
+-- @treturn string
+function template.length (input)
+  input = tonumber(input)
+  input = input / tex.sp('1' .. options.unit)
+  return string.format(
+    '%g%s',
+    format.number(input),
+    template.colored_string(options.unit, 'white', 'dim')
+  )
+end
+
+--- Convert a Lua table into a format string.
+--
+-- @tparam table table A table to generate a inline view of.
+--
+-- @treturn string
+function template.table_inline(table)
+  local tex_escape = ''
+  if options.channel == 'tex' then
+    tex_escape = '\\'
+  end
+  if type(table) == 'table' then
+    local output = tex_escape .. '{'
+    local kv_list = ''
+    for key, value in pairs(table) do
+        if type(key) ~= 'numbers' then
+          key = '\'' ..
+            template.colored_string(key, 'cyan', 'dim') .. '\''
+        end
+        kv_list = kv_list .. '[' .. key .. '] = ' ..
+          template.table_inline(value) .. ', '
+    end
+    output = output .. kv_list:gsub(', $', '')
+    return output .. tex_escape .. '}'
+  else
+    return tostring(table)
+  end
+end
+
+--- Format a key value pair (`key: value, `).
+--
+-- @tparam string key A key
+-- @tparam string|number value A value
+-- @tparam string color A color name (`black`, `red`, `green`,
+--   `yellow`, `blue`, `magenta`, `cyan`, `white`).
+--
+-- @treturn string
+function template.key_value(key, value, color)
+  if type(color) ~= 'string' then
+    color = 'yellow'
+  end
+  if options.channel == 'tex' then
+    key = format.underscore(key)
+  end
+  local output = template.colored_string(key .. ':', color)
+  if value then
+    output = output .. ' ' .. value .. ', '
+  end
+  return output
+end
+
+---
+-- @treturn string
+function template.type(type, id)
+  local output
+  if options.channel == 'tex' then
+    output = format.underscore(type)
+  else
+    output = type
+  end
+  output = string.upper(output)
+  if options.verbosity > 1 then
+    output = output .. format.type_id(id)
+  end
+  return template.colored_string(
+    output .. format.whitespace(),
+    template.node_colors[type][1],
+    template.node_colors[type][2]
+  )
+end
+
+---
+-- @treturn string
+function template.callback(callback_name, variables)
+  nodetree_print(
+    format.new_line(2) ..
+    'Callback: ' ..
+    template.colored_string(format.underscore(callback_name), 'red', '', true) ..
+    format.new_line()
+  )
+  if variables then
+    for name, value in pairs(variables) do
+      if value ~= nil and value ~= '' then
+        nodetree_print(
+          '- ' ..
+          format.underscore(name) ..
+          ': ' ..
+          tostring(value) ..
+          format.new_line()
+        )
+      end
+    end
+  end
+  nodetree_print(template.line('long'))
+end
+
+---
+-- @treturn string
+function template.branches(level, connection_type)
+  local output = ''
+  for i = 1, level - 1  do
+    output = output .. template.branch('list', tree_state[i]['list'], false)
+    output = output .. template.branch('field', tree_state[i]['field'], false)
+  end
+-- Format the last branches
+  if connection_type == 'list' then
+    output = output .. template.branch('list', tree_state[level]['list'], true)
+  else
+    output = output .. template.branch('list', tree_state[level]['list'], false)
+    output = output .. template.branch('field', tree_state[level]['field'], true)
+  end
+  return output
+end
+
+--- Extend the node library
+-- @section node_extended
+
+local node_extended = {}
+
+--- Get the ID of a node.
+--
+-- We have to convert the node into a string and than have to extract
+-- the ID from this string using a regular expression. If you convert a
+-- node into a string it looks like: `<node    nil <    172 >    nil :
+-- hlist 2>`.
+--
+-- @tparam node n A node.
+--
+-- @treturn string
+function node_extended.node_id(n)
   return string.gsub(tostring(n), '^<node%s+%S+%s+<%s+(%d+).*', '%1')
 end
-function nodex.subtype(n)
-  local typ = node.type(n.id)
-  local subtypes = {
+
+--- A table of all node subtype names.
+--
+-- __Nodes without subtypes:__
+--
+-- * `ins` (3)
+-- * `mark` (4)
+-- * `whatsit` (8)
+-- * `local_par` (9)
+-- * `dir` (10)
+-- * `penalty` (14)
+-- * `unset` (15)
+-- * `style` (16)
+-- * `choice` (17)
+-- * `fraction` (20)
+-- * `math_char` (23)
+-- * `sub_box` (24)
+-- * `sub_mlist` (25)
+-- * `math_text_char` (26)
+-- * `delim` (27)
+-- * `margin_kern` (28)
+-- * `align_record` (30)
+-- * `pseudo_file` (31)
+-- * `pseudo_line` (32)
+-- * `page_insert` (33)
+-- * `split_insert` (34)
+-- * `expr_stack` (35)
+-- * `nested_list` (36)
+-- * `span` (37)
+-- * `attribute` (38)
+-- * `glue_spec` (39)
+-- * `attribute_list` (40)
+-- * `temp` (41)
+-- * `align_stack` (42)
+-- * `movement_stack` (43)
+-- * `if_stack` (44)
+-- * `unhyphenated` (45)
+-- * `hyphenated` (46)
+-- * `delta` (47)
+-- * `passive` (48)
+-- * `shape` (49)
+--
+-- @treturn table
+local function get_node_subtypes ()
+    local subtypes = {
+    -- hlist (0)
     hlist = {
       [0] = 'unknown',
       [1] = 'line',
@@ -20,23 +636,53 @@
       [5] = 'cell',
       [6] = 'equation',
       [7] = 'equationnumber',
+      [8] = 'math',
+      [9] = 'mathchar',
+      [10] = 'hextensible',
+      [11] = 'vextensible',
+      [12] = 'hdelimiter',
+      [13] = 'vdelimiter',
+      [14] = 'overdelimiter',
+      [15] = 'underdelimiter',
+      [16] = 'numerator',
+      [17] = 'denominator',
+      [18] = 'limits',
+      [19] = 'fraction',
+      [20] = 'nucleus',
+      [21] = 'sup',
+      [22] = 'sub',
+      [23] = 'degree',
+      [24] = 'scripts',
+      [25] = 'over',
+      [26] = 'under',
+      [27] = 'accent',
+      [28] = 'radical',
     },
+    -- vlist (1)
     vlist = {
       [0] = 'unknown',
       [4] = 'alignment',
       [5] = 'cell',
     },
+    -- rule (2)
     rule = {
-      [0] = 'unknown',
+      [0] = 'normal',
       [1] = 'box',
       [2] = 'image',
       [3] = 'empty',
       [4] = 'user',
+      [5] = 'over',
+      [6] = 'under',
+      [7] = 'fraction',
+      [8] = 'radical',
+      [9] = 'outline',
     },
+    -- adjust (5)
     adjust = {
       [0] = 'normal',
       [1] = 'pre',
     },
+    -- boundary (6)
     boundary = {
       [0] = 'cancel',
       [1] = 'user',
@@ -43,6 +689,7 @@
       [2] = 'protrusion',
       [3] = 'word',
     },
+    -- disc (7)
     disc  = {
       [0] = 'discretionary',
       [1] = 'explicit',
@@ -51,10 +698,12 @@
       [4] = 'first',
       [5] = 'second',
     },
+    -- math (11)
     math = {
       [0] = 'beginmath',
       [1] = 'endmath',
     },
+    -- glue (12)
     glue = {
       [0]   = 'userskip',
       [1]   = 'lineskip',
@@ -83,6 +732,7 @@
       [102] = 'xleaders',
       [103] = 'gleaders',
     },
+    -- kern (13)
     kern = {
       [0] = 'fontkern',
       [1] = 'userkern',
@@ -89,6 +739,18 @@
       [2] = 'accentkern',
       [3] = 'italiccorrection',
     },
+    -- penalty (14)
+    penalty = {
+      [0] = 'userpenalty',
+      [1] = 'linebreakpenalty',
+      [2] = 'linepenalty',
+      [3] = 'wordpenalty',
+      [4] = 'finalpenalty',
+      [5] = 'noadpenalty',
+      [6] = 'beforedisplaypenalty',
+      [7] = 'afterdisplaypenalty',
+      [8] = 'equationnumberpenalty',
+    },
     noad = {
       [0] = 'ord',
       [1] = 'opdisplaylimits',
@@ -104,6 +766,7 @@
       [11] = 'over',
       [12] = 'vcenter',
     },
+    -- radical (19)
     radical = {
       [0] = 'radical',
       [1] = 'uradical',
@@ -113,6 +776,7 @@
       [5] = 'udelimiterunder',
       [6] = 'udelimiterover',
     },
+    -- accent (21)
     accent = {
       [0] = 'bothflexible',
       [1] = 'fixedtop',
@@ -119,12 +783,20 @@
       [2] = 'fixedbottom',
       [3] = 'fixedboth',
     },
+    -- fence (22)
     fence = {
       [0] = 'unset',
       [1] = 'left',
       [2] = 'middle',
       [3] = 'right',
+      [4] = 'no',
     },
+    -- margin_kern (28)
+    margin_kern = {
+      [0] = 'left',
+      [1] = 'right',
+    },
+    -- glyph (29)
     glyph = {
       [0] = 'character',
       [1] = 'ligature',
@@ -134,232 +806,45 @@
     },
   }
   subtypes.whatsit = node.whatsits()
-  local out = ''
+  return subtypes
+end
+
+---
+-- @treturn string
+function node_extended.subtype(n)
+  local typ = node.type(n.id)
+  local subtypes = get_node_subtypes()
+
+  local output
   if subtypes[typ] and subtypes[typ][n.subtype] then
-    out = subtypes[typ][n.subtype]
+    output = subtypes[typ][n.subtype]
     if options.verbosity > 1 then
-      out = out .. tpl.type_id(n.subtype)
+      output = output .. format.type_id(n.subtype)
     end
-    return out
+    return output
   else
     return tostring(n.subtype)
   end
-  assert(false)
 end
-function tpl.round(number)
-  local mult = 10^(options.decimalplaces or 0)
-  return math.floor(number * mult + 0.5) / mult
-end
-function tpl.length(input)
-  input = tonumber(input)
-  input = input / tex.sp('1' .. options.unit)
-  return string.format('%g%s', tpl.round(input), options.unit)
-end
-function tpl.fill(number, order, field)
-  if order ~= nil and order ~= 0 then
-    if field == 'stretch' then
-      out = '+'
-    else
-      out = '-'
-    end
-    return out .. string.format(
-      '%gfi%s', number / 2^16,
-      string.rep('l', order - 1)
-    )
-  else
-    return tpl.length(number)
-  end
-end
-tpl.node_colors = {
-  hlist = {'red', 'bright'},
-  vlist = {'green', 'bright'},
-  rule = {'blue', 'bright'},
-  ins = {'blue'},
-  mark = {'magenta'},
-  adjust = {'cyan'},
-  boundary = {'red', 'bright'},
-  disc = {'green', 'bright'},
-  whatsit = {'yellow', 'bright'},
-  local_par = {'blue', 'bright'},
-  dir = {'magenta', 'bright'},
-  math = {'cyan', 'bright'},
-  glue = {'magenta', 'bright'},
-  kern = {'green', 'bright'},
-  penalty = {'yellow', 'bright'},
-  unset = {'blue'},
-  style = {'magenta'},
-  choice = {'cyan'},
-  noad = {'red'},
-  radical = {'green'},
-  fraction = {'yellow'},
-  accent = {'blue'},
-  fence = {'magenta'},
-  math_char = {'cyan'},
-  sub_box = {'red', 'bright'},
-  sub_mlist = {'green', 'bright'},
-  math_text_char = {'yellow', 'bright'},
-  delim = {'blue', 'bright'},
-  margin_kern = {'magenta', 'bright'},
-  glyph = {'cyan', 'bright'},
-  align_record = {'red'},
-  pseudo_file = {'green'},
-  pseudo_line = {'yellow'},
-  page_insert = {'blue'},
-  split_insert = {'magenta'},
-  expr_stack = {'cyan'},
-  nested_list = {'red'},
-  span = {'green'},
-  attribute = {'yellow'},
-  glue_spec = {'magenta'},
-  attribute_list = {'cyan'},
-  temp = {'magenta'},
-  align_stack = {'red', 'bright'},
-  movement_stack = {'green', 'bright'},
-  if_stack = {'yellow', 'bright'},
-  unhyphenated = {'magenta', 'bright'},
-  hyphenated = {'cyan', 'bright'},
-  delta = {'red'},
-  passive = {'green'},
-  shape = {'yellow'},
-}
-function tpl.color_code(code)
-  return string.char(27) .. '[' .. tostring(code) .. 'm'
-end
-function tpl.color(color, mode, background)
-  if options.color ~= 'colored' then
-    return ''
-  end
-  local out = ''
-  local code = ''
-  if mode == 'bright' then
-    out = tpl.color_code(1)
-  elseif mode == 'dim' then
-    out = tpl.color_code(2)
-  end
-  if not background then
-    if color == 'reset' then code = 0
-    elseif color == 'red' then code = 31
-    elseif color == 'green' then code = 32
-    elseif color == 'yellow' then code = 33
-    elseif color == 'blue' then code = 34
-    elseif color == 'magenta' then code = 35
-    elseif color == 'cyan' then code = 36
-    else code = 37 end
-  else
-    if color == 'black' then code = 40
-    elseif color == 'red' then code = 41
-    elseif color == 'green' then code = 42
-    elseif color == 'yellow' then code = 43
-    elseif color == 'blue' then code = 44
-    elseif color == 'magenta' then code = 45
-    elseif color == 'cyan' then code = 46
-    elseif color == 'white' then code = 47
-    else code = 40 end
-  end
-  return out .. tpl.color_code(code)
-end
-function tpl.key_value(key, value)
-  local out = tpl.color('yellow') .. key .. ': '
-  if value then
-    out = out .. tpl.color('white') .. value .. '; '
-  end
-  return out .. tpl.color('reset')
-end
-function tpl.char(input)
-  return string.format('%q', unicode.utf8.char(input))
-end
-function tpl.type(type, id)
-  local out = tpl.color(
-    tpl.node_colors[type][1],
-    tpl.node_colors[type][2]
-    )
-    .. string.upper(type)
-  if options.verbosity > 1 then
-    out = out .. tpl.type_id(id)
-  end
-  return out .. tpl.color('reset')  .. ' '
-end
-function tpl.callback_variable(variable_name, variable)
-  if variable ~= nil and variable ~= '' then
-    tpl.print(variable_name .. ': ' .. tostring(variable))
-  end
-end
-function tpl.line(length)
-  if length == 'long' then
-    return '------------------------------------------'
-  else
-    return '-----------------------'
-  end
-end
-function tpl.callback(callback_name, variables)
-  tpl.print('\n\n')
-  tpl.print('Callback: ' .. tpl.color('red', '', true) ..
-    callback_name .. tpl.color('reset')
-  )
-  if variables then
-    for name, value in pairs(variables) do
-      if value ~= nil and value ~= '' then
-        tpl.print('  - ' .. name .. ': ' .. tostring(value))
-      end
-    end
-  end
-  tpl.print(tpl.line('long'))
-end
-function tpl.type_id(id)
-  return '[' .. tostring(id) .. ']'
-end
-function tpl.branch(connection_type, connection_state, last)
-  local c = connection_type
-  local s = connection_state
-  local l = last
-  if c == 'list' and s == 'stop' and l == false then
-    return '  '
-  elseif c == 'field' and s == 'stop' and l == false then
-    return '  '
-  elseif c == 'list' and s == 'continue' and l == false then
-    return '│ '
-  elseif c == 'field' and s == 'continue' and l == false then
-    return '║ '
-  elseif c == 'list' and s == 'continue' and l == true then
-    return '├─'
-  elseif c == 'field' and s == 'continue' and l == true then
-    return '╠═'
-  elseif c == 'list' and s == 'stop' and l == true then
-    return '└─'
-  elseif c == 'field' and s == 'stop' and l == true then
-    return '╚═'
-  end
-end
-function tpl.branches(level, connection_type)
-  local out = ''
-  for i = 1, level - 1  do
-    out = out .. tpl.branch('list', tree.state[i]['list'], false)
-    out = out .. tpl.branch('field', tree.state[i]['field'], false)
-  end
-  if connection_type == 'list' then
-    out = out .. tpl.branch('list', tree.state[level]['list'], true)
-  else
-    out = out .. tpl.branch('list', tree.state[level]['list'], false)
-    out = out .. tpl.branch('field', tree.state[level]['field'], true)
-  end
-  return out
-end
-function tpl.print(text)
 
-  if options.channel == 'log' then
-    if not log then
-      log = io.open(tex.jobname .. '_nodetree.log', 'a')
-    end
-    log:write(text, '\n')
-  else
-    print('  ' .. text)
-  end
-end
+--- Build the node tree.
+-- @section tree
+
+local tree = {}
+
+---
+-- @tparam node head
+-- @tparam string field
+--
+-- @treturn string
 function tree.format_field(head, field)
-  local out = ''
-  if not head[field] or head[field] == 0 then
+  local output
+-- Character "0" should be printed in a tree, because in TeX fonts the
+-- 0 slot usually has a symbol.
+  if not head[field] or (head[field] == 0 and field ~= "char") then
     return ''
   end
+
   if options.verbosity < 2 and
     -- glyph
     field == 'font' or
@@ -381,10 +866,11 @@
   then
     return ''
   end
+
   if field == 'prev' or field == 'next' then
-    out = nodex.node_id(head[field])
+    output = node_extended.node_id(head[field])
   elseif field == 'subtype' then
-    out = nodex.subtype(head)
+    output = format.underscore(node_extended.subtype(head))
   elseif
     field == 'width' or
     field == 'height' or
@@ -391,27 +877,62 @@
     field == 'depth' or
     field == 'kern' or
     field == 'shift' then
-    out = tpl.length(head[field])
+    output = template.length(head[field])
   elseif field == 'char' then
-    out = tpl.char(head[field])
+    output = template.char(head[field])
   elseif field == 'glue_set' then
-    out = tpl.round(head[field])
+    output = format.number(head[field])
   elseif field == 'stretch' or field == 'shrink' then
-    out = tpl.fill(head[field], head[field .. '_order'], field)
+    output = template.fill(head[field], head[field .. '_order'], field)
   else
-    out = tostring(head[field])
+    output = tostring(head[field])
   end
-  return tpl.key_value(field, out)
+
+  return template.key_value(field, output)
 end
+
+---
+-- Attributes are key/value number pairs. They are printed as an inline
+-- list. The attribute `0` with the value `0` is skipped because this
+-- attribute is in every node by default.
+--
+-- @tparam node head
+--
+-- @treturn string
+function tree.format_attributes(head)
+  if not head then
+    return ''
+  end
+  local output = ''
+  local attr = head.next
+  while attr do
+    if attr.number ~= 0 or (attr.number == 0 and attr.value ~= 0) then
+      output = output .. tostring(attr.number) .. '=' .. tostring(attr.value) .. ' '
+    end
+    attr = attr.next
+  end
+  return output
+end
+
+---
+-- @tparam number level `level` is a integer beginning with 1.
+-- @tparam number connection_type The variable `connection_type`
+--   is a string, which can be either `list` or `field`.
+-- @tparam connection_state `connection_state` is a string, which can
+--   be either `continue` or `stop`.
 function tree.set_state(level, connection_type, connection_state)
-  if not tree.state[level] then
-    tree.state[level] = {}
+  if not tree_state[level] then
+    tree_state[level] = {}
   end
-  tree.state[level][connection_type] = connection_state
+  tree_state[level][connection_type] = connection_state
 end
+
+---
+-- @tparam table fields
+-- @tparam number level
 function tree.analyze_fields(fields, level)
   local max = 0
-  local connection_state = ''
+  local connection_state
   for _ in pairs(fields) do
     max = max + 1
   end
@@ -424,13 +945,23 @@
       connection_state = 'continue'
     end
     tree.set_state(level, 'field', connection_state)
-    tpl.print(tpl.branches(level, 'field') .. tpl.key_value(field_name))
+    nodetree_print(
+      format.node_begin() ..
+      template.branches(level, 'field') ..
+      template.key_value(field_name) ..
+      format.node_end() ..
+      format.new_line()
+    )
     tree.analyze_list(recursion_node, level + 1)
   end
 end
+
+---
+-- @tparam node head
+-- @tparam number level
 function tree.analyze_node(head, level)
   local connection_state
-  local out = ''
+  local output
   if head.next then
     connection_state = 'continue'
   else
@@ -437,24 +968,67 @@
     connection_state = 'stop'
   end
   tree.set_state(level, 'list', connection_state)
-  out = tpl.branches(level, 'list')
-    .. tpl.type(node.type(head.id), head.id)
+  output = template.branches(level, 'list')
+    .. template.type(node.type(head.id), head.id)
   if options.verbosity > 1 then
-    out = out .. tpl.key_value('no', nodex.node_id(head))
+    output = output .. template.key_value('no', node_extended.node_id(head))
   end
+
+  -- We store the attributes output to append it to the field list.
+  local attributes
+
+  -- We store fields which are nodes for later treatment.
   local fields = {}
-  for field_id, field_name in pairs(node.fields(head.id, head.subtype)) do
-    if field_name ~= 'next' and
-      field_name ~= 'prev' and
+
+  -- Inline fields, for example: char: 'm', width: 25pt, height: 13.33pt,
+  local output_fields = ''
+  for _, field_name in pairs(node.fields(head.id, head.subtype)) do
+    if field_name == 'attr' then
+      attributes = tree.format_attributes(head.attr)
+    elseif field_name ~= 'next' and field_name ~= 'prev' and
       node.is_node(head[field_name]) then
       fields[field_name] = head[field_name]
     else
-      out = out .. tree.format_field(head, field_name)
+      output_fields = output_fields .. tree.format_field(head, field_name)
     end
   end
-  tpl.print(out)
+  if output_fields ~= '' then
+    output = output .. output_fields
+  end
+
+  -- Append the attributes output if available
+  if attributes ~= '' then
+    output = output .. template.key_value('attr', attributes, 'blue')
+  end
+
+  output = output:gsub(', $', '')
+
+  nodetree_print(
+    format.node_begin() ..
+    output ..
+    format.node_end() ..
+    format.new_line()
+  )
+
+  local property = node.getproperty(head)
+  if property then
+    nodetree_print(
+      format.node_begin() ..
+      template.branches(level, 'field') ..
+      '  ' ..
+      template.colored_string('properties:', 'blue') .. ' ' ..
+      template.table_inline(property) ..
+      format.node_end() ..
+      format.new_line()
+    )
+  end
+
   tree.analyze_fields(fields, level)
 end
+
+---
+-- @tparam node head
+-- @tparam number level
 function tree.analyze_list(head, level)
   while head do
     tree.analyze_node(head, level)
@@ -461,256 +1035,513 @@
     head = head.next
   end
 end
+
+---
+-- @tparam node head
 function tree.analyze_callback(head)
   tree.analyze_list(head, 1)
-  tpl.print(tpl.line('short') .. '\n')
+  nodetree_print(template.line('short') .. format.new_line())
 end
-function callbacks.contribute_filter(extrainfo)
-  tpl.callback('contribute_filter', {extrainfo = extrainfo})
-  return true
-end
-function callbacks.buildpage_filter(extrainfo)
-  tpl.callback('buildpage_filter', {extrainfo = extrainfo})
-  return true
-end
-function callbacks.pre_linebreak_filter(head, groupcode)
-  tpl.callback('pre_linebreak_filter', {groupcode = groupcode})
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.linebreak_filter(head, is_display)
-  tpl.callback('linebreak_filter', {is_display = is_display})
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.append_to_vlist_filter(head, locationcode, prevdepth, mirrored)
-  local variables = {
-    locationcode = locationcode,
-    prevdepth = prevdepth,
-    mirrored = mirrored,
-  }
-  tpl.callback('append_to_vlist_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.post_linebreak_filter(head, groupcode)
-  tpl.callback('post_linebreak_filter', {groupcode = groupcode})
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.hpack_filter(head, groupcode, size, packtype, direction, attributelist)
-  local variables = {
-    groupcode = groupcode,
-    size = size,
-    packtype = packtype,
-    direction = direction,
-    attributelist = attributelist,
-  }
-  tpl.callback('hpack_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.vpack_filter(head, groupcode, size, packtype, maxdepth, direction, attributelist)
-  local variables = {
-    groupcode = groupcode,
-    size = size,
-    packtype = packtype,
-    maxdepth = tpl.length(maxdepth),
-    direction = direction,
-    attributelist = attributelist,
-  }
-  tpl.callback('vpack_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.hpack_quality(incident, detail, head, first, last)
-  local variables = {
-    incident = incident,
-    detail = detail,
-    first = first,
-    last = last,
-  }
-  tpl.callback('hpack_quality', variables)
-  tree.analyze_callback(head)
-end
-function callbacks.vpack_quality(incident, detail, head, first, last)
-  local variables = {
-    incident = incident,
-    detail = detail,
-    first = first,
-    last = last,
-  }
-  tpl.callback('vpack_quality', variables)
-  tree.analyze_callback(head)
-end
-function callbacks.process_rule(head, width, height)
-  local variables = {
-    width = width,
-    height = height,
-  }
-  tpl.callback('process_rule', variables)
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.pre_output_filter(head, groupcode, size, packtype, maxdepth, direction)
-  local variables = {
-    groupcode = groupcode,
-    size = size,
-    packtype = packtype,
-    maxdepth = maxdepth,
-    direction = direction,
-  }
-  tpl.callback('pre_output_filter', variables)
-  tree.analyze_callback(head)
-  return true
-end
-function callbacks.hyphenate(head, tail)
-  tpl.callback('hyphenate')
-  tpl.print('head:')
-  tree.analyze_callback(head)
-  tpl.print('tail:')
-  tree.analyze_callback(tail)
-end
-function callbacks.ligaturing(head, tail)
-  tpl.callback('ligaturing')
-  tpl.print('head:')
-  tree.analyze_callback(head)
-  tpl.print('tail:')
-  tree.analyze_callback(tail)
-end
-function callbacks.kerning(head, tail)
-  tpl.callback('kerning')
-  tpl.print('head:')
-  tree.analyze_callback(head)
-  tpl.print('tail:')
-  tree.analyze_callback(tail)
-end
-function callbacks.insert_local_par(local_par, location)
-  tpl.callback('insert_local_par', {location = location})
-  tree.analyze_callback(local_par)
-  return true
-end
-function callbacks.mlist_to_hlist(head, display_type, need_penalties)
-  local variables = {
-    display_type = display_type,
-    need_penalties = need_penalties,
-  }
-  tpl.callback('mlist_to_hlist', variables)
-  tree.analyze_callback(head)
-  return node.mlist_to_hlist(head, display_type, need_penalties)
-end
-function base.normalize_options()
-  options.verbosity = tonumber(options.verbosity)
-  options.decimalplaces = tonumber(options.decimalplaces)
-end
-function base.set_default_options()
-  local defaults = {
-    verbosity = 1,
-    callback = 'postlinebreak',
-    engine = 'luatex',
-    color = 'colored',
-    decimalplaces = 2,
-    unit = 'pt',
-    channel = 'term',
-  }
+
+--- Callback wrapper.
+-- @section callbacks
+
+local callbacks = {
+
+  ---
+  -- @tparam string extrainfo
+  contribute_filter = function(extrainfo)
+    template.callback('contribute_filter', {extrainfo = extrainfo})
+    return true
+  end,
+
+  ---
+  -- @tparam string extrainfo
+  buildpage_filter = function(extrainfo)
+    template.callback('buildpage_filter', {extrainfo = extrainfo})
+    return true
+  end,
+
+  ---
+  -- @tparam string n
+  -- @tparam string i
+  build_page_insert = function(n, i)
+    print('lol')
+    template.callback('build_page_insert', {n = n, i = i})
+    return 0
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam string groupcode
+  pre_linebreak_filter = function(head, groupcode)
+    template.callback('pre_linebreak_filter', {groupcode = groupcode})
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam boolean is_display
+  linebreak_filter = function(head, is_display)
+    template.callback('linebreak_filter', {is_display = is_display})
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam node box
+  -- @tparam string locationcode
+  -- @tparam number prevdepth
+  -- @tparam boolean mirrored
+  append_to_vlist_filter = function(box, locationcode, prevdepth, mirrored)
+    local variables = {
+      locationcode = locationcode,
+      prevdepth = prevdepth,
+      mirrored = mirrored,
+    }
+    template.callback('append_to_vlist_filter', variables)
+    tree.analyze_callback(box)
+    return box
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam string groupcode
+  post_linebreak_filter = function(head, groupcode)
+    template.callback('post_linebreak_filter', {groupcode = groupcode})
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam string groupcode
+  -- @tparam number size
+  -- @tparam string packtype
+  -- @tparam string direction
+  -- @tparam node attributelist
+  hpack_filter = function(head, groupcode, size, packtype, direction, attributelist)
+    local variables = {
+      groupcode = groupcode,
+      size = size,
+      packtype = packtype,
+      direction = direction,
+      attributelist = attributelist,
+    }
+    template.callback('hpack_filter', variables)
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam string groupcode
+  -- @tparam number size
+  -- @tparam string packtype
+  -- @tparam number maxdepth
+  -- @tparam string direction
+  -- @tparam node attributelist
+  vpack_filter = function(head, groupcode, size, packtype, maxdepth, direction, attributelist)
+    local variables = {
+      groupcode = groupcode,
+      size = size,
+      packtype = packtype,
+      maxdepth = template.length(maxdepth),
+      direction = direction,
+      attributelist = attributelist,
+    }
+    template.callback('vpack_filter', variables)
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam string incident
+  -- @tparam number detail
+  -- @tparam node head
+  -- @tparam number first
+  -- @tparam number last
+  hpack_quality = function(incident, detail, head, first, last)
+    local variables = {
+      incident = incident,
+      detail = detail,
+      first = first,
+      last = last,
+    }
+    template.callback('hpack_quality', variables)
+    tree.analyze_callback(head)
+  end,
+
+  ---
+  -- @tparam string incident
+  -- @tparam number detail
+  -- @tparam node head
+  -- @tparam number first
+  -- @tparam number last
+  vpack_quality = function(incident, detail, head, first, last)
+    local variables = {
+      incident = incident,
+      detail = detail,
+      first = first,
+      last = last,
+    }
+    template.callback('vpack_quality', variables)
+    tree.analyze_callback(head)
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam number width
+  -- @tparam number height
+  process_rule = function(head, width, height)
+    local variables = {
+      width = width,
+      height = height,
+    }
+    template.callback('process_rule', variables)
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam string groupcode
+  -- @tparam number size
+  -- @tparam string packtype
+  -- @tparam number maxdepth
+  -- @tparam string direction
+  pre_output_filter = function(head, groupcode, size, packtype, maxdepth, direction)
+    local variables = {
+      groupcode = groupcode,
+      size = size,
+      packtype = packtype,
+      maxdepth = maxdepth,
+      direction = direction,
+    }
+    template.callback('pre_output_filter', variables)
+    tree.analyze_callback(head)
+    return true
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam node tail
+  hyphenate = function(head, tail)
+    template.callback('hyphenate')
+    nodetree_print('head:')
+    tree.analyze_callback(head)
+    nodetree_print('tail:')
+    tree.analyze_callback(tail)
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam node tail
+  ligaturing = function(head, tail)
+    template.callback('ligaturing')
+    nodetree_print('head:')
+    tree.analyze_callback(head)
+    nodetree_print('tail:')
+    tree.analyze_callback(tail)
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam node tail
+  kerning = function(head, tail)
+    template.callback('kerning')
+    nodetree_print('head:')
+    tree.analyze_callback(head)
+    nodetree_print('tail:')
+    tree.analyze_callback(tail)
+  end,
+
+  ---
+  -- @tparam node local_par
+  -- @tparam string location
+  insert_local_par = function(local_par, location)
+    template.callback('insert_local_par', {location = location})
+    tree.analyze_callback(local_par)
+    return true
+  end,
+
+  ---
+  -- @tparam node head
+  -- @tparam string display_type
+  -- @tparam boolean need_penalties
+  mlist_to_hlist = function(head, display_type, need_penalties)
+    local variables = {
+      display_type = display_type,
+      need_penalties = need_penalties,
+    }
+    template.callback('mlist_to_hlist', variables)
+    tree.analyze_callback(head)
+    return node.mlist_to_hlist(head, display_type, need_penalties)
+  end,
+}
+
+--- Set a single option key value pair.
+--
+-- @tparam string key The key of the option pair.
+-- @tparam number|string value The value of the option pair.
+local function set_option(key, value)
   if not options then
     options = {}
   end
-  for key, value in pairs(defaults) do
-    if not options[key] then
-      options[key] = value
-    end
+  if key == 'verbosity' or key == 'decimalplaces' then
+    options[key] = tonumber(value)
+  else
+    options[key] = value
   end
-  base.normalize_options()
 end
-function base.set_option(key, value)
+
+--- Set multiple key value pairs using a table.
+--
+-- @tparam table opts Options
+local function set_options(opts)
   if not options then
     options = {}
   end
-  options[key] = value
-  base.normalize_options()
+  for key, value in pairs(opts) do
+    set_option(key, value)
+  end
 end
-function base.get_option(key)
-  if not options then
-    options = {}
+
+--- Check if the given callback name exists.
+--
+-- Throw an error if it doen’t.
+--
+-- @tparam string callback_name The name of a callback to check.
+--
+-- @treturn string The unchanged input of the function.
+local function check_callback_name(callback_name)
+  local info = callback.list()
+  if info[callback_name] == nil then
+    tex.error(
+      'Package "nodetree": Unkown callback name or callback alias: "' ..
+      callback_name ..
+      '"'
+    )
   end
-  if options[key] then
-    return options[key]
-  end
+  return callback_name
 end
-function base.get_callback_name(alias)
+
+--- Get the real callback name from an alias string.
+--
+-- @tparam string alias The alias of a callback name or the callback
+-- name itself.
+--
+-- @treturn string The real callback name.
+local function get_callback_name(alias)
+  local callback_name
+  -- Listed as in the LuaTeX reference manual.
   if alias == 'contribute' or alias == 'contributefilter' then
-    return 'contribute_filter'
-  elseif alias == 'buildpage' or alias == 'buildpagefilter' then
-    return 'buildpage_filter'
+    callback_name = 'contribute_filter'
+
+  -- Formerly called buildpage, now there is a build_page_insert.
+  elseif alias == 'buildfilter' or alias == 'buildpagefilter' then
+    callback_name = 'buildpage_filter'
+
+  -- Untested: I don’t know how to invoke this filter.
+  elseif alias == 'buildinsert' or alias == 'buildpageinsert' then
+    callback_name = 'build_page_insert'
+
   elseif alias == 'preline' or alias == 'prelinebreakfilter' then
-    return 'pre_linebreak_filter'
+    callback_name = 'pre_linebreak_filter'
+
   elseif alias == 'line' or alias == 'linebreakfilter' then
-    return 'linebreak_filter'
+    callback_name = 'linebreak_filter'
+
   elseif alias == 'append' or alias == 'appendtovlistfilter' then
-    return 'append_to_vlist_filter'
-  elseif alias == 'postline' or alias == 'postlinebreakfilter' then
-    return 'post_linebreak_filter'
+    callback_name = 'append_to_vlist_filter'
+
+  -- postlinebreak is not documented.
+  elseif alias == 'postline' or alias == 'postlinebreak' or alias == 'postlinebreakfilter' then
+    callback_name = 'post_linebreak_filter'
+
   elseif alias == 'hpack' or alias == 'hpackfilter' then
-    return 'hpack_filter'
+    callback_name = 'hpack_filter'
+
   elseif alias == 'vpack' or alias == 'vpackfilter' then
-    return 'vpack_filter'
+    callback_name = 'vpack_filter'
+
   elseif alias == 'hpackq' or alias == 'hpackquality' then
-    return 'hpack_quality'
+    callback_name = 'hpack_quality'
+
   elseif alias == 'vpackq' or alias == 'vpackquality' then
-    return 'vpack_quality'
+    callback_name = 'vpack_quality'
+
   elseif alias == 'process' or alias == 'processrule' then
-    return 'process_rule'
+    callback_name = 'process_rule'
+
   elseif alias == 'preout' or alias == 'preoutputfilter' then
-    return 'pre_output_filter'
+    callback_name = 'pre_output_filter'
+
   elseif alias == 'hyph' or alias == 'hyphenate' then
-    return 'hyphenate'
+    callback_name = 'hyphenate'
+
   elseif alias == 'liga' or alias == 'ligaturing' then
-    return 'ligaturing'
+    callback_name = 'ligaturing'
+
   elseif alias == 'kern' or alias == 'kerning' then
-   return 'kerning'
+    callback_name = 'kerning'
+
   elseif alias == 'insert' or alias == 'insertlocalpar' then
-    return 'insert_local_par'
+    callback_name = 'insert_local_par'
+
   elseif alias == 'mhlist' or alias == 'mlisttohlist' then
-    return 'mlist_to_hlist'
+    callback_name = 'mlist_to_hlist'
+
   else
-    return 'post_linebreak_filter'
+    callback_name = alias
   end
+  return check_callback_name(callback_name)
 end
-function base.register(cb)
+
+--- Register a callback.
+--
+-- @tparam string cb The name of a callback.
+local function register_callback(cb)
   if options.engine == 'lualatex' then
     luatexbase.add_to_callback(cb, callbacks[cb], 'nodetree')
   else
-    id, error = callback.register(cb, callbacks[cb])
+    callback.register(cb, callbacks[cb])
   end
 end
-function base.register_callbacks()
-  for alias in string.gmatch(options.callback, '([^,]+)') do
-    base.register(base.get_callback_name(alias))
-  end
-end
-function base.unregister(cb)
+
+--- Unregister a callback.
+--
+-- @tparam string cb The name of a callback.
+local function unregister_callback(cb)
   if options.engine == 'lualatex' then
     luatexbase.remove_from_callback(cb, 'nodetree')
   else
-    id, error = callback.register(cb, nil)
+    register_callback(cb, nil)
   end
 end
-function base.unregister_callbacks()
-  for alias in string.gmatch(options.callback, '([^,]+)') do
-    base.unregister(base.get_callback_name(alias))
+
+--- Exported functions.
+-- @section export
+
+local export = {
+  set_option = set_option,
+  set_options = set_options,
+
+  ---
+  register_callbacks = function()
+    if options.channel == 'log' or options.channel == 'tex' then
+      -- nt = nodetree
+      -- jobname.nttex
+      -- jobname.ntlog
+      local file_name = tex.jobname .. '.nt' .. options.channel
+      io.open(file_name, 'w'):close() -- Clear former content
+      output_file = io.open(file_name, 'a')
+    end
+    for alias in string.gmatch(options.callback, '([^,]+)') do
+      register_callback(get_callback_name(alias))
+    end
+  end,
+
+  ---
+  unregister_callbacks = function()
+    for alias in string.gmatch(options.callback, '([^,]+)') do
+      unregister_callback(get_callback_name(alias))
+    end
+  end,
+
+  --- Compile a TeX snippet.
+  --
+  -- Write some TeX snippets into a temporary LaTeX file, compile this
+  -- file using `latexmk` and read the generated `*.nttex` file and
+  -- return its content.
+  --
+  -- @tparam string tex_markup
+  --
+  -- @treturn string
+  compile_include = function(tex_markup)
+    -- Generate a subfolder for all tempory files: _nodetree-jobname.
+    local parent_path = lfs.currentdir() .. '/' .. '_nodetree-' .. tex.jobname
+    lfs.mkdir(parent_path)
+
+    -- Generate the temporary LuaTeX or LuaLaTeX file.
+    example_counter = example_counter + 1
+    local filename_tex = example_counter .. '.tex'
+    local absolute_path_tex = parent_path .. '/' .. filename_tex
+    output_file = io.open(absolute_path_tex, 'w')
+
+    local format_option = function (key, value)
+      return '\\NodetreeSetOption[' .. key .. ']{' .. value .. '}' .. '\n'
+    end
+
+    -- Process the options
+    local options =
+      format_option('channel', 'tex') ..
+      format_option('verbosity', options.verbosity) ..
+      format_option('unit', options.unit) ..
+      format_option('decimalplaces', options.decimalplaces) ..
+      '\\NodetreeUnregisterCallback{post_linebreak_filter}'  .. '\n' ..
+      '\\NodetreeRegisterCallback{' .. options.callback .. '}'
+
+    local prefix = '%!TEX program = lualatex\n' ..
+                  '\\documentclass{article}\n' ..
+                  '\\usepackage{nodetree}\n' ..
+                  options .. '\n' ..
+                  '\\begin{document}\n'
+    local suffix = '\n\\end{document}'
+    output_file:write(prefix .. tex_markup .. suffix)
+    output_file:close()
+
+    -- Compile the temporary LuaTeX or LuaLaTeX file.
+    os.spawn({ 'latexmk', '-cd', '-pdflua', absolute_path_tex })
+    local include_file = assert(io.open(parent_path .. '/' .. example_counter .. '.nttex', 'rb'))
+    local include_content = include_file:read("*all")
+    include_file:close()
+    include_content = include_content:gsub('[\r\n]', '')
+    tex.print(include_content)
+  end,
+
+  --- Check for `--shell-escape`
+  --
+  check_shell_escape = function()
+    local info = status.list()
+    if info.shell_escape == 0 then
+      tex.error('Package "nodetree-embed": You have to use the --shell-escape option')
+    end
+  end,
+
+  --- Print a node tree.
+  ---
+  -- @tparam node head The head node of a node list.
+  -- @tparam table opts Options as a table.
+  print = function(head, opts)
+    if opts and type(opts) == 'table' then
+      set_options(opts)
+    end
+    nodetree_print(format.new_line())
+    tree.analyze_list(head, 1)
+  end,
+
+  --- Format a scaled point value into a formated string.
+  --
+  -- @tparam number sp A scaled point value
+  --
+  -- @treturn string
+  format_dim = function(sp)
+    return template.length(sp)
+  end,
+
+  --- Get a default option that is not changed.
+  -- @tparam string key The key of the option.
+  --
+  -- @treturn string|number|boolean
+  get_default_option = function(key)
+    return default_options[key]
   end
-end
-function base.execute()
-  local c = base.get_callback()
-  if options.engine == 'lualatex' then
-    luatexbase.add_to_callback(c, callbacks.post_linebreak_filter, 'nodetree')
-  else
-    id, error = callback.register(c, callbacks.post_linebreak_filter)
-  end
-end
-function base.analyze(head)
-  tpl.print('\n')
-  tree.analyze_list(head, 1)
-end
-return base
+}
+
+--- Use export.print
+-- @tparam node head
+export.analyze = export.print
+
+return export

Modified: trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.sty
===================================================================
--- trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.sty	2020-05-30 20:58:42 UTC (rev 55346)
+++ trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.sty	2020-05-30 20:59:30 UTC (rev 55347)
@@ -8,7 +8,7 @@
 %% 
 %% This is a generated file.
 %% 
-%% Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
+%% Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
 %% ----------------------------------------------------------------------
 %% This work may be distributed and/or modified under the conditions of
 %% the LaTeX Project Public License, either version 1.3c of this license
@@ -22,34 +22,33 @@
 %% 
 \NeedsTeXFormat{LaTeX2e}[1999/12/01]
 \ProvidesPackage{nodetree}
-    [2016/07/18 v1.2 Visualize node lists in a tree view]
+    [2020/05/29 v2.0 Visualize node lists in a tree view]
 \input{nodetree}
-\directlua{
-  nodetree.set_option('engine', 'lualatex')
-}
 \RequirePackage{kvoptions}
 \SetupKeyvalOptions{
   family=NT,
-  prefix=NT@
+  prefix=NTK@
 }
 \DeclareStringOption[term]{channel}
-\define at key{NT}{channel}[]{\nodetreeoption[channel]{#1}}
+\define at key{NT}{channel}[]{\NodetreeSetOption[channel]{#1}}
 \DeclareStringOption[postlinebreak]{callback}
-\define at key{NT}{callback}[]{\nodetreeoption[callback]{#1}}
+\define at key{NT}{callback}[]{\NodetreeSetOption[callback]{#1}}
 \DeclareStringOption[1]{verbosity}
-\define at key{NT}{verbosity}[]{\nodetreeoption[verbosity]{#1}}
+\define at key{NT}{verbosity}[]{\NodetreeSetOption[verbosity]{#1}}
 \DeclareStringOption[colored]{color}
-\define at key{NT}{color}[]{\nodetreeoption[color]{#1}}
+\define at key{NT}{color}[]{\NodetreeSetOption[color]{#1}}
 \DeclareStringOption[1]{unit}
-\define at key{NT}{unit}[]{\nodetreeoption[unit]{#1}}
+\define at key{NT}{unit}[]{\NodetreeSetOption[unit]{#1}}
 \DeclareStringOption[1]{decimalplaces}
-\define at key{NT}{decimalplaces}[]{\nodetreeoption[decimalplaces]{#1}}
-\ProcessKeyvalOptions*
+\define at key{NT}{decimalplaces}[]{\NodetreeSetOption[decimalplaces]{#1}}
+\ProcessKeyvalOptions{NT}
 \directlua{
-  nodetree.set_default_options()
   nodetree.register_callbacks()
 }
-\newcommand{\nodetreeset}[1]{\setkeys{nodetree}{#1}}
+\newcommand{\NodetreeSet}[1]{%
+  \setkeys{NT}{#1}%
+}
+\let\nodetreeset\NodetreeSet
 \endinput
 %%
 %% End of file `nodetree.sty'.

Modified: trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.tex
===================================================================
--- trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.tex	2020-05-30 20:58:42 UTC (rev 55346)
+++ trunk/Master/texmf-dist/tex/luatex/nodetree/nodetree.tex	2020-05-30 20:59:30 UTC (rev 55347)
@@ -8,7 +8,7 @@
 %% 
 %% This is a generated file.
 %% 
-%% Copyright (C) 2016 by Josef Friedrich <josef at friedrich.rocks>
+%% Copyright (C) 2016-2020 by Josef Friedrich <josef at friedrich.rocks>
 %% ----------------------------------------------------------------------
 %% This work may be distributed and/or modified under the conditions of
 %% the LaTeX Project Public License, either version 1.3c of this license
@@ -22,26 +22,44 @@
 %% 
 \directlua{
   nodetree = require('nodetree')
-  nodetree.set_option('engine', 'luatex')
-  nodetree.set_default_options()
 }
-\def\nodetreeoption[#1]#2{
+\def\NodetreeSetOption[#1]#2{
   \directlua{
     nodetree.set_option('#1', '#2')
   }
 }
-\def\nodetreeregister#1{
+\let\nodetreeoption\NodetreeSetOption
+\def\NodetreeResetOption#1{
+  \NodetreeSetOption[#1]{%
+    \directlua{
+      tex.print(nodetree.get_default_option('#1'))
+    }%
+  }%
+}
+\def\NodetreeReset{
+  \NodetreeResetOption{callback}
+  \NodetreeResetOption{channel}
+  \NodetreeResetOption{color}
+  \NodetreeResetOption{decimalplaces}
+  \NodetreeResetOption{engine}
+  \NodetreeResetOption{unit}
+  \NodetreeResetOption{verbosity}
+}
+\let\nodetreereset\NodetreeReset
+\def\NodetreeRegisterCallback#1{
   \directlua{
     nodetree.set_option('callback', '#1')
     nodetree.register_callbacks()
   }
 }
-\def\nodetreeunregister#1{
+\let\nodetreeregister\NodetreeRegisterCallback
+\def\NodetreeUnregisterCallback#1{
   \directlua{
     nodetree.set_option('callback', '#1')
     nodetree.unregister_callbacks()
   }
 }
+\let\nodetreeunregister\NodetreeUnregisterCallback
 \endinput
 %%
 %% End of file `nodetree.tex'.



More information about the tex-live-commits mailing list.