%% This is part of the OpTeX project, see http://petr.olsak.net/optex \_codedecl \printdoc {Macros for documentation printing <2023-12-10>} % loaded on demand by \load[doc] \_doc ----------------------------- General declarations. \_cod ----------------------------- \_fontfam[lmfonts] \_let \mlinkcolor=\Red % main doc. points \_let \ulinkcolor=\Blue % user doc. points \_let \fnamecolor=\Brown % file names in listing headers \_def \bgverbcolor {\_setcmykcolor{0 0 .3 .03}} % background for listings \_def \outlinkcolor {\_setcmykcolor{1 0 1 .2}} % green for outerlinks \_def \inlinkcolor {\_setcmykcolor{0 1 0 .1}} % magenta for internal links \_hyperlinks \inlinkcolor \outlinkcolor \_enlang \_enquotes \_doc ----------------------------- Maybe, somebody needs `\seccc` or `\secccc`? \_cod ----------------------------- \_eoldef\seccc#1{\_medskip \_noindent{\_bf#1}\_par\_nobreak\_firstnoindent} \_def\secccc{\_medskip\_noindent $\_bullet$ } \_doc ----------------------------- `\enddocument` can be redefined. \_cod ----------------------------- \_let\enddocument=\_bye \_doc ----------------------------- A full page of listing causes underfull `\vbox` in output routine. We need to add a small tolerance. \_cod ----------------------------- \_pgbottomskip=0pt plus10pt minus2pt \_doc ----------------------------- The listing mode is implemented here. The \`\maxlines` is maximal lines of code printed in the listing mode. \_cod ----------------------------- \_newcount \_maxlines \_maxlines=100000 \_public \maxlines ; \_eoldef\_cod#1{\_par \_wipeepar \_vskip\_parskip \_medskip \_ttskip \_begingroup \_typosize[8/10] \_let\_printverbline=\_printcodeline \_ttline=\_inputlineno \_setverb \_ifnum\_ttline<0 \_let\_printverblinenum=\_relax \_else \_initverblinenum \_fi \_adef{ }{\ }\_adef\^^I{\t}\_parindent=\_ttindent \_parskip=0pt \_def\t{\_hskip \_dimexpr\_tabspaces em/2\_relax}% \_relax \_ttfont \_endlinechar=`^^J \_def\_tmpb{\_start}% \_readverbline } \_def\_readverbline #1^^J{% \_def\_tmpa{\_empty#1}% \_let\_next=\_readverbline \_ea\_isinlist\_ea\_tmpa\_ea{\_Doc}\_iftrue \_let\_next=\_processinput \_fi \_ea\_isinlist\_ea\_tmpa\_ea{\_Doctab}\_iftrue \_let\_next=\_processinput \_fi \_ea\_isinlist\_ea\_tmpa\_ea{\_Endcode}\_iftrue \_def\_next{\_processinput\_endinput}\_fi \_ifx\_next\_readverbline \_addto\_tmpb{#1^^J}\_fi \_next } {\_catcode`\ =13 \_gdef\_aspace{ }}\_def\_asp{\_ea\_noexpand\_aspace} \_edef\_Doc{\_asp\_asp\_bslash _doc} \_bgroup \_lccode`~=`\^^I \_lowercase{\_egroup\_edef\_Doctab{\_noexpand~\_bslash _doc}} \_edef\_Endcode{\_noexpand\_empty\_bslash _endcode} \_doc ----------------------------- The scanner of the control sequences in the listing mode replaces all occurrences of `\` by \`\_makecs`. This macro reads next tokens and accumulates them to `\_tmpa` as long as they have category 11. It means that `\_tmpa` includes the name of the following control sequence when \fw\`\_makecsF` is run. The printing form of the control sequence is set to `\_tmpb` and the test of existence `\,`is performed. If it is true then active hyperlink is created. If not, then the first `_` or `.` is removed from `\_tmpa` and the test is repeated. \_cod ----------------------------- \_def\_makecs{\_def\_tmpa{}\_futurelet\_next\_makecsD} \_def\_makecsD{\_if.\_next \_ea\_makecsB \_else \_ea\_makecsA \_fi} % \.foo is accepted \_def\_makecsA{\_ifcat a\_noexpand\_next \_ea\_makecsB \_else \_ea\_makecsF \_fi} \_def\_makecsB#1{\_addto\_tmpa{#1}\_futurelet\_next\_makecsA} \_def\_makecsF{\_let\_tmpb=\_tmpa \_ifx\_tmpa\_empty \_csstring\\% \_else \_ifcsname ,\_tmpa\_endcsname \_trycs{el:\_tmpa}{\_intlink}% \_else \_remfirstunderscoreordot\_tmpa \_ifx\_tmpa\_empty \_let\_tmpa=\_tmpb \_fi \_ifcsname ,\_tmpa\_endcsname \_trycs{el:\_tmpa}{\_intlink}% \_else \_csstring\\\_tmpb \_fi\_fi\_fi } \_def\_processinput{% \_let\_start=\_relax \_ea\_replstring\_ea\_tmpb\_ea{\_aspace^^J}{^^J} \_addto\_tmpb{\_fin}% \_isinlist\_tmpb{\_start^^J}\_iftrue \_advance\_ttline by1\_fi \_replstring\_tmpb{\_start^^J}{\_start}% \_replstring\_tmpb{\_start}{}% \_replstring\_tmpb{^^J\_fin}{\_fin}% \_replstring\_tmpb{^^J\_fin}{}% \_replstring\_tmpb{\_fin}{}% \_ea\_prepareverbdata\_ea\_tmpb\_ea{\_tmpb^^J}% \_replthis{\_csstring\\}{\_noexpand\_makecs}% \_ea\_printverb \_tmpb\_fin \_par \_endgroup \_ttskip \_isnextchar\_par{}{\_noindent}% } \_def\_remfirstunderscoreordot#1{\_ea\_remfirstuordotA#1\_relax#1} \_def\_remfirstuordotA#1#2\_relax#3{\_if _#1\_def#3{#2}\_fi \_if\_string#1.\_def#3{#2}\_fi} \_doc ----------------------------- By default the internal link is created by \`\_intlink` inside listing mode. But you can define `\el:` which has precedence and it can create an external link. The `\_tmpa` includes the name used in the link and `\_tmpb` is the name to be printed. See \^`\_makecsF` above and the example at the beginning of this section. \_cod ----------------------------- \_def\_intlink{\_link[cs:\_tmpa]{\ulinkcolor}{\_csstring\\\_tmpb}} \_doc ----------------------------- The lines in the listing mode have a yellow background. \_cod ----------------------------- \_def\_printcodeline#1{\_advance \_maxlines by-1 \_ifnum \_maxlines<0 \_ea \_endverbprinting \_fi \_ifx\_printfilename\_relax \_penalty \_ttpenalty \_fi \_vskip-4pt \_noindent\_rlap{\bgverbcolor \_vrule height8pt depth5pt width\_hsize}% \_printfilename \_indent \_printverblinenum #1\_par} \_def\_printfilename{\_hbox to0pt{% \_hskip\_hsize\_vbox to0pt{\_vss\_llap{\fnamecolor\docfile}\_kern7.5pt}\_hss}% \_let\_printfilename=\_relax } \_everytt={\_let\_printverblinenum=\_relax} \_long\_def\_endverbprinting#1\_fin#2\_fin{\_fi\_fi \_global\_maxlines=100000 \_noindent\_typosize[8/]\_dots etc. (see {\_tt\fnamecolor\docfile})} \_doc ----------------------------- \`\docfile` is currently documented file.\nl \`\printdoc` and \`\printdoctail` macros are defined here. \_cod ----------------------------- \_def\docfile{} \_def\_printdoc #1 {\_par \_def\docfile{#1}% \_everytt={\_ttshift=-15pt \_let\_printverblinenum=\_relax}% \_ea\_cod \_input #1 \_everytt={\_let\_printverblinenum=\_relax}% \_def\docfile{}% } \_def\_printdoctail #1 {\_bgroup \_everytt={}\_ttline=-1 \_ea\_printdoctailA \_input #1 \_egroup} {\_long\_gdef\_printdoctailA#1\_endcode{}} \_public \printdoc \printdoctail ; \_doc ----------------------------- You can do `\verbinuput \vitt{} (-) ` if you need analogical design like in listing mode. \_cod ----------------------------- \_def\_vitt#1{\_def\docfile{#1}\_ttline=-1 \_everytt={\_typosize[8/10]\_let\_printverbline=\_printcodeline \_medskip}} \_public \vitt ; \_doc ----------------------------- The Index entries are without the trailing backslash in `.ref` file. When printing Index, we distinguish the Index entries with their main documentation point (they are created as links and backslash is added), Index entries with only user documentation points have backslash added but no link is created. Other index entries are printed as usual without backslash. \_cod ----------------------------- \_addto \_ignoredcharsen {_} % \foo, \_foo is the same in the fist pass of sorting \_let\_optexprintii=\_printii % original \_printii used for other Index entries \_def\_printii #1&{% \_ifcsname cs:#1\_endcsname \_noindent \_hskip-\_iindent {\_tt \_link[cs:#1]\ulinkcolor{\_bslash#1} }\_else \_ifcsname cs:^#1\_endcsname \_noindent \_hskip-\_iindent {\_tt\_bslash#1 }\_else \_afterfi{\_afterfi{\_optexprintii #1&}}\_fi\_fi } \_def\_pgprintA #1{#1} % no hyperlinks from page numbers \_def\_printiipages#1&{\_let\_pgtype=\_undefined \_tmpnum=0 {\_rm\_printpages #1,:,\_par}} \_sdef{_tocl:1}#1#2#3{\_nofirst\_bigskip \_bf\_llaptoclink{#1}{#2}\_hfill \_pgn{#3}\_tocpar\_medskip} \_doc ----------------------------- If this macro is loaded by \^`\load` then we need to initialize catcodes using the `\_afterload` macro. \_cod ----------------------------- \_def\_afterload{\_catcode`\<=13 \_catcode`\`=13 \_wlog {doc.opm: catcodes of < and ` activated.}% } \_doc ----------------------------- The \code{} will be print as . \_cod ----------------------------- \_let\lt=< \_catcode`\<=13 \_def<#1>{$\langle\hbox{\it#1\/}\rangle$} \_everyintt{\_catcode`\<=13 \_catcode`\.=11 } \_doc ----------------------------- Main documentation points and hyperlinks to/from it. Main documentation point: \code{\\`\\foo`}. User documentation point: \code{\\^`\\foo}, first occurrence only. The next occurrences are only links to the main documentation point. Link to user documentation point: \code{\\~`\\foo}. \_cod ----------------------------- \_def\_docrefcodes{\_catcode`\.=11\_relax} \_verbchar` \_def\`{\_bgroup \_docrefcodes \_mainpoint} \_def\_mainpoint #1`{\_egroup\_leavevmode\_edef\_tmp{\_csstring#1}\_iindex{\_tmp}% \_ifcsname cs:\_tmp\_endcsname \_moremainpoints \_else \_dest[cs:\_tmp]\_fi \_sxdef{cs:\_tmp}{}% \_hbox{\_ifcsname cs:^\_tmp\_endcsname \_link[cs:^\_tmp]{\mlinkcolor}{\_tt\_csstring\\\_tmp}\_else {\_tt\mlinkcolor\_csstring\\\_tmp}\_fi}% } \_def\^`{\_bgroup \_docrefcodes \_docpoint} \_def\_docpoint #1{\_egroup\_leavevmode\_edef\_tmp{\_csstring#1}\_iindex{\_tmp}% \_hbox{\_ifcsname cs:^\_tmp\_endcsname \_else \_dest[cs:^\_tmp]\_sxdef{cs:^\_tmp}{}\_fi \_link[cs:\_tmp]{\ulinkcolor}{\_tt\_string#1}}% \_futurelet\_next\_cslinkA } \_def\_cslinkA{\_ifx\_next`\_ea\_ignoreit \_else \_ea\_ea\_ea`\_ea\_string\_fi} \_def\~`{\_bgroup \_docrefcodes \_doctpoint} \_def\_doctpoint #1{\_egroup\_leavevmode\_edef\_tmp{\_csstring#1}\_iindex{\_tmp}% \_hbox{\_link[cs:^\_tmp]{\ulinkcolor}{\_tt\_string#1}}% \_futurelet\_next\_cslinkA } \_def\_moremainpoints{\_opwarning{Second main documentation point \_bslash\_tmp}} \_doc ----------------------------- The \`\fw` macro for forward links to user documentation point (given later) is defined here. \_cod ----------------------------- \_def\_fw\`#1`{{\_slet{cs:^\_csstring#1}{}\`#1`}} \_public \fw ; \_endcode %------------------------------------------- \noindent The \^`\printdoc` `` and \^`\printdoctail` `` commands are defined after the file `doc.opm` is load by \^`\load`~`[doc]`. The `\printdoc` starts reading of given `` from the second line. The file is read in {\em the listing mode}. The `\prindoctail` starts reading given `` from the first occurrence of the `\_endcode`. The file is read in normal mode (like `\input `). The {\em listing mode} prints the lines as a listing of a code. This mode is finished when first {\visiblesp` \_doc`} occurs or first `\_endcode` occurs. At least two spaces or one tab character must precede before such `\_doc`. On the other hand, the `\_endcode` must be at the left edge of the line without spaces. If this rule is not met then the listing mode continues. If the first line or the last line of the listing mode is empty then such lines are not printed. The maximal number of printed lines in the listing mode is \^`\maxlines`. It is set to almost infinity (100000). You can set it to a more sensible value. Such a setting is valid only for the first following listing mode. When the listing mode is finished by `\_doc` then the next lines are read in the normal way, but the material between `\begtt` ... `\endtt` pair is shifted by three letters left. The reason is that the three spaces of indentation is recommended in the `\_doc` ... `\_cod` pair and this shifting is compensation for this indentation. The `\_cod` macro ignores the rest of the current line and starts the listing mode again. When the listing mode is finished by the `\_endcode` then the `\endinput` is applied, the reading of the file opened by `\printdoc` is finished. You cannot reach the end of the file (without `\_endcode`) in the listing mode. The main documentation point is denoted by \code{\\`\\}``\code{`} in red, for example \code{\\`\\foo`}. The user documentation point is the first occurrence of \code{\\^`\\}``\code{`}, for example \code{\\^`\\foo`}. There can be more such markups, all of them are hyperlinks to the main documentation point. And main documentation point is a hyperlink to the user documentation point if this point precedes. Finally, the \code{\\~`\\}``\code{`} (for example \code{\\~`\\foo`}) are hyperlinks to the user documentation point. By default, the hyperink from main documentation point to the user documentation point is active only if it is backward link, i.e.\ the main documentation point is given later. The reason is that we don't know if such user documentation point will exist when creating main documentation point and we don't want broken links. If you are sure that user documentation point will follow then use prefix \^`\fw` before~\code{\\`}, for example \code{\\fw\\`\\foo`} is main documentation point where the user documentation point is given later and forward hyperlink is created here. Control sequences and their page positions of main documentation points and user documentation points are saved to the index. The listing mode creates all control sequences which are listed in the index as an active link to the main documentation point of such control sequence and prints them in blue. Moreower, active links are control sequences of the type `\_foo` or `\.foo` although the documentation mentions only `\foo`. Another text is printed in black. The listing mode is able to generate external links to another \OpTeX/-like documentation, if the macros `\,` and `\el:` are defined. The second macro should create a hyperlink using `\_tmpa` where the link name of the is saved and `\_tmpb` where the name of the to be printed is saved (`\tmpb` can include preceding `_` or `.` unlike `\_tmpa`). For example, suppose, that we have created `optex-doc.eref` file by: \begtt TEXINPUTS='.;$TEXMF/{doc,tex}//' optex optex-doc grep Xindex optex-doc.ref > optex-doc.eref \endtt The `.eref` file includes only `\_Xindex{}{}` lines from `optex-doc.ref` file. Then we can use following macros: \begtt \def\_Xindex#1#2{\sdef{,#1}{}\slet{el:#1}{optexdoclink}} \def\optexdoclink{% \edef\extlink{url:\optexdocurl\csstring\#cs:\_tmpa}% \_ea\_urlactive\_ea[\extlink]{\Cyan}{\csstring\\\_tmpb}} \def\optexdocurl{http://petr.olsak.net/ftp/olsak/optex/optex-doc.pdf} \isfile{optex-doc.eref}\iftrue \input{optex-doc.eref}\fi \endtt All `\el:`, where is from `optex-doc.ref`, have the same meaning: `\optexdoclink` in this example. And `\optexdoclink` creates the external link in `\Cyan` color. \secc Implementation \endinput 2023-12-10 \catcode`.=11 removed, issue solved by \_makecsD. 2022-12-11 \_opwaning "Second main documentation point" introduced. 2022-12-11 \_docrefcodes added (bug due to 2022-11-13 fixed). 2022-11-21 magenta color for internal links instead green. 2022-11-13 \catcode`.=11: only local settings 2022-07-01 \_printii improved, colors declaration part added. 2021-05-15 \_endinput shifted after \_processinput when \_endcode is scanned. 2021-05-14 \_catcodedot, \_Doctab introduced. 2021-05-13 \def\t added, bug fixed. 2021-05-03 External links fom listing mode allowed. 2021-05-02 \_forwadlink replaced by \_fw, to be more consistent. 2021-05-02 \fw introduced, \.foo -> \foo allowed. 2020-04-28 \levevmode in \^ macros added (bug fixed) 2020-04-22 released