% \iffalse meta-comment
%
%% File: tagpdf-mc-generic.dtx
%
% Copyright (C) 2019-2025 Ulrike Fischer
%
% It may be distributed and/or modified under the conditions of the
% LaTeX Project Public License (LPPL), either version 1.3c of this
% license or (at your option) any later version. The latest version
% of this license is in the file
%
% https://www.latex-project.org/lppl.txt
%
% This file is part of the "tagpdf bundle" (The Work in LPPL)
% and all files in that bundle must be distributed together.
%
% -----------------------------------------------------------------------
%
% The development version of the bundle can be found at
%
% https://github.com/latex3/tagpdf
%
% for those people who are interested.
%<*driver>
\DocumentMetadata{}
\documentclass{l3doc}
\usepackage{array,booktabs,caption}
\hypersetup{pdfauthor=Ulrike Fischer,
pdftitle=tagpdf-mc module (tagpdf)}
\begin{document}
\DocInput{\jobname.dtx}
\end{document}
%
% \fi
% \title{^^A
% The \pkg{tagpdf-mc-generic} module\\ Code related to Marked Content (mc-chunks), generic mode ^^A
% \\ Part of the tagpdf package
% }
%
% \author{^^A
% Ulrike Fischer\thanks
% {^^A
% E-mail:
% \href{mailto:fischer@troubleshooting-tex.de}
% {fischer@troubleshooting-tex.de}^^A
% }^^A
% }
%
% \date{Version 0.99l, released 2025-01-12}
% \maketitle
% \begin{documentation}
% \end{documentation}
% \begin{implementation}
% \section{Marked content code -- generic mode}
% \begin{macrocode}
%<@@=tag>
%<*generic>
\ProvidesExplPackage {tagpdf-mc-code-generic} {2025-01-12} {0.99l}
{part of tagpdf - code related to marking chunks - generic mode}
%
%<*debug>
\ProvidesExplPackage {tagpdf-debug-generic} {2025-01-12} {0.99l}
{part of tagpdf - debugging code related to marking chunks - generic mode}
%
% \end{macrocode}
%
%\subsection{Variables}
%
% \begin{macrocode}
%<*generic>
% \end{macrocode}
%
% \begin{variable}{\l_@@_mc_ref_abspage_tl}
% We need a ref-label system to ensure that the MCID cnt
% restarts at 0 on a new page
% This will be used to store the tagabspage attribute retrieved from
% a label.
% \begin{macrocode}
\tl_new:N \l_@@_mc_ref_abspage_tl
% \end{macrocode}
% \end{variable}
% \begin{variable}{\l_@@_mc_tmpa_tl}
% temporary variable
% \begin{macrocode}
\tl_new:N \l_@@_mc_tmpa_tl
% \end{macrocode}
% \end{variable}
% \begin{variable}
% {
% \g_@@_mc_marks,
% }
% a marks register to keep track of the mc's at page breaks and a sequence
% to keep track of the data for the continuation
% extra-tmb. We probably will need to track mc-marks
% in more than one stream, so the seq contains the name of the stream.
%
% \begin{macrocode}
\newmarks \g_@@_mc_marks
% \end{macrocode}
% \end{variable}
%
% \begin{variable}
% {
% \g_@@_mc_main_marks_seq,
% \g_@@_mc_footnote_marks_seq,
% \g_@@_mc_multicol_marks_seq
% }
% Each stream has an associated global seq variable holding the
% bottom marks from the/a previous chunk in the stream.
% We provide three by default: main, footnote and multicol.
% TODO: perhaps an interface for more streams will be needed.
% \begin{macrocode}
\seq_new:N \g_@@_mc_main_marks_seq
\seq_new:N \g_@@_mc_footnote_marks_seq
\seq_new:N \g_@@_mc_multicol_marks_seq
% \end{macrocode}
% \end{variable}
% \begin{macro}{\tag_mc_new_stream:n}
% \begin{macrocode}
\cs_new_protected:Npn \tag_mc_new_stream:n #1
{
\seq_new:c { g_@@_mc_multicol_#1_seq }
}
% \end{macrocode}
% \end{macro}
% \begin{variable}{\l_@@_mc_firstmarks_seq,\l_@@_mc_botmarks_seq}
% The marks content contains a number of data which we will have to access and
% compare, so we will store it locally in two sequences.
% topmarks is unusable in LaTeX so we ignore it.
%
% \begin{macrocode}
\seq_new:N \l_@@_mc_firstmarks_seq
\seq_new:N \l_@@_mc_botmarks_seq
% \end{macrocode}
% \end{variable}
%
% \subsection{Functions}
%
% \begin{macro}{\@@_mc_begin_marks:nn,\@@_mc_artifact_begin_marks:n,\@@_mc_end_marks:}
% Generic mode need to set marks for the page break and split stream handling.
% We always set two marks to be able to detect the case when no mark is on a
% page/galley. MC-begin commands will set (b,-,data) and (b,+,data),
% MC-end commands will set (e,-,data) and (e,+,data).
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_begin_marks:nn #1 #2 %#1 tag, #2 label
{
\tex_marks:D \g_@@_mc_marks
{
b-, %first of begin pair
\int_use:N\c@g_@@_MCID_abs_int, %mc-num
\g_@@_struct_stack_current_tl, %structure num
#1, %tag
\bool_if:NT \l_@@_mc_key_stash_bool{stash}, % stash info
#2, %label
}
\tex_marks:D \g_@@_mc_marks
{
b+, % second of begin pair
\int_use:N\c@g_@@_MCID_abs_int, %mc-num
\g_@@_struct_stack_current_tl, %structure num
#1, %tag
\bool_if:NT \l_@@_mc_key_stash_bool{stash}, % stash info
#2, %label
}
}
\cs_generate_variant:Nn \@@_mc_begin_marks:nn {oo}
\cs_new_protected:Npn \@@_mc_artifact_begin_marks:n #1 %#1 type
{
\tex_marks:D \g_@@_mc_marks
{
b-, %first of begin pair
\int_use:N\c@g_@@_MCID_abs_int, %mc-num
-1, %structure num
#1 %type
}
\tex_marks:D \g_@@_mc_marks
{
b+, %first of begin pair
\int_use:N\c@g_@@_MCID_abs_int, %mc-num
-1, %structure num
#1 %Type
}
}
\cs_new_protected:Npn \@@_mc_end_marks:
{
\tex_marks:D \g_@@_mc_marks
{
e-, %first of end pair
\int_use:N\c@g_@@_MCID_abs_int, %mc-num
\g_@@_struct_stack_current_tl, %structure num
}
\tex_marks:D \g_@@_mc_marks
{
e+, %second of end pair
\int_use:N\c@g_@@_MCID_abs_int, %mc-num
\g_@@_struct_stack_current_tl, %structure num
}
}
% \end{macrocode}
% \end{macro}
% \begin{macro}{\@@_mc_disable_marks:}
% This disables the marks. They can't be reenabled, so it should only
% be used in groups.
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_disable_marks:
{
\cs_set_eq:NN \@@_mc_begin_marks:nn \use_none:nn
\cs_set_eq:NN \@@_mc_artifact_begin_marks:n \use_none:n
\cs_set_eq:NN \@@_mc_end_marks: \prg_do_nothing:
}
% \end{macrocode}
% \end{macro}
% \begin{macro}{\@@_mc_get_marks:}
% This stores the current content of the marks in the sequences. It naturally
% should only be used in places where it makes sense.
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_get_marks:
{
\exp_args:NNe
\seq_set_from_clist:Nn \l_@@_mc_firstmarks_seq
{ \tex_firstmarks:D \g_@@_mc_marks }
\exp_args:NNe
\seq_set_from_clist:Nn \l_@@_mc_botmarks_seq
{ \tex_botmarks:D \g_@@_mc_marks }
}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\@@_mc_store:nnn}
% This inserts the mc-chunk \meta{mc-num} into the structure {struct-num}
% after the \meta{mc-prev}.
% The structure must already exist.
% The additional mcid dictionary is stored in a property. The item is retrieved
% when the kid entry is built. We test if there is already
% an addition and append if needed.
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_store:nnn #1 #2 #3 %#1 mc-prev, #2 mc-num #3 structure-num
{
%\prop_show:N \g_@@_struct_cont_mc_prop
\prop_get:NnNTF \g_@@_struct_cont_mc_prop {#1} \l_@@_tmpa_tl
{
\prop_gput:Nne \g_@@_struct_cont_mc_prop {#1}{ \l_@@_tmpa_tl \@@_struct_mcid_dict:n {#2}}
}
{
\prop_gput:Nne \g_@@_struct_cont_mc_prop {#1}{ \@@_struct_mcid_dict:n {#2}}
}
\prop_gput:Nee \g_@@_mc_parenttree_prop
{#2}
{#3}
}
\cs_generate_variant:Nn \@@_mc_store:nnn {eee}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{\@@_mc_insert_extra_tmb:n,\@@_mc_insert_extra_tme:n}
% These two functions should be used in the output routine at the place
% where a mc-literal could be missing due to a page break or some other split.
% They check (with the help of the marks) if a extra-tmb or extra-tme is needed.
% The tmb command stores also the mc into the structure, the tme has to store
% the data for a following extra-tmb.
% The argument takes a stream name like main or footnote to allow different handling
% there.
% The content of the marks must be stored before (with |\@@_mc_get_marks:|
% or manually)
% into |\l_@@_mc_firstmarks_seq| and |\l_@@_mc_botmarks_seq| so that the tests
% can use them.
%
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_insert_extra_tmb:n #1 % #1 stream: e.g. main or footnote
{
\@@_check_typeout_v:n {=>~ first~ \seq_use:Nn \l_@@_mc_firstmarks_seq {,~}}
\@@_check_typeout_v:n {=>~ bot~ \seq_use:Nn \l_@@_mc_botmarks_seq {,~}}
\@@_check_if_mc_tmb_missing:TF
{
\@@_check_typeout_v:n {=>~ TMB~ ~ missing~ --~ inserted}
%test if artifact
\int_compare:nNnTF { \seq_item:cn { g_@@_mc_#1_marks_seq } {3} } = {-1}
{
\tl_set:Ne \l_@@_tmpa_tl { \seq_item:cn { g_@@_mc_#1_marks_seq } {4} }
\@@_mc_handle_artifact:N \l_@@_tmpa_tl
}
{
\exp_args:Ne
\@@_mc_bdc_mcid:n
{
\seq_item:cn { g_@@_mc_#1_marks_seq } {4}
}
\str_if_eq:eeTF
{
\seq_item:cn { g_@@_mc_#1_marks_seq } {5}
}
{}
{
%store
\@@_mc_store:eee
{
\seq_item:cn { g_@@_mc_#1_marks_seq } {2}
}
{ \int_eval:n{\c@g_@@_MCID_abs_int} }
{
\seq_item:cn { g_@@_mc_#1_marks_seq } {3}
}
}
{
%stashed -> warning!!
}
}
}
{
\@@_check_typeout_v:n {=>~ TMB~ not~ missing}
}
}
\cs_new_protected:Npn \@@_mc_insert_extra_tme:n #1 % #1 stream, eg. main or footnote
{
\@@_check_if_mc_tme_missing:TF
{
\@@_check_typeout_v:n {=>~ TME~ ~ missing~ --~ inserted}
\@@_mc_emc:
\seq_gset_eq:cN
{ g_@@_mc_#1_marks_seq }
\l_@@_mc_botmarks_seq
}
{
\@@_check_typeout_v:n {=>~ TME~ not~ missing}
}
}
% \end{macrocode}
% \end{macro}
%
% \subsection{Looking at MC marks in boxes}
% \begin{macro}{\@@_add_missing_mcs:Nn}
% Assumptions:
% \begin{itemize}
% \item
% test for tagging active outside;
% \item
% mark retrieval also outside.
% \end{itemize}
%
% This takes a box register as its first argument (or the register
% number in a count register, as used by \pkg{multicol}). It adds
% an extra tmb at the top of the box if necessary and similarly
% an extra tme at the end. This is done by adding hboxes in a way
% that the positioning and the baseline of the given box is not
% altered. The result is written back to the box.
%
% The second argument is the stream this box belongs to und is
% currently either \texttt{main} for the main galley,
% \texttt{footnote} for footnote note text, or \texttt{multicol}
% for boxes produced for columns in that environment. Other streams
% may follow over time.
% \begin{macrocode}
\cs_new_protected:Npn\@@_add_missing_mcs:Nn #1 #2 {
\vbadness \@M
\vfuzz \c_max_dim
\vbox_set_to_ht:Nnn #1 { \box_ht:N #1 } {
\hbox_set:Nn \l_@@_tmpa_box { \@@_mc_insert_extra_tmb:n {#2} }
\hbox_set:Nn \l_@@_tmpb_box { \@@_mc_insert_extra_tme:n {#2} }
\int_compare:nNnT {\l_@@_loglevel_int} > { 0 }
{
\seq_log:c { g_@@_mc_#2_marks_seq}
}
% \end{macrocode}
% The box placed on the top gets zero size and thus will not affect
% the box dimensions of the box we are modifying.
% \begin{macrocode}
\box_set_ht:Nn \l_@@_tmpa_box \c_zero_dim
\box_set_dp:Nn \l_@@_tmpa_box \c_zero_dim
% \end{macrocode}
% The box added at the bottom will get the depth of the original
% box. This way we can arrange that from the outside everything
% looks as before.
% \begin{macrocode}
\box_set_ht:Nn \l_@@_tmpb_box \c_zero_dim
\box_set_dp:Nn \l_@@_tmpb_box { \box_dp:N #1 }
% \end{macrocode}
% We need to set \cs{boxmaxdepth} in case the original box has an
% unusually large depth, otherwise that depth is not preserved when
% we string things together.
% \begin{macrocode}
\boxmaxdepth \@maxdepth
\box_use_drop:N \l_@@_tmpa_box
\vbox_unpack_drop:N #1
% \end{macrocode}
% Back up by the depth of the box as we add that later again.
% \begin{macrocode}
\tex_kern:D -\box_dp:N \l_@@_tmpb_box
% \end{macrocode}
% And we don't want any glue added when we add the box.
% \begin{macrocode}
\nointerlineskip
\box_use_drop:N \l_@@_tmpb_box
}
}
% \end{macrocode}
% \end{macro}
% \begin{macro}{\tag_mc_add_missing_to_stream:Nn,\@@_add_missing_mcs_to_stream:Nn}
% This is the main command to add mc to the stream. It is therefore
% guarded by the mc-boolean.
%
% If we aren't in the main stream then processing is a bit more
% complicated because to get at the marks in the box we need to
% artificially split it and then look at the split marks.
%
% First argument is the box to update and the second is the \enquote{stream}.
% In lua mode the command is a no-op.
% \begin{macrocode}
\cs_new_protected:Npn \@@_add_missing_mcs_to_stream:Nn #1#2
{
\@@_check_if_active_mc:T {
% \end{macrocode}
% First set up a temp box for trial splitting.
% \begin{macrocode}
\vbadness\maxdimen
\box_set_eq:NN \l_@@_tmpa_box #1
% \end{macrocode}
% Split the box to the largest size available. This should give us
% all content (but to be sure that there is no issue we could test
% out test box is empty now (not done).
% \begin{macrocode}
\vbox_set_split_to_ht:NNn \l_@@_tmpa_box \l_@@_tmpa_box \c_max_dim
% \end{macrocode}
% As a side effect of this split we should now have the first and
% bottom split marks set up. We use this to set up
% \cs{l_@@_mc_firstmarks_seq}
% \begin{macrocode}
\exp_args:NNe
\seq_set_from_clist:Nn \l_@@_mc_firstmarks_seq
{ \tex_splitfirstmarks:D \g_@@_mc_marks }
% \end{macrocode}
% Some debugging info:
% \begin{macrocode}
% \iow_term:n { First~ mark~ from~ this~ box: }
% \seq_log:N \l_@@_mc_firstmarks_seq
% \end{macrocode}
% If this mark was empty then clearly the bottom mark will too be
% empty. Thus in this case we make use of the saved bot mark from
% the previous chunk. Note that if this is the first chunk in the
% stream the global seq would contain a random value, but then we
% can't end in this branch because the basis assumption is that
% streams are properly marked up so the first chunk would always
% have a mark at the beginning!
% \begin{macrocode}
\seq_if_empty:NTF \l_@@_mc_firstmarks_seq
{
\@@_check_typeout_v:n
{
No~ marks~ so~ use~ saved~ bot~ mark:~
\seq_use:cn {g_@@_mc_#2_marks_seq} {,~} \iow_newline:
}
\seq_set_eq:Nc \l_@@_mc_firstmarks_seq {g_@@_mc_#2_marks_seq}
% \end{macrocode}
% We also update the bot mark to the same value so that we can
% later apply \cs{@@_add_missing_mcs:Nn} with the data
% structures in place (see assumptions made there).
% \begin{macrocode}
\seq_set_eq:NN \l_@@_mc_botmarks_seq \l_@@_mc_firstmarks_seq
}
% \end{macrocode}
% If there was a first mark then there is also a bot mark (and it
% can't be the same as our marks always come in pairs).
% So if that branch is chosen we update
% \cs{l_@@_mc_botmarks_seq} from the bot mark.
% \begin{macrocode}
{
\@@_check_typeout_v:n
{
Pick~ up~ new~ bot~ mark!
}
\exp_args:NNe
\seq_set_from_clist:Nn \l_@@_mc_botmarks_seq
{ \tex_splitbotmarks:D \g_@@_mc_marks }
}
% \end{macrocode}
% Finally we call \cs{@@_add_missing_mcs:Nn} to add any missing
% tmb/tme as needed,
% \begin{macrocode}
\@@_add_missing_mcs:Nn #1 {#2}
%%
\seq_gset_eq:cN {g_@@_mc_#2_marks_seq} \l_@@_mc_botmarks_seq
%%
}
}
\cs_set_eq:NN \tag_mc_add_missing_to_stream:Nn \@@_add_missing_mcs_to_stream:Nn
% \end{macrocode}
% \end{macro}
%
% \begin{macro}[pTF]{\@@_mc_if_in:,\tag_mc_if_in:}
% This is a test if a mc is open or not. It depends simply on a global boolean:
% mc-chunks are added linearly so nesting should not be relevant.
%
% One exception are header and footer (perhaps they are more, but for now
% it doesn't seem so, so there are no dedicated code to handle this situation):
% When they are built and added to the page we could be both inside or outside a mc-chunk.
% But header and footer should ignore this and not push/pop or warn about nested mc.
% It is therefore important there to set and reset the boolean manually.
% See the tagpddocu-patches.sty for an example.
% \begin{macrocode}
\prg_new_conditional:Nnn \@@_mc_if_in: {p,T,F,TF}
{
\bool_if:NTF \g_@@_in_mc_bool
{ \prg_return_true: }
{ \prg_return_false: }
}
\prg_new_eq_conditional:NNn \tag_mc_if_in: \@@_mc_if_in: {p,T,F,TF}
% \end{macrocode}
% \end{macro}
%
% \begin{macro}{ \@@_mc_bmc:n,\@@_mc_emc:,\@@_mc_bdc:nn}
% These are the low-level commands. There are now equal to the
% pdfmanagement commands generic mode, but we use an indirection
% in case luamode need something else.
% change 04.08.2018: the commands do not check the validity of the arguments or try
% to escape them, this should be done before using them.
% change 2023-08-18: we are delaying the writing to the shipout.
% \begin{macrocode}
% #1 tag, #2 properties
\cs_set_eq:NN \@@_mc_bmc:n \pdf_bmc:n
\cs_set_eq:NN \@@_mc_emc: \pdf_emc:
\cs_set_eq:NN \@@_mc_bdc:nn \pdf_bdc:nn
\cs_set_eq:NN \@@_mc_bdc_shipout:ee \pdf_bdc_shipout:ee
% \end{macrocode}
% \end{macro}
%
% \begin{macro}
% {
% \@@_mc_bdc_mcid:nn,\@@_mc_bdc_mcid:n,
% \@@_mc_handle_mcid:nn,\@@_mc_handle_mcid:VV
% }
%
% This create a BDC mark with an |/MCID| key. Most of the work here is to get
% the current number value for the MCID: they must be numbered by page
% starting with 0 and then successively.
% The first argument is the tag, e.g. |P| or |Span|, the second is used to pass
% more properties.
% Starting with texlive 2023 this is much simpler and faster as we can use
% delay the numbering to the shipout.
% We also define a wrapper around the low-level command as luamode will need
% something different.
% \begin{macrocode}
\bool_if:NTF\g_@@_delayed_shipout_bool
{
\hook_gput_code:nnn {shipout/before}{tagpdf}{ \flag_clear:n { @@/mcid } }
\cs_set_protected:Npn \@@_mc_bdc_mcid:nn #1 #2
{
\int_gincr:N \c@g_@@_MCID_abs_int
\@@_property_record:eV
{
mcid-\int_use:N \c@g_@@_MCID_abs_int
}
\c_@@_property_mc_clist
\@@_mc_bdc_shipout:ee
{#1}
{
/MCID~\flag_height:n { @@/mcid }
\flag_raise:n { @@/mcid }~ #2
}
}
}
% \end{macrocode}
% if the engine is too old, we have to revert to earlier method.
% \begin{macrocode}
{
\msg_new:nnn { tagpdf } { old-engine }
{
The~engine~or~the~PDF management~is~too~old~or\\
delayed~shipout~has~been~disabled.\\
Fast~numbering~of~MC-chunks~not~available.\\
More~compilations~will~be~needed~in~generic~mode.
}
\msg_warning:nn { tagpdf} { old-engine }
\@@_prop_new:N \g_@@_MCID_byabspage_prop
\int_new:N \g_@@_MCID_tmp_bypage_int
\cs_generate_variant:Nn \@@_mc_bdc:nn {ne}
% \end{macrocode}
% revert the attribute:
% \begin{macrocode}
\property_gset:nnnn {tagmcid } { now }
{0} { \int_use:N \g_@@_MCID_tmp_bypage_int }
\cs_new_protected:Npn \@@_mc_bdc_mcid:nn #1 #2
{
\int_gincr:N \c@g_@@_MCID_abs_int
\tl_set:Ne \l_@@_mc_ref_abspage_tl
{
\property_ref:enn %3 args
{
mcid-\int_use:N \c@g_@@_MCID_abs_int
}
{ tagabspage }
{-1}
}
\prop_get:NoNTF
\g_@@_MCID_byabspage_prop
{
\l_@@_mc_ref_abspage_tl
}
\l_@@_mc_tmpa_tl
{
%key already present, use value for MCID and add 1 for the next
\int_gset:Nn \g_@@_MCID_tmp_bypage_int { \l_@@_mc_tmpa_tl }
\@@_prop_gput:Nee
\g_@@_MCID_byabspage_prop
{ \l_@@_mc_ref_abspage_tl }
{ \int_eval:n {\l_@@_mc_tmpa_tl +1} }
}
{
%key not present, set MCID to 0 and insert 1
\int_gzero:N \g_@@_MCID_tmp_bypage_int
\@@_prop_gput:Nee
\g_@@_MCID_byabspage_prop
{ \l_@@_mc_ref_abspage_tl }
{1}
}
\@@_property_record:eV
{
mcid-\int_use:N \c@g_@@_MCID_abs_int
}
\c_@@_property_mc_clist
\@@_mc_bdc:ne
{#1}
{ /MCID~\int_eval:n { \g_@@_MCID_tmp_bypage_int }~ \exp_not:n { #2 } }
}
}
\cs_new_protected:Npn \@@_mc_bdc_mcid:n #1
{
\@@_mc_bdc_mcid:nn {#1} {}
}
\cs_new_protected:Npn \@@_mc_handle_mcid:nn #1 #2 %#1 tag, #2 properties
{
\@@_mc_bdc_mcid:nn {#1} {#2}
}
\cs_generate_variant:Nn \@@_mc_handle_mcid:nn {VV}
% \end{macrocode}
% \end{macro}
% \begin{macro}{\@@_mc_handle_stash:n,\@@_mc_handle_stash:e}
% This is the handler which puts a mc into the
% the current structure. The argument is the number of the mc.
% Beside storing the mc into the structure, it also has to record the
% structure for the parent tree.
% The name is a bit confusing, it does \emph{not} handle mc with the stash key
% \ldots.
% TODO: why does luamode use it for begin + use, but generic mode only for begin?
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_handle_stash:n #1 %1 mcidnum
{
\@@_check_mc_used:n {#1}
\@@_struct_kid_mc_gput_right:nn
{ \g_@@_struct_stack_current_tl }
{#1}
\prop_gput:Nee \g_@@_mc_parenttree_prop
{#1}
{ \g_@@_struct_stack_current_tl }
}
\cs_generate_variant:Nn \@@_mc_handle_stash:n { e }
% \end{macrocode}
% \end{macro}
% \begin{macro}
% {
% \@@_mc_bmc_artifact:,
% \@@_mc_bmc_artifact:n,
% \@@_mc_handle_artifact:N
% }
% Two commands to create artifacts, one without type, and one with.
% We define also a wrapper handler as luamode will need a different definition.
% TODO: perhaps later: more properties for artifacts
% \begin{macrocode}
\cs_new_protected:Npn \@@_mc_bmc_artifact:
{
\@@_mc_bmc:n {Artifact}
}
\cs_new_protected:Npn \@@_mc_bmc_artifact:n #1
{
\@@_mc_bdc:nn {Artifact}{/Type/#1}
}
\cs_new_protected:Npn \@@_mc_handle_artifact:N #1
% #1 is a var containing the artifact type
{
\int_gincr:N \c@g_@@_MCID_abs_int
\tl_if_empty:NTF #1
{ \@@_mc_bmc_artifact: }
{ \exp_args:NV\@@_mc_bmc_artifact:n #1 }
}
% \end{macrocode}
% \end{macro}
% \begin{macro}{ \@@_get_data_mc_tag: }
% This allows to retrieve the active mc-tag.
% It is use by the get command.
% \begin{macrocode}
\cs_new:Nn \@@_get_data_mc_tag: { \g_@@_mc_key_tag_tl }
%
% \end{macrocode}
% \end{macro}
%
%
% \begin{macro}{\tag_mc_begin:n,\tag_mc_end:}
% These are the core public commands to open and close an mc.
% They don't need to be in the same group or grouping level,
% but the code expect that they are issued linearly. The tag and
% the state is passed to the end command through a global var and
% a global boolean.
% \begin{macrocode}
%\cs_new_protected:Npn \tag_mc_begin:n #1 { \@@_whatsits: \int_gincr:N \c@g_@@_MCID_abs_int }
%\cs_new_protected:Nn \tag_mc_end:{ \@@_whatsits: }
%<*generic|debug>
%<*generic>
\cs_set_protected:Npn \tag_mc_begin:n #1 %#1 keyval
{
\@@_check_if_active_mc:T
{
%
%<*debug>
\cs_set_protected:Npn \tag_mc_begin:n #1 %#1 keyval
{
\@@_check_if_active_mc:TF
{
\@@_debug_mc_begin_insert:n { #1 }
%
\group_begin: %hm
\@@_check_mc_if_nested:
\bool_gset_true:N \g_@@_in_mc_bool
% \end{macrocode}
% set default MC tags to structure:
% \begin{macrocode}
\tl_set_eq:NN \l_@@_mc_key_tag_tl \g_@@_struct_tag_tl
\tl_gset_eq:NN\g_@@_mc_key_tag_tl \g_@@_struct_tag_tl
\keys_set:nn { @@ / mc } {#1}
\bool_if:NTF \l_@@_mc_artifact_bool
{ %handle artifact
\@@_mc_handle_artifact:N \l_@@_mc_artifact_type_tl
\exp_args:NV
\@@_mc_artifact_begin_marks:n \l_@@_mc_artifact_type_tl
}
{ %handle mcid type
\@@_check_mc_tag:N \l_@@_mc_key_tag_tl
\@@_mc_handle_mcid:VV
\l_@@_mc_key_tag_tl
\l_@@_mc_key_properties_tl
\@@_mc_begin_marks:oo{\l_@@_mc_key_tag_tl}{\l_@@_mc_key_label_tl}
\tl_if_empty:NF {\l_@@_mc_key_label_tl}
{
\exp_args:NV
\@@_mc_handle_mc_label:e \l_@@_mc_key_label_tl
}
\bool_if:NF \l_@@_mc_key_stash_bool
{
\exp_args:NV\@@_struct_get_parentrole:nNN
\g_@@_struct_stack_current_tl
\l_@@_get_parent_tmpa_tl
\l_@@_get_parent_tmpb_tl
\@@_check_parent_child:VVnnN
\l_@@_get_parent_tmpa_tl
\l_@@_get_parent_tmpb_tl
{MC}{}
\l_@@_parent_child_check_tl
\int_compare:nNnT {\l_@@_parent_child_check_tl}<{0}
{
\prop_get:cnN
{ g_@@_struct_ \g_@@_struct_stack_current_tl _prop}
{S}
\l_@@_tmpa_tl
\msg_warning:nneee
{ tag }
{role-parent-child}
{ \l_@@_get_parent_tmpa_tl/\l_@@_get_parent_tmpb_tl }
{ MC~(real content) }
{ not~allowed~
(struct~\g_@@_struct_stack_current_tl,~\l_@@_tmpa_tl)
}
}
\@@_mc_handle_stash:e { \int_use:N \c@g_@@_MCID_abs_int }
}
}
\group_end:
}
%<*debug>
{
\@@_debug_mc_begin_ignore:n { #1 }
}
%
}
%<*generic>
\cs_set_protected:Nn \tag_mc_end:
{
\@@_check_if_active_mc:T
{
%
%<*debug>
\cs_set_protected:Nn \tag_mc_end:
{
\@@_check_if_active_mc:TF
{
\@@_debug_mc_end_insert:
%
\@@_check_mc_if_open:
\bool_gset_false:N \g_@@_in_mc_bool
\tl_gset:Nn \g_@@_mc_key_tag_tl { }
\@@_mc_emc:
\@@_mc_end_marks:
}
%<*debug>
{
\@@_debug_mc_end_ignore:
}
%
}
%
% \end{macrocode}
% \end{macro}
%
% \subsection{Keys}
% Definitions are different in luamode.
% |tag| and |raw| are expanded as |\lua_now:e| in lua does it too and
% we assume that their values are safe.
% \begin{macro}
% {
% tag (mc-key),
% raw (mc-key),
% alt (mc-key),
% actualtext (mc-key),
% label (mc-key),
% artifact (mc-key)
% }
% \begin{macrocode}
%<*generic>
\keys_define:nn { @@ / mc }
{
tag .code:n = % the name (H,P,Span) etc
{
\tl_set:Ne \l_@@_mc_key_tag_tl { #1 }
\tl_gset:Ne \g_@@_mc_key_tag_tl { #1 }
},
raw .code:n =
{
\tl_put_right:Ne \l_@@_mc_key_properties_tl { #1 }
},
alt .code:n = % Alt property
{
\str_set_convert:Noon
\l_@@_tmpa_str
{ #1 }
{ default }
{ utf16/hex }
\tl_put_right:Nn \l_@@_mc_key_properties_tl { /Alt~< }
\tl_put_right:No \l_@@_mc_key_properties_tl { \l_@@_tmpa_str>~ }
},
alttext .meta:n = {alt=#1},
actualtext .code:n = % ActualText property
{
\tl_if_empty:oF{#1}
{
\str_set_convert:Noon
\l_@@_tmpa_str
{ #1 }
{ default }
{ utf16/hex }
\tl_put_right:Nn \l_@@_mc_key_properties_tl { /ActualText~< }
\tl_put_right:No \l_@@_mc_key_properties_tl { \l_@@_tmpa_str>~ }
}
},
label .tl_set:N = \l_@@_mc_key_label_tl,
artifact .code:n =
{
\exp_args:Nne
\keys_set:nn
{ @@ / mc }
{ __artifact-bool, __artifact-type=#1 }
},
artifact .default:n = {notype}
}
%
% \end{macrocode}
% \end{macro}
% \end{implementation}
% \PrintIndex