% \iffalse %% File: gtl.dtx Copyright (C) 2013,2015,2017,2018,2024 Bruno Le Floch %% %% 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 %% %% This work has the LPPL maintenance status 'maintained' %% and the current maintainer is Bruno Le Floch. %% %% This work consists of the files gtl.dtx and gtl.ins and %% derived file gtl.sty. %% ----------------------------------------------------------------------- % %<*driver> %\fi %\iffalse \documentclass[full]{l3doc} \usepackage{gtl} \begin{document} \DocInput{gtl.dtx} \end{document} % % \fi % % \title{The \textsf{gtl} package: \\ % manipulate unbalanced lists of tokens\thanks{This % file has version number 0.6, last revised 2024/01/04.}} % \author{Bruno Le Floch} % \date{2024/01/04} % % \maketitle % \tableofcontents % % \begin{documentation} % % \section{\pkg{gtl} documentation} % % The \texttt{expl3} programming language provides various tools to % manipulate lists of tokens (package \pkg{l3tl}). However, those token % lists must have balanced braces, or more precisely balanced % begin-group and end-group characters. The \pkg{gtl} package % manipulates instead lists of tokens which may be unbalanced, with more % begin-group or more end-group characters. % % A technical comment: currently, all begin-group characters are % assumed to have the character code of ``|{|'' and all end-group % characters that of~``|}|''. % % Please report bugs (or suggestions) on the issue tracker % (\url{https://github.com/blefloch/latex-gtl/issues}). % % ^^A The following code checks the test coverage % \begingroup\ExplSyntaxOn % \msg_new:nnn { gtldoc } { discarded } { Discarded~some~material~after~closing~brace~of~{function}~or~TEST~or~TESTEXP~argument:~``#1'' } % \msg_new:nnn { gtldoc } { bad-test } { TEST~or~TESTEXP~followed~by~unexpected~character~``#1'' } % \msg_new:nnn { gtldoc } { multi-function } { Some~functions~are~defined~twice } % \msg_new:nnn { gtldoc } { doc-no-test } { ``#1''~is~documented~but~has~no~test } % \msg_new:nnn { gtldoc } { test-no-doc } { ``#1''~is~tested~but~not~documented } % \msg_new:nnn { gtldoc } { junk } { There~is~some~unexpected~junk.~Giving~up~on~this~line. } % \tl_new:N \l__gtldoc_arg_tl % \seq_new:N \l__gtldoc_internal_seq % \cs_generate_variant:Nn \use:nn { no } % \cs_generate_variant:Nn \tl_if_in:nnTF { no } % \cs_generate_variant:Nn \tl_if_in:nnT { no } % \cs_generate_variant:Nn \tl_remove_all:Nn { Nx } % \cs_generate_variant:Nn \seq_set_filter:NNn { NNx } % \group_begin: % \seq_new:N \l__gtldoc_fn_seq % \iow_new:N \g__gtldoc_ior % \tl_new:N \l__gtldoc_internal_tl % \str_new:N \l__gtldoc_TF_str % \seq_new:N \l__gtldoc_arg_seq % \seq_new:N \l__gtldoc_arg_TF_seq % \cs_new:Npn \__gtldoc_remove_backslash:N #1 % { \exp_after:wN \token_if_eq_charcode:NNF \c_backslash_str #1 {#1} } % \cs_new_protected:Npn \gtldoc_get_fn:n #1 % { % \seq_clear:N \l__gtldoc_fn_seq % \clist_map_inline:nn {#1} % { % \__gtldoc_get_fn:nn { function } {#1} ^^A todo: this is wasteful, going through the file twice! % \__gtldoc_get_fn:nn { variable } {#1} % } % } % \cs_new_protected:Npn \__gtldoc_get_fn:nn #1 % { % \exp_args:Noo \__gtldoc_get_fn_aux:nnn % { \tl_to_str:n { begin } } { \tl_to_str:n { {#1} } } % } % \cs_new_eq:NN \__gtldoc_test:w ? % \cs_new_eq:NN \__gtldoc_skip:w ? % \cs_new_protected:Npn \__gtldoc_get_fn_aux:nnn #1#2#3 % { % \cs_gset_protected:Npn \__gtldoc_test:w ##1 #1 ##2 #2 { } % \cs_gset_protected:Npn \__gtldoc_skip:w ##1 #1 ##2 #2 % { % \tl_clear:N \l__gtldoc_arg_tl % \str_clear:N \l__gtldoc_TF_str % \tl_if_blank:nF {##2} % { % \msg_error:nn { gtldoc } { junk } % \use_none_delimit_by_q_recursion_stop:w % } % \__gtldoc_aux:N % } % \ior_open:Nn \g__gtldoc_ior {#3} % \ior_str_map_inline:Nn \g__gtldoc_ior % { % \tl_if_in:noT {##1} { \tl_to_str:n { end { documentation } } } ^^A careful if you remove spaces! % { \ior_map_break: } % \tl_if_empty:oF { \__gtldoc_test:w ##1 #1 #2 } % { \__gtldoc_skip:w ##1 \q_recursion_tail \q_recursion_stop } % } % \ior_close:N \g__gtldoc_ior % } % \cs_new_protected:Npn \__gtldoc_aux:N #1 % { % \token_if_eq_meaning:NNT #1 \q_recursion_tail % { \__gtldoc_more:Nw \__gtldoc_aux:N } % \str_case_e:nnF {#1} % { % \c_percent_str { \__gtldoc_aux:N } ^^A ignore percent % ^ { \__gtldoc_hat:N } ^^A check for "^^A" % [ { \__gtldoc_opt:w } % \c_left_brace_str { \__gtldoc_arg:w } % } % { \__gtldoc_varg:N #1 } % } % \cs_new_protected:Npn \__gtldoc_more:Nw #1 #2 \q_recursion_stop % { ^^A grab one more line and restart the loop % \ior_str_get:NN \g__gtldoc_ior \l__gtldoc_internal_tl % \exp_after:wN #1 \l__gtldoc_internal_tl % \q_recursion_tail \q_recursion_stop % } % \cs_new_protected:Npn \__gtldoc_hat:N #1 % { % \token_if_eq_charcode:NNTF #1 ^ % { \__gtldoc_hat_hat:N } % { \__gtldoc_varg:N ^ #1 } % } % \cs_new_protected:Npn \__gtldoc_hat_hat:N #1 ^^A in principle we should worry about crazy people doing ^^01 or ^^5b etc % { % \token_if_eq_charcode:NNTF #1 A % { \__gtldoc_more:Nw \__gtldoc_aux:N } ^^A we found a comment % { \__gtldoc_varg:N ^ ^ #1 } % } % \cs_new_protected:Npn \__gtldoc_opt:w #1 \q_recursion_tail % { % \tl_if_in:nnTF {#1} { ] } % { \__gtldoc_opt_aux:w #1 \q_recursion_tail } % { \__gtldoc_more:Nw \__gtldoc_opt:w } % } % \cs_new_protected:Npn \__gtldoc_opt_aux:w #1 ] % { % \tl_if_in:noTF {#1} { \tl_to_str:n { pTF } } % { \str_set:Nn \l__gtldoc_TF_str { pTF } } % { % \tl_if_in:noT {#1} { \tl_to_str:n { TF } } % { \str_set:Nn \l__gtldoc_TF_str { TF } } % } % \__gtldoc_aux:N % } % \cs_new_protected:Npn \__gtldoc_arg:w % { \exp_after:wN \__gtldoc_varg:N \c_right_brace_str } % \cs_new_protected:Npn \__gtldoc_varg:N #1 % { \exp_args:No \__gtldoc_varg_aux:n { \token_to_str:N #1 } } % \cs_new_protected:Npn \__gtldoc_varg_aux:n #1 % { % \cs_gset_protected:Npn \__gtldoc_varg_aux:w ##1 #1 { {##1} } % \cs_gset_protected:Npn \__gtldoc_varg_loop:w ##1 \q_recursion_tail % { % \tl_if_in:noTF {##1} {#1} % { % \exp_after:wN \__gtldoc_varg_last:nw % \__gtldoc_varg_aux:w ##1 \q_recursion_tail % } % { % \tl_put_right:Ne \l__gtldoc_arg_tl { ##1 \iow_char:N \^^M } % \__gtldoc_more:Nw \__gtldoc_varg_loop:w % } % } % \__gtldoc_varg_loop:w % } % \cs_new_eq:NN \__gtldoc_varg_aux:w ? % \cs_new_eq:NN \__gtldoc_varg_loop:w ? % \cs_new_protected:Npn \__gtldoc_varg_last:nw % #1#2 \q_recursion_tail \q_recursion_stop % { % \tl_if_blank:nF {#2} % { \msg_warning:nnn { gtldoc } { discarded } {#2} } % \tl_put_right:Nn \l__gtldoc_arg_tl {#1} % \tl_remove_all:Ne \l__gtldoc_arg_tl { \iow_char:N \^^M \iow_char:N \% } ^^A from l3doc % \tl_remove_all:Ne \l__gtldoc_arg_tl { \tl_to_str:n { ^ ^ A } } ^^A from l3doc % \tl_remove_all:Ne \l__gtldoc_arg_tl { \iow_char:N \^^I } ^^A from l3doc % \tl_remove_all:Ne \l__gtldoc_arg_tl { \iow_char:N \^^M } ^^A from l3doc % ^^A \@@_replace_at_at:N \l_@@_tmpa_tl ^^A from l3doc but not done here % \exp_args:NNe \seq_set_from_clist:Nn \l__gtldoc_arg_seq % { \tl_to_str:N \l__gtldoc_arg_tl } % \str_if_empty:NF \l__gtldoc_TF_str % { % \seq_clear:N \l__gtldoc_arg_TF_seq % \seq_map_inline:Nn \l__gtldoc_arg_seq % { % \seq_put_right:Ne \l__gtldoc_arg_TF_seq { ##1 TF } % \seq_put_right:Ne \l__gtldoc_arg_TF_seq { ##1 T } % \seq_put_right:Ne \l__gtldoc_arg_TF_seq { ##1 F } % \str_if_eq:eeT { \l__gtldoc_TF_str } { pTF } % { \seq_put_right:Ne \l__gtldoc_arg_TF_seq { \__gtldoc_pred:w ##1 } } % } % \seq_set_map_e:NNn \l__gtldoc_arg_seq % \l__gtldoc_arg_TF_seq { \tl_to_str:n {##1} } % } % \seq_concat:NNN \l__gtldoc_fn_seq \l__gtldoc_fn_seq \l__gtldoc_arg_seq % } % \use:no { \cs_new:Npn \__gtldoc_pred:w #1 } { \token_to_str:N : } { #1 _p: } % \group_end: % \group_begin: % \seq_new:N \l__gtldoc_tested_fn_seq % \cs_new_protected:Npn \gtldoc_get_tests:n #1 % { % \seq_clear:N \l__gtldoc_tested_fn_seq % \clist_map_function:nN {#1} \__gtldoc_get_test:n % } % \cs_new_protected:Npn \__gtldoc_get_test:n #1 % { % \ior_open:Nn \g__gtldoc_ior {#1} % \ior_str_map_inline:Nn \g__gtldoc_ior % { \__gtldoc_test_aux:n {##1} } % \ior_close:N \g__gtldoc_ior % } % \cs_gset_protected:Npn \__gtldoc_tmp:nn #1#2 % { % \cs_new_protected:Npn \__gtldoc_test_skipi:w ##1 #1 { } % \cs_new_protected:Npn \__gtldoc_test_skipii:w ##1 #2 { } % \cs_new_protected:Npn \__gtldoc_test_aux:n ##1 % { % \tl_if_empty:oF { \__gtldoc_test_skipi:w ##1 #1 } % { % \tl_if_empty:oTF { \__gtldoc_test_skipii:w ##1 #2 } % { \__gtldoc_test_auxi:w } { \__gtldoc_test_auxii:w } % ##1 \q_recursion_tail \q_recursion_stop % } % } % \cs_new_protected:Npn \__gtldoc_test_auxi:w ##1 #1 % { \__gtldoc_test_auxiii:N } % \cs_new_protected:Npn \__gtldoc_test_auxii:w ##1 #2 % { \__gtldoc_test_auxiii:N } % } % \exp_args:Noo \__gtldoc_tmp:nn % { \token_to_str:N \TEST } % { \token_to_str:N \TESTEXP } % \cs_new_protected:Npn \__gtldoc_test_auxiii:N #1 % { % \token_if_eq_meaning:NNT \q_recursion_tail #1 % { \__gtldoc_more:Nw \__gtldoc_test_auxiii:N } % \exp_after:wN \token_if_eq_charcode:NNTF \c_left_brace_str #1 % { \__gtldoc_test_arg:w } % { % \msg_error:nnn { gtldoc } { bad-test } {#1} % \use_none_delimit_by_q_recursion_stop:w % } % } % \cs_new_protected:Npn \__gtldoc_test_arg:w % { % \tl_clear:N \l__gtldoc_arg_tl % \exp_args:No \__gtldoc_test_arg_aux:N \c_right_brace_str % } % \cs_new_protected:Npn \__gtldoc_test_arg_aux:N #1 % { % \cs_gset_protected:Npn \__gtldoc_test_arg_aux:w ##1 #1 { {##1} } % \cs_gset_protected:Npn \__gtldoc_test_arg_loop:w ##1 \q_recursion_tail % { % \tl_if_in:noTF {##1} {#1} % { % \exp_after:wN \__gtldoc_test_arg_last:nw % \__gtldoc_test_arg_aux:w ##1 \q_recursion_tail % } % { % \tl_put_right:Nn \l__gtldoc_arg_tl {##1} % \__gtldoc_more:Nw \__gtldoc_test_arg_loop:w % } % } % \__gtldoc_test_arg_loop:w % } % \cs_new_eq:NN \__gtldoc_test_arg_aux:w ? % \cs_new_eq:NN \__gtldoc_test_arg_loop:w ? % \cs_new_protected:Npn \__gtldoc_test_arg_last:nw % #1#2 \q_recursion_tail \q_recursion_stop % { % \tl_if_blank:nF {#2} % { \msg_warning:nnn { gtldoc } { discarded } {#2} } % \tl_put_right:Nn \l__gtldoc_arg_tl {#1} % \exp_args:NNe \seq_set_from_clist:Nn \l__gtldoc_arg_seq % { \tl_to_str:N \l__gtldoc_arg_tl } % \exp_args:NNNe \seq_set_filter:NNn \l__gtldoc_arg_seq \l__gtldoc_arg_seq % { \exp_not:N \tl_if_head_eq_charcode_p:nN {##1} \c_backslash_str } % \seq_concat:NNN \l__gtldoc_tested_fn_seq \l__gtldoc_tested_fn_seq \l__gtldoc_arg_seq % } % \group_end: % \gtldoc_get_fn:n { gtl.dtx } % \gtldoc_get_tests:n % { % ../../testfiles/gtl000.lvt , % ../../testfiles/gtl001.lvt , % ../../testfiles/gtl002.lvt , % ../../testfiles/gtl003.lvt , % ../../testfiles/gtl004.lvt , % ../../testfiles/gtl005.lvt , % ../../testfiles/gtl006.lvt , % ../../testfiles-plain/gtl007.lvt % } % \group_begin: % \seq_set_eq:NN \l__gtldoc_internal_seq \l__gtldoc_fn_seq % \seq_remove_duplicates:N \l__gtldoc_internal_seq % \int_compare:nF % { \seq_count:N \l__gtldoc_internal_seq = \seq_count:N \l__gtldoc_fn_seq } % { % \msg_error:nn { gtldoc } { multi-function } % \seq_set_eq:NN \l__gtldoc_fn_seq \l__gtldoc_internal_seq % } % \seq_remove_duplicates:N \l__gtldoc_tested_fn_seq % \seq_map_inline:Nn \l__gtldoc_fn_seq % { % \seq_if_in:NnTF \l__gtldoc_tested_fn_seq {#1} % { \seq_remove_all:Nn \l__gtldoc_tested_fn_seq {#1} } % { \msg_error:nnn { gtldoc } { doc-no-test } {#1} } % } % \seq_map_inline:Nn \l__gtldoc_tested_fn_seq % { \msg_error:nnn { gtldoc } { test-no-doc } {#1} } % \group_end: % \endgroup % % \subsection{Creating and initialising extended token lists} % % \begin{function}{\gtl_new:N} % \begin{syntax} % \cs{gtl_new:N} \meta{gtl~var} % \end{syntax} % Creates a new \meta{gtl~var} or raises an error if the name is % already taken. The declaration is global. The \meta{gtl~var} will % initially be empty. % \end{function} % % \begin{function}[updated = 2024-01-04]{\gtl_const:Nn, \gtl_const:Ne, \gtl_const:Nx} % \begin{syntax} % \cs{gtl_const:Nn} \meta{gtl~var} \Arg{token list} % \end{syntax} % Creates a new constant \meta{gtl~var} or raises an error if the name % is already taken. The value of the \meta{gtl~var} will be set % globally to the balanced \meta{token list}. % \end{function} % % \begin{function}{\gtl_clear:N, \gtl_gclear:N} % \begin{syntax} % \cs{gtl_clear:N} \meta{gtl~var} % \end{syntax} % Empties the \meta{gtl~var}, locally or globally. % \end{function} % % \begin{function}{\gtl_clear_new:N, \gtl_gclear_new:N} % \begin{syntax} % \cs{gtl_clear_new:N} \meta{gtl~var} % \end{syntax} % Ensures that the \meta{gtl~var} exists globally by applying % \cs{gtl_new:N} if necessary, then applies \cs{gtl_(g)clear:N} to % leave the \meta{gtl~var} empty. % \end{function} % % \begin{function}{\gtl_set_eq:NN, \gtl_gset_eq:NN} % \begin{syntax} % \cs{gtl_set_eq:NN} \meta{gtl~var_1} \meta{gtl~var_2} % \end{syntax} % Sets the content of \meta{gtl~var_1} equal to that of % \meta{gtl~var_2}. % \end{function} % % \begin{function}{\gtl_concat:NNN, \gtl_gconcat:NNN} % \begin{syntax} % \cs{gtl_concat:NNN} \meta{gtl~var_1} \meta{gtl~var_2} \meta{gtl~var_3} % \end{syntax} % Concatenates the content of \meta{gtl~var_2} and \meta{gtl~var_3} % together and saves the result in \meta{gtl~var_1}. The % \meta{gtl~var_2} will be placed at the left side of the new extended % token list. % \end{function} % % \begin{function}[EXP, pTF]{\gtl_if_exist:N} % \begin{syntax} % \cs{gtl_if_exist_p:N} \meta{gtl~var} % \cs{gtl_if_exist:NTF} \meta{gtl~var} \Arg{true code} \Arg{false code} % \end{syntax} % Tests whether the \meta{gtl~var} is currently defined. This does % not check that the \meta{gtl~var} really is an extended token list % variable. % \end{function} % % \subsection{Adding data to token list variables} % % \begin{function}[updated = 2024-01-04]{\gtl_set:Nn, \gtl_set:Ne, \gtl_set:Nx, \gtl_gset:Nn, \gtl_gset:Ne, \gtl_gset:Nx} % \begin{syntax} % \cs{gtl_set:Nn} \meta{gtl~var} \Arg{token list} % \end{syntax} % Sets \meta{gtl~var} to contain the balanced \meta{token list}, % removing any previous content from the variable. % \end{function} % % \begin{function}{\gtl_put_left:Nn, \gtl_gput_left:Nn} % \begin{syntax} % \cs{gtl_put_left:Nn} \meta{gtl~var} \Arg{token list} % \end{syntax} % Appends the balanced \meta{token list} to the left side of the % current content of \meta{gtl~var}. % \end{function} % % \begin{function}{\gtl_put_right:Nn, \gtl_gput_right:Nn} % \begin{syntax} % \cs{gtl_put_right:Nn} \meta{gtl~var} \Arg{token list} % \end{syntax} % Appends the balanced \meta{token list} to the right side of the % current content of \meta{gtl~var}. % \end{function} % % \subsection{Extended token list conditionals} % % \begin{function}[EXP, pTF]{\gtl_if_blank:N} % \begin{syntax} % \cs{gtl_if_blank_p:N} \Arg{gtl~var} % \cs{gtl_if_blank:NTF} \Arg{gtl~var} \Arg{true code} \Arg{false code} % \end{syntax} % Tests if the \meta{gtl~var} consists only of blank spaces. The test % is \texttt{true} if \meta{gtl~var} consists of zero or more explicit % space characters (explicit tokens with character code~$32$ and % category code~$10$), and is \texttt{false} otherwise. % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_empty:N} % \begin{syntax} % \cs{gtl_if_empty_p:N} \meta{gtl~var} % \cs{gtl_if_empty:NTF} \meta{gtl~var} \Arg{true code} \Arg{false code} % \end{syntax} % Tests if the \meta{gtl~var} is entirely empty (\emph{i.e.}~contains % no tokens at all). % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_eq:NN} % \begin{syntax} % \cs{gtl_if_eq_p:NN} \Arg{gtl~var_1} \Arg{gtl~var_2} % \cs{gtl_if_eq:NNTF} \Arg{gtl~var_1} \Arg{gtl~var_2} \Arg{true code} \Arg{false code} % \end{syntax} % Tests if \meta{gtl~var_1} and \meta{gtl~var_2} have the same % content. The test is \texttt{true} if the two contain the same list % of tokens (identical in both character code and category code). % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_single_token:N} % \begin{syntax} % \cs{gtl_if_single_token_p:N} \meta{gtl~var} % \cs{gtl_if_single_token:NTF} \meta{gtl~var} \Arg{true code} \Arg{false code} % \end{syntax} % Tests if the content of the \meta{gtl~var} consists of a single % token. Such a token list has token count $1$ according to % \cs{gtl_count_tokens:N}. % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_tl:N} % \begin{syntax} % \cs{gtl_if_tl_p:N} \meta{gtl~var} % \cs{gtl_if_tl:NTF} \meta{gtl~var} \Arg{true code} \Arg{false code} % \end{syntax} % Tests if the \meta{gtl~var} is balanced. % \end{function} % % \subsection{The first token from an extended token list} % % \begin{function}[EXP]{\gtl_head:N} % \begin{syntax} % \cs{gtl_head:N} \meta{gtl~var} % \end{syntax} % Leaves in the input stream the first token in the \meta{gtl~var}. % If the \meta{gtl~var} is empty, nothing is left in the input stream. % \end{function} % % \begin{function}[EXP]{\gtl_head_do:NN} % \begin{syntax} % \cs{gtl_head_do:NN} \meta{gtl~var} \meta{cs} % \end{syntax} % Leaves in the input stream the \meta{control sequence} followed by % the first token in \meta{gtl~var}. If the \meta{gtl~var} is empty, % the \meta{cs} is followed by \cs{q_no_value}. % \end{function} % % \begin{function}[EXP, TF]{\gtl_head_do:NN} % \begin{syntax} % \cs{gtl_head_do:NNTF} \meta{gtl~var} \meta{cs} \Arg{true code} \Arg{false code} % \end{syntax} % If the \meta{gtl~var} is empty, leaves the \meta{false code} in the % input stream. Otherwise leaves the \meta{control sequence} % followed by the first token in \meta{gtl~var} and the \meta{true % code}. % \end{function} % % \begin{function}{\gtl_get_left:NN} % \begin{syntax} % \cs{gtl_get_left:NN} \meta{gtl~var_1} \meta{gtl~var_2} % \end{syntax} % Stores the first token from \meta{gtl~var_1} in \meta{gtl~var_2} as % an single-token extended token list, without removing it from % \meta{gtl~var_1}. % \end{function} % % \begin{function}{\gtl_pop_left:N, \gtl_gpop_left:N} % \begin{syntax} % \cs{gtl_pop_left:N} \meta{gtl~var} % \end{syntax} % Remove the first token from \meta{gtl~var_1}. % If the \meta{gtl~var_1} is empty nothing happens. % \end{function} % % \begin{function}{\gtl_pop_left:NN, \gtl_gpop_left:NN} % \begin{syntax} % \cs{gtl_pop_left:NN} \meta{gtl~var_1} \meta{gtl~var_2} % \end{syntax} % Stores the first token from \meta{gtl~var_1} in \meta{gtl~var_2} as % an single-token extended token list, and remove it from % \meta{gtl~var_1}. If the \meta{gtl~var_1} is empty it remains so, % and \meta{gtl~var_2} is set to contain \cs{q_no_value}. % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_head_eq_catcode:NN} % \begin{syntax} % \cs{gtl_if_head_eq_catcode_p:NN} \Arg{gtl~var} \meta{test token} % \cs{gtl_if_head_eq_catcode:NNTF} \Arg{gtl~var} \meta{test token} % ~~\Arg{true code} \Arg{false code} % \end{syntax} % Tests if the first token in \meta{gtl~var} has the same category % code as the \meta{test token}. In the case where \meta{gtl~var} is % empty, the test will always be \texttt{false}. % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_head_eq_charcode:NN} % \begin{syntax} % \cs{gtl_if_head_eq_charcode_p:NN} \Arg{gtl~var} \meta{test token} % \cs{gtl_if_head_eq_charcode:NNTF} \Arg{gtl~var} \meta{test token} % ~~\Arg{true code} \Arg{false code} % \end{syntax} % Tests if the first token in \meta{gtl~var} has the same character % code as the \meta{test token}. In the case where \meta{gtl~var} is % empty, the test will always be \texttt{false}. % \end{function} % % \begin{function}[EXP,pTF]{\gtl_if_head_eq_meaning:NN} % \begin{syntax} % \cs{gtl_if_head_eq_meaning_p:NN} \Arg{gtl~var} \meta{test token} % \cs{gtl_if_head_eq_meaning:NNTF} \Arg{gtl~var} \meta{test token} % ~~\Arg{true code} \Arg{false code} % \end{syntax} % Tests if the first token in \meta{gtl~var} has the same meaning as % the \meta{test token}. In the case where \meta{gtl~var} is empty, % the test will always be \texttt{false}. % \end{function} % % \begin{function}[EXP,pTF] % { % \gtl_if_head_is_group_begin:N, % \gtl_if_head_is_group_end:N, % \gtl_if_head_is_N_type:N, % \gtl_if_head_is_space:N, % } % \begin{syntax} % \cs{gtl_if_head_is_group_begin_p:N} \Arg{gtl~var} % \cs{gtl_if_head_is_group_begin:NTF} \Arg{gtl~var} % ~~\Arg{true code} \Arg{false code} % \end{syntax} % Tests whether the first token in \meta{gtl~var} is an explicit % begin-group character, an explicit end-group character, an |N|-type % token, or a space. In the case where \meta{gtl~var} is empty, the % test will always be \texttt{false}. % \end{function} % % \subsection{The first few tokens from an extended token list} % % \begin{function}[EXP]{\gtl_left_tl:N} % \begin{syntax} % \cs{gtl_left_tl:N} \meta{gtl~var} % \end{syntax} % Leaves in the input stream all tokens in \meta{gtl~var} until the % first extra begin-group or extra end-group character, within % \cs{exp_not:n}. This is the longest balanced token list starting % from the left of \meta{gtl~var}. % \end{function} % % \begin{function}{\gtl_pop_left_tl:N, \gtl_gpop_left_tl:N} % \begin{syntax} % \cs{gtl_pop_left_tl:N} \meta{gtl~var} % \end{syntax} % Remove from the \meta{gtl~var} all tokens before the first extra % begin-group or extra end-group character. The tokens that are % removed form the longest balanced token list starting from the left % of \meta{gtl~var}. % \end{function} % % \begin{function}[EXP]{\gtl_left_item:NF} % \begin{syntax} % \cs{gtl_left_item:NF} \meta{gtl~var} \Arg{false code} % \end{syntax} % Leaves in the input stream the first \meta{item} of the % \meta{gtl~var}: this is identical to \cs{tl_head:n} applied to the % result of \cs{gtl_left_tl:N}. If there is no such item, the % \meta{false code} is left in the input stream. % \end{function} % % \begin{function}[TF]{\gtl_pop_left_item:NN, \gtl_gpop_left_item:NN} % \begin{syntax} % \cs{gtl_pop_left_item:NNTF} \meta{gtl~var} \meta{tl~var} % \Arg{true~code} \Arg{false~code} % \end{syntax} % Stores the first item of \meta{gtl~var} in \meta{tl~var}, locally, % and removes it from \meta{gtl~var}, together with any space before % it. If there is no such item, the \meta{gtl~var} is not affected, % and the meta{tl~var} may or may not be affected. % \end{function} % % \begin{function}[EXP]{\gtl_left_text:NF} % \begin{syntax} % \cs{gtl_left_text:NF} \meta{gtl~var} \Arg{false code} % \end{syntax} % Starting from the first token in \meta{gtl~var}, this function finds % a pattern of the form \meta{tokens_1} \Arg{tokens_2}, where the % \meta{tokens_1} contain no begin-group nor end-group characters, % then leaves \meta{tokens_1} \Arg{tokens_2} in the input stream, % within \cs{exp_not:n}. If no such pattern exists (this happens if % the result of \cs{gtl_left_tl:N} contains no brace group), the % \meta{false code} is run instead. % \end{function} % % \begin{function}{\gtl_pop_left_text:N, \gtl_gpop_left_text:N} % \begin{syntax} % \cs{gtl_pop_left_text:N} \meta{gtl~var} % \end{syntax} % Starting from the first token in \meta{gtl~var}, this function finds % a pattern of the form \meta{tokens_1} \Arg{tokens_2}, where the % \meta{tokens_1} contain no begin-group nor end-group characters, % then removes \meta{tokens_1} \Arg{tokens_2} from \meta{gtl~var}. If % no such pattern exists (this happens if the result of % \cs{gtl_left_tl:N} contains no brace group), the \meta{gtl~var} is % not modified instead. % \end{function} % % \subsection{Working with the contents of extended token lists} % % \begin{function}[EXP]{\gtl_count_tokens:N} % \begin{syntax} % \cs{gtl_count_tokens:N} \meta{gtl~var} % \end{syntax} % Counts the number of tokens in the \meta{gtl~var} and leaves this % information in the input stream. % \end{function} % % \begin{function}[EXP]{\gtl_extra_begin:N, \gtl_extra_end:N} % \begin{syntax} % \cs{gtl_extra_begin:N} \meta{gtl~var} % \end{syntax} % Counts the number of explicit extra begin-group (or end-group) % characters in the \meta{gtl~var} and leaves this information in the % input stream. % \end{function} % % \begin{function}{\gtl_show:N, \gtl_log:N} % \begin{syntax} % \cs{gtl_show:N} \meta{gtl~var} % \end{syntax} % Displays the content of the \meta{gtl~var} on the terminal or in % the log file. % \end{function} % % \begin{function}[EXP, added = 2018-04-04]{\gtl_to_str:N, \gtl_to_str:n} % \begin{syntax} % \cs{gtl_to_str:N} \meta{gtl~var} % \end{syntax} % Converts the content of the \meta{gtl~var} into a series of % characters with category code $12$ (other) with the exception of % spaces, which retain category code $10$ (space). This string is % then left in the input stream. % \end{function} % % \subsection{Constant extended token lists} % % \begin{variable}{\c_empty_gtl} % Constant that is always empty. % \end{variable} % % \begin{variable}{\c_group_begin_gtl} % An explicit begin-group character contained in an extended token % list. % \end{variable} % % \begin{variable}{\c_group_end_gtl} % An explicit end-group character contained in an extended token list. % \end{variable} % % \subsection{Future perhaps} % % \begin{itemize} % \item Test if a token appears in an extended token list. % \item Test if an extended token list appears in another. % \item Remove an extended token list from another, once or every time % it appears. % \item Replace an extended token list by another in a third: once, or % every time it appears. % \item Case statement. % \item Mapping? % \item Inserting an extended token list into the input stream, with all % its glorious unbalanced braces. % \item Convert in various ways to a token list. % \item Reverse the order of tokens. % \item Extract a token given its position. % \item Extract a range of tokens given their position. % \item Trim spaces. % \item Crazy idea below. % \end{itemize} % % We could add (with lots of work) the expandable function % \begin{quote}\ttfamily\obeylines\obeyspaces % \cs{gtl_concat:nF} % \{ % \Arg{tl_1} \Arg{start_1} \Arg{stop_1} % \Arg{tl_2} \Arg{start_2} \Arg{stop_2} % \ldots{} % \Arg{tl_n} \Arg{start_n} \Arg{stop_n} % \} % \Arg{false code} % \end{quote} % For each triplet, this function builds the sub-token list of \meta{tl_i} % corresponding to the tokens ranging from position \meta{start_i} to % position \meta{stop_i} of \meta{tl_i}. The results obtained for each triplet % are then concatenated. If nothing bad happens (see below), the % concatenation is left in the input stream, and the \meta{false code} is % removed. Two cases can lead to running the \meta{false code} (and dropping % the first argument altogether). The first case is when the number of % brace groups in \cs{gtl_concat:nF} is not a multiple of 3. The second % case is when the concatenation gives rise to an unbalanced token list: % then the result is not a valid token list. Note that each part is % allowed to be unbalanced: only the full result must be balanced. % % \end{documentation} % % \begin{implementation} % % \section{\pkg{gtl} implementation} % % Some support packages are loaded first, then we declare the package's % name, date, version, and purpose. % % \begin{macrocode} %<*package> % \end{macrocode} % % \begin{macrocode} %<@@=gtl> % \end{macrocode} % % Load \pkg{expl3}, either through \cs{RequirePackage} or through % inputting the generic loader, depending on the format in use. % \begin{macrocode} \begingroup\expandafter\expandafter\expandafter\endgroup \expandafter\ifx\csname RequirePackage\endcsname\relax \input expl3-generic.tex \else \RequirePackage{expl3}[2017/11/14] \fi \ExplSyntaxOn \cs_if_exist:NTF \ProvidesExplPackage { \cs_new_eq:NN \@@_end_package_hook: \prg_do_nothing: \ExplSyntaxOff \ProvidesExplPackage } { \cs_new_eq:NN \@@_end_package_hook: \ExplSyntaxOff \group_begin: \ExplSyntaxOff \cs_set_protected:Npn \@@_tmp:w #1#2#3#4 { \group_end: \tl_gset:ce { ver @ #1 . sty } { #2 ~ v#3 ~ #4 } \cs_if_exist_use:NF \wlog { \iow_log:e } { Package: ~ #1 ~ #2 ~ v#3 ~ #4 } } \@@_tmp:w } {gtl} {2024/01/04} {0.6} {Manipulate unbalanced lists of tokens} % \end{macrocode} % % \subsection{Helpers} % % \begin{macrocode} \cs_generate_variant:Nn \use:nn { no } % \end{macrocode} % % \begin{macro}[EXP]{\@@_exp_not_n:N} % Used in one case where we need to prevent expansion of a token % within an |x|-expanding definition. Using \cs{exp_not:N} there % would fail when the argument is a macro parameter character. % \begin{macrocode} \cs_new:Npn \@@_exp_not_n:N #1 { \exp_not:n {#1} } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP]{\@@_brace:nn, \@@_brace_swap:nn} % Those functions are used to add some tokens, |#1|, to an item |#2| % in an extended token list: \cs{@@_brace:nn} adds tokens on the left, % while \cs{@@_brace_swap:nn} adds them on the right. % \begin{macrocode} \cs_new:Npn \@@_brace:nn #1#2 { { #1 #2 } } \cs_new:Npn \@@_brace_swap:nn #1#2 { { #2 #1 } } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP]{\@@_strip_nil_mark:w,} % \begin{macro}[EXP]{\@@_strip_nil_mark_aux:w} % Removes the following \cs{q_nil} \cs{q_mark} without losing any % braces, and places the result into \cs{exp_not:n}. % \begin{macrocode} \cs_new:Npn \@@_strip_nil_mark:w { \@@_strip_nil_mark_aux:w \prg_do_nothing: } \cs_new:Npn \@@_strip_nil_mark_aux:w #1 \q_nil \q_mark { \exp_not:o {#1} } % \end{macrocode} % \end{macro} % \end{macro} % % \subsection{Structure of extended token lists} % % Token lists must have balanced braces (or rather, begin-group and % end-group characters). Extended token lists lift this requirement, % and can represent arbitrary lists of tokens. A list of tokens can % fail to be balanced in two ways: one may encounter too many end-group % characters near the beginning of the list, or too many begin-group % characters near the end of the list. In fact, a list of tokens always % has the form % \begin{quote}\ttfamily % \meta{b_1} \} \ldots{} \meta{b_n} \} % \meta{c} % \{ \meta{e_1} \ldots{} \{ \meta{e_p} % \end{quote} % where the \meta{b_i}, \meta{c}, and \meta{e_i} are all balanced token % lists. This can be seen by listing the tokens, and keeping track of a % counter, which starts at~$0$, and is incremented at each begin-group % character, and decremented at each end-group character: then the % \meta{b_i} are delimited by positions where the counter reaches a new % minimum, whereas the \meta{e_i} are delimited by positions where the % counter last takes a given negative value. Such a token list is % stored as % \begin{quote}\ttfamily % \cs{s_@@} % \{ \Arg{b_1} \ldots{} \Arg{b_n} \} % \Arg{c} % \{ \Arg{e_p} \ldots{} \Arg{e_1} \} % \end{quote} % Note that the \meta{e_i} are in a reversed order, as this makes the % ends of extended token lists more accessible. Balanced token lists % have $n = p = 0$: the first and third parts are empty, while the % second contains the tokens. % % In the following code comments, the balanced token lists \meta{b_i} % are called ``leading chunks'', \meta{c} is called ``middle chunk'', % and \meta{e_i} are called ``trailing chunks''. It is important to % note that a balanced sub-list of a gtl must be entirely contained in % one of the chunk. % % \begin{variable}{\s_@@} % This marker appears at the start of extended token lists. % \begin{macrocode} \cs_new_eq:NN \s_@@ \scan_stop: % \end{macrocode} % \end{variable} % % \begin{macro} % { % \gtl_set:Nn, \gtl_gset:Nn, \gtl_const:Nn, % \gtl_set:Ne, \gtl_gset:Ne, \gtl_const:Ne, % \gtl_set:Nx, \gtl_gset:Nx, \gtl_const:Nx % } % Storing a balanced token list into an extended token list variable % simply means adding \cs{s_@@} and two empty brace % groups: there are no leading nor trailing chunks. % \begin{macrocode} \cs_new_protected:Npn \gtl_set:Nn { \@@_set:NNn \tl_set:Nn } \cs_new_protected:Npn \gtl_gset:Nn { \@@_set:NNn \tl_gset:Nn } \cs_new_protected:Npn \gtl_const:Nn { \@@_set:NNn \tl_const:Nn } \cs_generate_variant:Nn \gtl_set:Nn { Ne , Nx } \cs_generate_variant:Nn \gtl_gset:Nn { Ne , Nx } \cs_generate_variant:Nn \gtl_const:Nn { Ne , Nx } \cs_new_protected:Npn \@@_set:NNn #1#2#3 { #1 #2 { \s_@@ { } {#3} { } } } % \end{macrocode} % \end{macro} % % \begin{variable}{\c_empty_gtl} % An empty extended token list, obtained thanks to the % \cs{gtl_const:Nn} function just defined. % \begin{macrocode} \gtl_const:Nn \c_empty_gtl { } % \end{macrocode} % \end{variable} % % \begin{variable}{\c_group_begin_gtl, \c_group_end_gtl} % Extended token lists with exactly one begin-group/end-group % character are built by including a single (empty) leading or % trailing chunk. % \begin{macrocode} \tl_const:Nn \c_group_end_gtl { \s_@@ { { } } { } { } } \tl_const:Nn \c_group_begin_gtl { \s_@@ { } { } { { } } } % \end{macrocode} % \end{variable} % % \subsection{Creating extended token list variables} % % \begin{macro}{\gtl_new:N} % A new extended token list is created empty. % \begin{macrocode} \cs_new_protected:Npn \gtl_new:N #1 { \cs_new_eq:NN #1 \c_empty_gtl } % \end{macrocode} % \end{macro} % % \begin{macro}{\gtl_set_eq:NN, \gtl_gset_eq:NN} % All the data about an extended token list is stored as a single % token list, so copying is easy. % \begin{macrocode} \cs_new_eq:NN \gtl_set_eq:NN \tl_set_eq:NN \cs_new_eq:NN \gtl_gset_eq:NN \tl_gset_eq:NN % \end{macrocode} % \end{macro} % % \begin{macro}{\gtl_clear:N, \gtl_gclear:N} % Clearing an extended token list by setting it to the empty one. % \begin{macrocode} \cs_new_protected:Npn \gtl_clear:N #1 { \gtl_set_eq:NN #1 \c_empty_gtl } \cs_new_protected:Npn \gtl_gclear:N #1 { \gtl_gset_eq:NN #1 \c_empty_gtl } % \end{macrocode} % \end{macro} % % \begin{macro}{\gtl_clear_new:N, \gtl_gclear_new:N} % If the variable exists, clear it. Otherwise declare it. % \begin{macrocode} \cs_new_protected:Npn \gtl_clear_new:N #1 { \gtl_if_exist:NTF #1 { \gtl_clear:N #1 } { \gtl_new:N #1 } } \cs_new_protected:Npn \gtl_gclear_new:N #1 { \gtl_if_exist:NTF #1 { \gtl_gclear:N #1 } { \gtl_new:N #1 } } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP, pTF]{\gtl_if_exist:N} % Again a copy of token list functions. % \begin{macrocode} \prg_new_eq_conditional:NNn \gtl_if_exist:N \tl_if_exist:N { p , T , F , TF } % \end{macrocode} % \end{macro} % % \subsection{Adding data to extended token list variables} % % \begin{macro}{\gtl_put_left:Nn, \gtl_gput_left:Nn} % \begin{macro}[rEXP]{\@@_put_left:wn} % If there is no leading chunk in the gtl variable, then add the new % material to the middle chunk. Otherwise add it to the first % leading chunk, namely the first brace group in the first argument % of \cs{@@_put_left:wn}. % \begin{macrocode} \cs_new_protected:Npn \gtl_put_left:Nn #1#2 { \tl_set:Ne #1 { \exp_after:wN \@@_put_left:wn #1 {#2} } } \cs_new_protected:Npn \gtl_gput_left:Nn #1#2 { \tl_gset:Ne #1 { \exp_after:wN \@@_put_left:wn #1 {#2} } } \cs_new:Npn \@@_put_left:wn \s_@@ #1#2#3 #4 { \tl_if_empty:nTF {#1} { \exp_not:n { \s_@@ { } { #4 #2 } {#3} } } { \s_@@ { \exp_not:o { \@@_brace:nn {#4} #1 } } { \exp_not:n {#2} } { \exp_not:n {#3} } } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}{\gtl_put_right:Nn, \gtl_gput_right:Nn} % \begin{macro}[rEXP]{\@@_put_right:wn} % Symmetric of \cs{gtl_put_left:Nn}. % \begin{macrocode} \cs_new_protected:Npn \gtl_put_right:Nn #1#2 { \tl_set:Ne #1 { \exp_after:wN \@@_put_right:wn #1 {#2} } } \cs_new_protected:Npn \gtl_gput_right:Nn #1#2 { \tl_gset:Ne #1 { \exp_after:wN \@@_put_right:wn #1 {#2} } } \cs_new:Npn \@@_put_right:wn \s_@@ #1#2#3 #4 { \tl_if_empty:nTF {#3} { \exp_not:n { \s_@@ {#1} { #2 #4 } { } } } { \s_@@ { \exp_not:n {#1} } { \exp_not:n {#2} } { \exp_not:o { \@@_brace_swap:nn {#4} #3 } } } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}{\gtl_concat:NNN, \gtl_gconcat:NNN} % \begin{macro}[rEXP] % { % \@@_concat:ww, % \@@_concat_aux:nnnnnn, % \@@_concat_auxi:nnnnnn, % \@@_concat_auxii:nnnnnn, % \@@_concat_auxiii:w, % \@@_concat_auxiv:nnnn, % \@@_concat_auxv:wnwnn, % \@@_concat_auxvi:nnwnwnn % } % Concatenating two lists of tokens of the form % \begin{quote}\ttfamily % \cs{s_@@} % \{ \Arg{b_1} \ldots{} \Arg{b_n} \} % \Arg{c} % \{ \Arg{e_p} \ldots{} \Arg{e_1} \} % \end{quote} % is not an easy task. The \meta{e} parts of the first join with the % \meta{b} parts of the second to make balanced pairs, and the % follow-up depends on whether there were more \meta{e} parts or more % \meta{b} parts. % \begin{macrocode} \cs_new_protected:Npn \gtl_concat:NNN #1#2#3 { \tl_set:Ne #1 { \exp_last_two_unbraced:Noo \@@_concat:ww #2 #3 } } \cs_new_protected:Npn \gtl_gconcat:NNN #1#2#3 { \tl_gset:Ne #1 { \exp_last_two_unbraced:Noo \@@_concat:ww #2 #3 } } \cs_new:Npn \@@_concat:ww \s_@@ #1#2#3 \s_@@ #4#5#6 { \tl_if_blank:nTF {#3} { \tl_if_blank:nTF {#4} { \@@_concat_aux:nnnnnn } { \@@_concat_auxi:nnnnnn } } { \tl_if_blank:nTF {#4} { \@@_concat_auxii:nnnnnn } { \@@_concat_auxiv:nnnn } } {#1} {#2} {#3} {#4} {#5} {#6} } \cs_new:Npn \@@_concat_aux:nnnnnn #1#2#3#4#5#6 { \exp_not:n { \s_@@ {#1} { #2 #5 } {#6} } } \cs_new:Npn \@@_concat_auxi:nnnnnn #1#2#3#4#5#6 { \s_@@ { \exp_not:n {#1} \exp_not:f { \@@_concat_auxiii:w \@@_brace:nn {#2} #4 ~ \q_stop } } { \exp_not:n {#5} } { \exp_not:n {#6} } } \cs_new:Npn \@@_concat_auxii:nnnnnn #1#2#3#4#5#6 { \s_@@ { \exp_not:n {#1} } { \exp_not:n {#2} } { \exp_not:n {#6} \exp_not:f { \@@_concat_auxiii:w \@@_brace_swap:nn {#5} #3 ~ \q_stop } } } \cs_new:Npn \@@_concat_auxiii:w #1 ~ #2 \q_stop {#1} \cs_new:Npn \@@_concat_auxiv:nnnn #1#2#3#4 { \tl_if_single:nTF {#3} { \@@_concat_auxv:wnwnn } { \@@_concat_auxvi:nnwnwnn } #3 ~ \q_mark #4 ~ \q_mark {#1} {#2} } \cs_new:Npn \@@_concat_auxv:wnwnn #1#2 \q_mark #3#4 \q_mark #5#6 { \@@_concat:ww \s_@@ {#5} { #6 { #1 #3 } } { } \s_@@ {#4} } \cs_new:Npn \@@_concat_auxvi:nnwnwnn #1#2#3 \q_mark #4#5 \q_mark #6#7 { \@@_concat:ww \s_@@ {#6} {#7} { { #2 { #1 #4 } } #3 } \s_@@ {#5} } % \end{macrocode} % \end{macro} % \end{macro} % % \subsection{Showing extended token lists} % % \begin{macro}[EXP]{\gtl_to_str:N, \gtl_to_str:n} % \begin{macro}[EXP] % { % \@@_to_str:w, % \@@_to_str_loopi:nnw, % \@@_to_str_testi:nnw, % \@@_to_str_endi:nnn, % \@@_to_str_loopii:nnw, % \@@_to_str_endii:nnw % } % \begin{macrocode} \cs_new:Npn \gtl_to_str:N #1 { \exp_after:wN \@@_to_str:w #1 } \cs_new:Npn \gtl_to_str:n #1 { \@@_to_str:w #1 } \cs_new:Npn \@@_to_str:w \s_@@ #1#2#3 { \@@_to_str_loopi:nnw { } #1 \q_nil \q_mark {#2} {#3} } \cs_new:Npe \@@_to_str_loopi:nnw #1#2 { \exp_not:N \quark_if_nil:nTF {#2} { \exp_not:N \@@_to_str_testi:nnw {#1} {#2} } { \exp_not:N \@@_to_str_loopi:nnw { #1 #2 \iow_char:N \} } } } \cs_new:Npe \@@_to_str_testi:nnw #1#2#3 \q_mark { \exp_not:N \tl_if_empty:nTF {#3} { \exp_not:N \@@_to_str_endi:nnn {#1} } { \exp_not:N \@@_to_str_loopi:nnw { #1 #2 \iow_char:N \} } #3 \exp_not:N \q_mark } } \cs_new:Npn \@@_to_str_endi:nnn #1#2#3 { \@@_to_str_loopii:nnw #3 { #1 #2 } \q_nil \q_stop } \cs_new:Npe \@@_to_str_loopii:nnw #1#2 { \exp_not:N \quark_if_nil:nTF {#2} { \exp_not:N \@@_to_str_testii:nnw {#1} {#2} } { \exp_not:N \@@_to_str_loopii:nnw { #2 \iow_char:N \{ #1 } } } \cs_new:Npe \@@_to_str_testii:nnw #1#2#3 \q_stop { \exp_not:N \tl_if_empty:nTF {#3} { \exp_not:N \tl_to_str:n {#1} } { \exp_not:N \@@_to_str_loopii:nnw { #2 \iow_char:N \{ #1 } #3 \exp_not:N \q_stop } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}{\gtl_show:N, \gtl_log:N} % \begin{macro}{\@@_show:NNN} % Display the variable name, then its string representation. Before % that, test that the variable indeed exists, and if appropriate % throw an error message by sending the undefined variable to % \cs{tl_show:N} or \cs{tl_log:N}. % \begin{macrocode} \cs_new_protected:Npn \gtl_show:N { \@@_show:NNN \tl_show:n \tl_show:N } \cs_new_protected:Npn \gtl_log:N { \@@_show:NNN \tl_log:n \tl_log:N } \cs_new_protected:Npn \@@_show:NNN #1#2#3 { \gtl_if_exist:NTF #3 { \exp_args:Ne #1 { \token_to_str:N #3 = \gtl_to_str:N #3 } } { #2 #3 } } % \end{macrocode} % \end{macro} % \end{macro} % % \subsection{Extended token list conditionals} % % \begin{macro}[EXP, pTF]{\gtl_if_eq:NN} % Two extended token lists are equal if the underlying token lists % are the same. % \begin{macrocode} \prg_new_eq_conditional:NNn \gtl_if_eq:NN \tl_if_eq:NN { p , T , F , TF } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP, pTF]{\gtl_if_empty:N} % An extended token list is empty if it is equal to the empty gtl. % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_empty:N #1 { p , T , F , TF } { \tl_if_eq:NNTF #1 \c_empty_gtl { \prg_return_true: } { \prg_return_false: } } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP, pTF]{\gtl_if_tl:N} % \begin{macro}[EXP]{\@@_if_tl_return:w} % A gtl is balanced if it has neither leading nor trailing chunk. % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_tl:N #1 { p , T , F , TF } { \exp_after:wN \@@_if_tl_return:w #1 } \cs_new:Npn \@@_if_tl_return:w \s_@@ #1#2#3 { \tl_if_empty:nTF { #1 #3 } { \prg_return_true: } { \prg_return_false: } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}[EXP, pTF]{\gtl_if_single_token:N} % \begin{macro}[EXP]{\@@_if_single_token_return:w} % If there are neither leading nor trailing chunks then the gtl is a % single token if and only if the middle chunk is a single token. % Otherwise the gtl is a single token only if it is exactly a % begin-group or an end-group token. % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_single_token:N #1 { p , T , F , TF } { \exp_after:wN \@@_if_single_token_return:w #1 #1 } \cs_new:Npn \@@_if_single_token_return:w \s_@@ #1#2#3 #4 { \tl_if_empty:nTF { #1 #3 } { \tl_if_single_token:nTF {#2} { \prg_return_true: } { \prg_return_false: } } { \gtl_if_eq:NNTF #4 \c_group_begin_gtl { \prg_return_true: } { \gtl_if_eq:NNTF #4 \c_group_end_gtl { \prg_return_true: } { \prg_return_false: } } } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}[EXP,pTF]{\gtl_if_blank:N} % \begin{macro}[EXP]{\@@_if_blank_return:w} % A gtl is blank if its middle chunk is blank and it has no leading % nor trailing chunk (those would lead to |#1| or |#3| containing % brace groups). % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_blank:N #1 { p , T , F , TF } { \exp_after:wN \@@_if_blank_return:w #1 } \cs_new:Npn \@@_if_blank_return:w \s_@@ #1#2#3 { \tl_if_blank:nTF { #1 #2 #3 } { \prg_return_true: } { \prg_return_false: } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}[EXP,pTF] % { % \gtl_if_head_is_group_begin:N, % \gtl_if_head_is_group_end:N, % \gtl_if_head_is_space:N, % \gtl_if_head_is_N_type:N, % } % Based on a five-way test \cs{@@_head:wnnnnn} defined later. The % five cases are: the gtl is empty, it starts with a begin-group, % with an end-group, with a space, or with an N-type token. In the % last case, the token is left in the input stream after the brace % group, hence the need for \cs{use_none:n} here. % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_head_is_group_begin:N #1 { p , T , F , TF } { \exp_after:wN \@@_head:wnnnnn #1 { \prg_return_false: } { \prg_return_true: } { \prg_return_false: } { \prg_return_false: } { \prg_return_false: \use_none:n } } \prg_new_conditional:Npnn \gtl_if_head_is_group_end:N #1 { p , T , F , TF } { \exp_after:wN \@@_head:wnnnnn #1 { \prg_return_false: } { \prg_return_false: } { \prg_return_true: } { \prg_return_false: } { \prg_return_false: \use_none:n } } \prg_new_conditional:Npnn \gtl_if_head_is_space:N #1 { p , T , F , TF } { \exp_after:wN \@@_head:wnnnnn #1 { \prg_return_false: } { \prg_return_false: } { \prg_return_false: } { \prg_return_true: } { \prg_return_false: \use_none:n } } \prg_new_conditional:Npnn \gtl_if_head_is_N_type:N #1 { p , T , F , TF } { \exp_after:wN \@@_head:wnnnnn #1 { \prg_return_false: } { \prg_return_false: } { \prg_return_false: } { \prg_return_false: } { \prg_return_true: \use_none:n } } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP, pTF]{\gtl_if_head_eq_catcode:NN} % \begin{macro}[EXP, pTF]{\gtl_if_head_eq_charcode:NN} % \begin{macro}[EXP]{\@@_if_head_eq_code_return:NNN} % In the empty case, |?| can match with |#2|, but then % \cs{use_none:nn} gets rid of \cs{prg_return_true:} and \cs{else:}, % to correctly leave \cs{prg_return_false:}. We could not simplify % this by placing the \cs{exp_not:N} |#2| after the construction % involving~|#1|, because |#2|~must be taken into the \TeX{} primitive % test, in case |#2|~itself is a primitive \TeX{} conditional, which % would mess up conditional nesting. % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_head_eq_catcode:NN #1#2 { p , T , F , TF } { \@@_if_head_eq_code_return:NNN \if_catcode:w #1#2 } \prg_new_conditional:Npnn \gtl_if_head_eq_charcode:NN #1#2 { p , T , F , TF } { \@@_if_head_eq_code_return:NNN \if_charcode:w #1#2 } \cs_new:Npn \@@_if_head_eq_code_return:NNN #1#2#3 { #1 \exp_not:N #3 \exp_after:wN \@@_head:wnnnnn #2 { ? \use_none:nn } { \c_group_begin_token } { \c_group_end_token } { \c_space_token } { \exp_not:N } \prg_return_true: \else: \prg_return_false: \fi: } % \end{macrocode} % \end{macro} % \end{macro} % \end{macro} % % \begin{macro}[EXP, pTF]{\gtl_if_head_eq_meaning:NN} % \begin{macro}[EXP]{\@@_if_head_eq_meaning_return:NN} % \begin{macrocode} \prg_new_conditional:Npnn \gtl_if_head_eq_meaning:NN #1#2 { p , T , F , TF } { \@@_if_head_eq_meaning_return:NN #1#2 } \cs_new:Npn \@@_if_head_eq_meaning_return:NN #1#2 { \exp_after:wN \@@_head:wnnnnn #1 { \if_false: } { \if_meaning:w #2 \c_group_begin_token } { \if_meaning:w #2 \c_group_end_token } { \if_meaning:w #2 \c_space_token } { \if_meaning:w #2 } \prg_return_true: \else: \prg_return_false: \fi: } % \end{macrocode} % \end{macro} % \end{macro} % % \subsection{First token of an extended token list} % % \begin{macro}[EXP]{\@@_head:wnnnnn} % \begin{macro}[EXP]{\@@_head_aux:nwnnnn} % \begin{macro}[EXP]{\@@_head_auxii:N, \@@_head_auxiii:Nnn} % This function performs |#4|~if the \texttt{gtl} is empty, |#5|~if it % starts with a begin-group character, |#6|~if it starts with an % end-group character, |#7|~if it starts with a space, and in other % cases (when the first token is |N|-type), it performs |#8|~followed % by the first token. % \begin{macrocode} \cs_new:Npn \@@_head:wnnnnn \s_@@ #1#2#3 #4#5#6#7#8 { \tl_if_empty:nTF {#1} { \tl_if_empty:nTF {#2} { \tl_if_empty:nTF {#3} {#4} {#5} } { \@@_head_aux:nwnnnn {#2} \q_stop {#5} {#6} {#7} {#8} } } { \@@_head_aux:nwnnnn #1 \q_stop {#5} {#6} {#7} {#8} } } \cs_new:Npn \@@_head_aux:nwnnnn #1#2 \q_stop #3#4#5#6 { \tl_if_head_is_group:nTF {#1} {#3} { \tl_if_empty:nTF {#1} {#4} { \tl_if_head_is_space:nTF {#1} {#5} { \if_false: { \fi: \@@_head_auxii:N #1 } {#6} } } } } \cs_new:Npn \@@_head_auxii:N #1 { \exp_after:wN \@@_head_auxiii:Nnn \exp_after:wN #1 \exp_after:wN { \if_false: } \fi: } \cs_new:Npn \@@_head_auxiii:Nnn #1#2#3 { #3 #1 } % \end{macrocode} % \end{macro} % \end{macro} % \end{macro} % % \begin{macro}[EXP]{\gtl_head:N} % If |#1| is empty, do nothing. If it starts with a begin-group % character or an end-group character leave the appropriate brace % (thanks to \cs{if_false:} tricks). If it starts with a space, leave % that, and finally if it starts with a normal token, leave it, within % \cs{exp_not:n}. % \begin{macrocode} \cs_new:Npn \gtl_head:N #1 { \exp_after:wN \@@_head:wnnnnn #1 { } { \exp_after:wN { \if_false: } \fi: } { \if_false: { \fi: } } { ~ } { \@@_exp_not_n:N } } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP]{\gtl_head_do:NN} % Similar to \cs{gtl_head:N}, but inserting |#2| before the resulting token. % \begin{macrocode} \cs_new:Npn \gtl_head_do:NN #1#2 { \exp_after:wN \@@_head:wnnnnn #1 { #2 \q_no_value } { \exp_after:wN #2 \exp_after:wN { \if_false: } \fi: } { \if_false: { \fi: #2 } } { #2 ~ } { #2 } } % \end{macrocode} % \end{macro} % % \begin{macro}[EXP, TF]{\gtl_head_do:NN} % Test for emptyness then use \cs{gtl_head_do:NN}, placing the % \meta{true code} or \meta{false code} as appropriate. % \begin{macrocode} \cs_new:Npn \gtl_head_do:NNT #1#2#3 { \gtl_if_empty:NTF #1 { } { \gtl_head_do:NN #1 #2 #3 } } \cs_new:Npn \gtl_head_do:NNF #1#2#3 { \gtl_if_empty:NTF #1 {#3} { \gtl_head_do:NN #1 #2 } } \cs_new:Npn \gtl_head_do:NNTF #1#2#3#4 { \gtl_if_empty:NTF #1 {#4} { \gtl_head_do:NN #1 #2 #3 } } % \end{macrocode} % \end{macro} % % \begin{macro}{\gtl_get_left:NN} % \begin{macrocode} \cs_new_protected:Npn \gtl_get_left:NN #1#2 { \exp_after:wN \@@_head:wnnnnn #1 { \gtl_set:Nn #2 { \q_no_value } } { \gtl_set_eq:NN #2 \c_group_begin_gtl } { \gtl_set_eq:NN #2 \c_group_end_gtl } { \gtl_set:Nn #2 { ~ } } { \gtl_set:Nn #2 } } % \end{macrocode} % \end{macro} % % \begin{macro}{\gtl_pop_left:N, \gtl_gpop_left:N} % \begin{macro}[rEXP] % { % \@@_pop_left:w, % \@@_pop_left_auxi:n, % \@@_pop_left_auxii:nnnw, % \@@_pop_left_auxiii:nnnw, % \@@_pop_left_auxiv:nn, % \@@_pop_left_auxv:nnn, % \@@_pop_left_auxvi:n % } % \begin{macrocode} \cs_new_protected:Npn \gtl_pop_left:N #1 { \gtl_if_empty:NF #1 { \tl_set:Ne #1 { \exp_after:wN \@@_pop_left:w #1 } } } \cs_new_protected:Npn \gtl_gpop_left:N #1 { \gtl_if_empty:NF #1 { \tl_gset:Ne #1 { \exp_after:wN \@@_pop_left:w #1 } } } \cs_new:Npn \@@_pop_left:w \s_@@ #1#2#3 { \tl_if_empty:nTF {#1} { \tl_if_empty:nTF {#2} { \@@_pop_left_auxi:n {#3} } { \@@_pop_left_auxiv:nn {#2} {#3} } } { \@@_pop_left_auxv:nnn {#1} {#2} {#3} } } \cs_new:Npn \@@_pop_left_auxi:n #1 { \s_@@ { } \@@_pop_left_auxii:nnnw { } { } #1 \q_nil \q_stop } \cs_new:Npn \@@_pop_left_auxii:nnnw #1#2#3 { \quark_if_nil:nTF {#3} { \@@_pop_left_auxiii:nnnw {#1} {#2} {#3} } { \@@_pop_left_auxii:nnnw { #1 #2 } { {#3} } } } \cs_new:Npn \@@_pop_left_auxiii:nnnw #1#2#3#4 \q_stop { \tl_if_empty:nTF {#4} { \exp_not:n { #2 {#1} } } { \@@_pop_left_auxii:nnnw { #1 #2 } { {#3} } } } \cs_new:Npn \@@_pop_left_auxiv:nn #1#2 { \s_@@ { \tl_if_head_is_group:nT {#1} { { \tl_head:n {#1} } } } { \tl_if_head_is_space:nTF {#1} { \exp_not:f } { \tl_tail:n } {#1} } { \exp_not:n {#2} } } \cs_new:Npn \@@_pop_left_auxv:nnn #1#2#3 { \s_@@ { \if_false: { \fi: \@@_pop_left_auxvi:n #1 } } { \exp_not:n {#2} } { \exp_not:n {#3} } } \cs_new:Npn \@@_pop_left_auxvi:n #1 { \tl_if_empty:nF {#1} { \tl_if_head_is_group:nT {#1} { { \tl_head:n {#1} } } { \tl_if_head_is_space:nTF {#1} { \exp_not:f } { \tl_tail:n } {#1} } } \exp_after:wN \exp_not:n \exp_after:wN { \if_false: } \fi: } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}{\gtl_pop_left:NN, \gtl_gpop_left:NN} % Getting the first token and removing it from the extended token list % is done in two steps. % \begin{macrocode} \cs_new_protected:Npn \gtl_pop_left:NN #1#2 { \gtl_get_left:NN #1 #2 \gtl_pop_left:N #1 } \cs_new_protected:Npn \gtl_gpop_left:NN #1#2 { \gtl_get_left:NN #1 #2 \gtl_gpop_left:N #1 } % \end{macrocode} % \end{macro} % % \subsection{Longest token list starting an extended token list} % % \begin{macro}[EXP]{\gtl_left_tl:N} % \begin{macro}[EXP]{\@@_left_tl:w} % If there is no leading chunk, return the middle chunk, otherwise % the first leading chunk. % \begin{macrocode} \cs_new:Npn \gtl_left_tl:N #1 { \exp_after:wN \@@_left_tl:w #1 } \cs_new:Npn \@@_left_tl:w \s_@@ #1#2#3 { \tl_if_empty:nTF {#1} { \exp_not:n {#2} } { \tl_head:n {#1} } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}{\gtl_pop_left_tl:N, \gtl_gpop_left_tl:N} % If there is no left chunk, remove the middle chunk, hence the % resulting gtl will start with two empty brace groups (one for the % absence of leading chunk, and one for the emptyness of the middle % chunk). If there are left chunks replace the first one by an empty % chunk. % \begin{macrocode} \cs_new_protected:Npn \gtl_pop_left_tl:N #1 { \tl_set:Ne #1 { \exp_after:wN \@@_pop_left_tl:w #1 } } \cs_new_protected:Npn \gtl_gpop_left_tl:N #1 { \tl_gset:Ne #1 { \exp_after:wN \@@_pop_left_tl:w #1 } } \cs_new:Npn \@@_pop_left_tl:w \s_@@ #1#2#3 { \s_@@ \tl_if_empty:nTF {#1} { { } { } } { { { } \tl_tail:n {#1} } { \exp_not:n {#2} } } { \exp_not:n {#3} } } % \end{macrocode} % \end{macro} % % \subsection{First item of an extended token list} % % \begin{macro}[EXP]{\gtl_left_item:NF} % \begin{macro}[EXP]{\@@_left_item:wF, \@@_left_item_auxi:nwF} % The left-most item of an extended token list is the head of its left % token list. The code thus starts like \cs{gtl_left_tl:N}. It ends % with a check to test if we should use the head, or issue the false % code. % \begin{macrocode} \cs_new:Npn \gtl_left_item:NF #1 { \exp_after:wN \@@_left_item:wF #1 } \cs_new:Npn \@@_left_item:wF \s_@@ #1#2#3 { \@@_left_item_auxi:nwF #1 {#2} \q_stop } \cs_new:Npn \@@_left_item_auxi:nwF #1#2 \q_stop #3 { \tl_if_blank:nTF {#1} {#3} { \tl_head:n {#1} } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}[TF]{\gtl_pop_left_item:NN, \gtl_gpop_left_item:NN} % \begin{macro} % {\@@_pop_left_item:wNNN, \@@_pop_left_item_aux:nwnnNNN} % If there is no extra end-group characters, and if the balanced part % is blank, we cannot extract an item: return \texttt{false}. If the % balanced part is not blank, store its first item into |#4|, and % store the altered generalized token list into~|#6|, locally or % globally. Otherwise, pick out the part before the first extra % end-group character as |#1| of the second auxiliary, and do % essentially the same: if it is blank, there is no item, and if it is % not blank, pop its first item. % \begin{macrocode} \prg_new_protected_conditional:Npnn \gtl_pop_left_item:NN #1#2 { TF , T , F } { \exp_after:wN \@@_pop_left_item:wNNN #1#2 \tl_set:Ne #1 } \prg_new_protected_conditional:Npnn \gtl_gpop_left_item:NN #1#2 { TF , T , F } { \exp_after:wN \@@_pop_left_item:wNNN #1#2 \tl_gset:Ne #1 } \cs_new_protected:Npn \@@_pop_left_item:wNNN \s_@@ #1#2#3 #4#5#6 { \tl_if_empty:nTF {#1} { \tl_if_blank:nTF {#2} { \prg_return_false: } { \tl_set:Ne #4 { \tl_head:n {#2} } #5 #6 { \s_@@ { } { \tl_tail:n {#2} } { \exp_not:n {#3} } } \prg_return_true: } } { \@@_pop_left_item_aux:nwnnNNN #1 \q_nil \q_stop {#2} {#3} #4 #5 #6 } } \cs_new_protected:Npn \@@_pop_left_item_aux:nwnnNNN #1#2 \q_stop #3#4#5#6#7 { \tl_if_blank:nTF {#1} { \prg_return_false: } { \tl_set:Ne #5 { \tl_head:n {#1} } #6 #7 { \s_@@ { { \tl_tail:n {#1} } \@@_strip_nil_mark:w #2 \q_mark } { \exp_not:n {#3} } { \exp_not:n {#4} } } \prg_return_true: } } % \end{macrocode} % \end{macro} % \end{macro} % % \subsection{First group in an extended token list} % % The functions of this section extract from an extended token list the % tokens that would be absorbed after |\def\foo|, namely tokens with no % begin-group nor end-group characters, followed by one group. Those % tokens are either left in the input stream or stored in a token list % variable, and the |pop| functions also remove those tokens from the % extended token list variable. % % \begin{macro}[EXP]{\gtl_left_text:NF} % \begin{macro}[EXP] % { % \@@_left_text:wF, % \@@_left_text_auxi:nwF, % \@@_left_text_auxii:wnwF, % \@@_left_text_auxiii:nnwF, % } % \begin{macrocode} \cs_new:Npn \gtl_left_text:NF #1 { \exp_after:wN \@@_left_text:wF #1 } \cs_new:Npn \@@_left_text:wF \s_@@ #1#2#3 { \tl_if_empty:nTF {#1} { \@@_left_text_auxi:nwF {#2} \q_stop } { \@@_left_text_auxi:nwF #1 \q_stop } } \cs_new:Npn \@@_left_text_auxi:nwF #1#2 \q_stop { \@@_left_text_auxii:wnwF #1 \q_mark { } \q_mark \q_stop } \cs_new:Npn \@@_left_text_auxii:wnwF #1 # { \@@_left_text_auxiii:nnwF {#1} } \cs_new:Npn \@@_left_text_auxiii:nnwF #1#2 #3 \q_mark #4 \q_stop #5 { \tl_if_empty:nTF {#4} {#5} { \exp_not:n { #1 {#2} } } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}{\gtl_pop_left_text:N, \gtl_gpop_left_text:N} % \begin{macro}[rEXP] % { % \@@_pop_left_text:w, % \@@_pop_left_text_auxi:n, % \@@_pop_left_text_auxii:wnw, % \@@_pop_left_text_auxiii:nnw, % \@@_pop_left_text_auxiv:nw, % } % \begin{macrocode} \cs_new_protected:Npn \gtl_pop_left_text:N #1 { \tl_set:Ne #1 { \exp_after:wN \@@_pop_left_text:w #1 } } \cs_new_protected:Npn \gtl_gpop_left_text:N #1 { \tl_gset:Ne #1 { \exp_after:wN \@@_pop_left_text:w #1 } } \cs_new:Npn \@@_pop_left_text:w \s_@@ #1#2#3 { \s_@@ \tl_if_empty:nTF {#1} { { } { \@@_pop_left_text_auxi:n {#2} } } { { \@@_pop_left_text_auxiv:nw #1 \q_nil \q_mark } { \exp_not:n {#2} } } { \exp_not:n {#3} } } \cs_new:Npn \@@_pop_left_text_auxi:n #1 { \@@_pop_left_text_auxii:wnw #1 \q_nil \q_mark { } \q_mark \q_stop } \cs_new:Npn \@@_pop_left_text_auxii:wnw #1 # { \@@_pop_left_text_auxiii:nnw {#1} } \cs_new:Npn \@@_pop_left_text_auxiii:nnw #1#2#3 \q_mark #4 \q_stop { \tl_if_empty:nTF {#4} { \@@_strip_nil_mark:w #1 } { \@@_strip_nil_mark:w #3 \q_mark } } \cs_new:Npn \@@_pop_left_text_auxiv:nw #1 { { \@@_pop_left_text_auxi:n {#1} } \@@_strip_nil_mark:w } % \end{macrocode} % \end{macro} % \end{macro} % % \subsection{Counting tokens} % % \begin{macro}[EXP]{\@@_tl_count:n} % \begin{macro}[EXP]{\@@_tl_count_loop:n, \@@_tl_count_test:w} % A more robust version of \cs{tl_count:n}, which will however break % if the token list contains \cs{q_stop} at the outer brace level. % This cannot happen when \cs{@@_tl_count:n} is called with lists of % braced items. The technique is to loop, and when seeing % \cs{q_mark}, make sure that this is really the end of the list. % \begin{macrocode} \cs_new:Npn \@@_tl_count:n #1 { \int_eval:n { 0 \@@_tl_count_loop:n #1 \q_nil \q_stop } } \cs_new:Npn \@@_tl_count_loop:n #1 { \quark_if_nil:nTF {#1} { \@@_tl_count_test:w } { + 1 \@@_tl_count_loop:n } } \cs_new:Npn \@@_tl_count_test:w #1 \q_stop { \tl_if_empty:nF {#1} { + 1 \@@_tl_count_loop:n #1 \q_stop } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}[EXP]{\gtl_extra_begin:N, \gtl_extra_end:N} % \begin{macro}[EXP]{\@@_extra_begin:w, \@@_extra_end:w} % Count the number of extra end-group or of extra begin-group % characters in an extended token list. This is the number of items % in the first or third brace groups. We cannot use \cs{tl_count:n}, % as \pkg{gtl} is meant to be robust against inclusion of quarks. % \begin{macrocode} \cs_new:Npn \gtl_extra_end:N #1 { \exp_after:wN \@@_extra_end:w #1 } \cs_new:Npn \@@_extra_end:w \s_@@ #1#2#3 { \@@_tl_count:n {#1} } \cs_new:Npn \gtl_extra_begin:N #1 { \exp_after:wN \@@_extra_begin:w #1 } \cs_new:Npn \@@_extra_begin:w \s_@@ #1#2#3 { \@@_tl_count:n {#3} } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macro}[EXP]{\gtl_count_tokens:N} % \begin{macro}[rEXP] % { % \@@_count_tokens:w, % \@@_count_auxi:nw, % \@@_count_auxii:w, % \@@_count_auxiii:n % } % \begin{macrocode} \cs_new:Npn \gtl_count_tokens:N #1 { \exp_after:wN \@@_count_tokens:w #1 } \cs_new:Npn \@@_count_tokens:w \s_@@ #1#2#3 { \int_eval:n { -1 \@@_count_auxi:nw #1 {#2} #3 \q_nil \q_stop } } \cs_new:Npn \@@_count_auxi:nw #1 { \quark_if_nil:nTF {#1} { \@@_count_auxii:w } { + 1 \@@_count_auxiii:n {#1} \@@_count_auxi:nw } } \cs_new:Npn \@@_count_auxii:w #1 \q_stop { \tl_if_empty:nF {#1} { + 2 \@@_count_auxi:nw #1 \q_stop } } \cs_new:Npn \@@_count_auxiii:n #1 { \tl_if_empty:nF {#1} { \tl_if_head_is_group:nTF {#1} { + 2 \exp_args:No \@@_count_auxiii:n { \use:n #1 } } { + 1 \tl_if_head_is_N_type:nTF {#1} { \exp_args:No \@@_count_auxiii:n { \use_none:n #1 } } { \exp_args:Nf \@@_count_auxiii:n {#1} } } } } % \end{macrocode} % \end{macro} % \end{macro} % % \begin{macrocode} \@@_end_package_hook: % \end{macrocode} % % \begin{macrocode} % % \end{macrocode} % % \end{implementation} % % \endinput