I am trying to re-implement a package with a command \foo
which I am guessing has 2 APIs (I am not very sure, maybe there are other ways of achieving it). The user-side API of the command is like this:
Case 1
xxxxxxxxxx
\foo
\baz abc def::% Marks the end of the argument, coded with u{::} in xparse.
\foobaz ghi jkl::
\bazfoo mno pqr::
\oof
This is the default behavior. The same behavior is repeated with the following, but with a value abc
to a key named foo
. We can interpret that the initial value of the key foo is abc
.
Case 2
xxxxxxxxxx
\foo[foo=abc]
\baz abc def::
\foobaz ghi jkl::
\bazfoo mno pqr::
\oof
Case 3
xxxxxxxxxx
\foo[foo=def]
abc[ghi/mno] def[ghi/jkl/pqr]
\oof
I have coded the simple-most case as follows:
For developing keys, I will use my favorite pair of packages expkv-opt,expkv-def
. The following should give us case 2.
So I think in order to equate the output of case 3 with the rest of the two; I will have to have an alternate argument structure for \foo
. Is there any other way to get it?
The input syntax is very weird. As I wrote in the comments, please first think if it really isn’t possible to support an input syntax that is more common in the LaTeX world.
I provide the following code to make it clear how the special parsing can be triggered when we detect the special case foo=def
in the options of \foo
. Consider it as code golf!
Note: I used l3keys
because I am familiar with it and not with expkv
; I have nothing particular against the latter. 😃
Regarding this part:
xxxxxxxxxx
\NewExpandableDocumentCommand \useFooKey { }
{
\tl_use:N \l_mypkg_fookey_tl
}
I used \NewExpandableDocumentCommand
in case you want to use \useFooKey
in expansion-only contexts (inside \edef
, \write
, \csname
, \numexpr
… similarly, expandable material works fine inside the argument of siunitx
macros like \num
, and there are many more places like that). But this is not a choice I can make because I don’t know what kind of tokens \l_mypkg_fookey_tl
will legitimately contain. All I know is that \tl_use:N
is expandable, thus works in expansion-only contexts (and can even be omitted because tl
vars are simple non-\protected
macros).
I can’t know if the material resulting from the expansion of \l_mypkg_fookey_tl
is designed to be used in expansion-only contexts: only you, the API designer, can know.
For instance, if \l_mypkg_fookey_tl
is supposed to contain non-expandable code fragments (e.g., containing \def
, \setcounter
or \addtocounter
calls, markup like \section{...}
…), probably the \NewExpandableDocumentCommand
should be replaced with \NewDocumentCommand
in order to prevent the expansion of \useFooKey
in certain kinds of expansion-only contexts (inside \edef
, \xdef
, \expanded
, \message
, \errmessage
, \special
, \mark
, \marks
; when writing the token list for \write
to a file; when looking ahead in an alignment for \noalign
or \omit
; plus everything that is based on one of these mechanisms of the TeX engine).
Also, I hesitated between “purity” and efficiency here: the \NewExpandableDocumentCommand
clearly marks \useFooKey
as belonging to the document-level layer, however we don’t need the (formerly from xparse
, now documented in usrguide3.pdf) heavy \NewExpandableDocumentCommand
machinery here, since \useFooKey
is a trivial macro that takes no argument. The simple fact of using \NewExpandableDocumentCommand
probably makes \useFooKey
a bit slow.
So, an alternative which I didn’t put in the code because I wanted to keep it short and readable, but is at least as good in my opinion, is:
xxxxxxxxxx
\cs_new:Npn \mypkg_use_foo_key:
{
\tl_use:N \l_mypkg_fookey_tl
}
\cs_new_eq:NN \useFooKey \mypkg_use_foo_key:
or, more generically:
xxxxxxxxxx
\cs_new:Npn \mypkg_use_key:n #1
{
\tl_use:c { l_mypkg_#1key_tl }
}
\cs_new_eq:NN \useKey \mypkg_use_key:n
(in which case you would input \useKey{foo}
instead of \useFooKey
).
Again, the \cs_new:Npn
used here was chosen under the assumption that \useFooKey
or \useKey{foo}
should expand in expansion-only contexts; if \useFooKey
or \useKey{foo}
should be expanded only when TeX performs its full expansion + execution interleaved process (typically, when typesetting), use \cs_new_protected:Npn
or \cs_new_protected:Nn
instead.