
我正在嘗試創建一個l3keys
名為 which 的列表mykeys
,用於控制命令的行為mycommand:n
。最初,這只是身份。設定command=\textit
應將其變更為\textit
(與身分組成)。一般來說,設定command=\foo
應該重新定義mycommand:n
為#1
↦ \foo{mycommand:n{#1}}
。然而,就目前的情況來看,它卻失敗得很慘。有可能使這項工作成功嗎?
\documentclass{article}
\usepackage{expl3}
\ExplSyntaxOn
\cs_new:Npn \identity:n #1 { #1 }
\keys_define:nn { mykeys }
{
command .multichoice:,
command / none .code:n = {
\cs_set_eq:NN \mycommand:n \identity:n %this resets \mycommand:n
%to become the identity
},
command / unknown .code:n = {
\cs_set:Npx \mycommand:n ##1
{
#1 { \mycommand:n { ##1 } }
}
}
}
\keys_set:nn { mykeys } { command=none }
\ExplSyntaxOff
\begin{document}
\ExplSyntaxOn
\mycommand:n{test1} % should produce test1
\keys_set:nn { mykeys } { command=\textit }
\mycommand:n{test2} % should produce \textit{test2}
\keys_set:nn { mykeys } { command=\textbf }
\mycommand:n{test3} % should produce \textbf{\textit{test3}}
\ExplSyntaxOff
\end{document}
答案1
我對介面不是特別感興趣,但在這裡我會按順序儲存應用程式的命令
\documentclass{article}
\usepackage{expl3,xparse}
\ExplSyntaxOn
\seq_new:N \l__mycommand_cmds_seq
\tl_new:N \l__mycommand_tmp_tl
\keys_define:nn { mykeys }
{
command .code:n =
{
\str_if_eq:nnTF {#1} { none }
{ \seq_clear:N \l__mycommand_cmds_seq }
{ \seq_put_right:Nn \l__mycommand_cmds_seq {#1} }
}
}
\keys_set:nn { mykeys } { command = none }
\cs_new_protected:Npn \mycommand_main:n #1
{
\group_begin:
\tl_set:Nn \l__mycommand_tmp_tl {#1}
\seq_map_inline:Nn \l__mycommand_cmds_seq
{
\tl_set:Nx \l__mycommand_tmp_tl
{ \exp_not:n {##1} { \exp_not:V \l__mycommand_tmp_tl } }
}
\exp_args:NV \group_end:
\l__mycommand_tmp_tl
}
\NewDocumentCommand \mysetup { +m } { \keys_set:nn { mykeys } {#1} }
\NewDocumentCommand \mycommand { m }
{ \mycommand_main:n {#1} }
\ExplSyntaxOff
\begin{document}
\mycommand{test1} % should produce test1
\mysetup{command = \textit}
\mycommand{test2} % should produce \textit{test2}
\mysetup{command = \textbf}
\mycommand{test3} % should produce \textbf{\textit{test3}}
\end{document}
人們可能會進行單令牌/巨集測試作為鍵值設定的一部分,但這可能有點矯枉過正。
人們可能會在 keyval 設定期間建立「功能」標記列表,但我懷疑讓序列可供檢查可能在調試時有用。
答案2
只要涉及的命令在一次擴展後公開其實現,您就可以執行以下操作
\documentclass{article}
\usepackage{expl3}
\ExplSyntaxOn
\cs_new:Npn \identity:n #1 { #1 }
\cs_new:Npn\wrap_def:NN#1#2{
\exp_after:wN\cs_set:Npn
\exp_after:wN#2\exp_after:wN##\exp_after:wN1\exp_after:wN{
\exp_after:wN#1\exp_after:wN{#2{##1}}}
}
\keys_define:nn { mykeys }
{
command .multichoice:,
command / none .code:n = {
\cs_set_eq:NN \mycommand:n \identity:n %this resets \mycommand:n
%to become the identity
},
command / unknown .code:n = {
\wrap_def:NN #1 \mycommand:n
}
}
\keys_set:nn { mykeys } { command=none }
\ExplSyntaxOff
\begin{document}
\ExplSyntaxOn
\mycommand:n{test1} % should produce test1
\keys_set:nn { mykeys } { command=\textit }
\mycommand:n{test2} % should produce \textit{test2}
\keys_set:nn { mykeys } { command=\textbf }
\mycommand:n{test3} % should produce \textbf{\textit{test3}}
\ExplSyntaxOff
\end{document}
答案3
實作大衛的建議:
\documentclass{article}
\usepackage{expl3}
\ExplSyntaxOn
\tl_new:N \l_gaussler_command_start_tl
\tl_new:N \l_gaussler_command_end_tl
\keys_define:nn { mykeys }
{
command .multichoice:,
command / none .code:n =
{
\tl_clear:N \l_gaussler_command_start_tl
\tl_clear:N \l_gaussler_command_end_tl
\cs_set_protected:Nn \gaussler_thecommand:n { ##1 }
},
command / unknown .code:n =
{
\tl_put_left:Nn \l_gaussler_command_start_tl { \exp_not:N #1 \gaussler_command_open: }
\tl_put_right:Nn \l_gaussler_command_end_tl { \gaussler_command_close: }
\cs_set_protected:Nx \gaussler_thecommand:n
{
\tl_use:N \l_gaussler_command_start_tl
##1
\tl_use:N \l_gaussler_command_end_tl
}
}
}
\cs_new:Nn \gaussler_command_open: { \if_true: { \else: } \fi: }
\cs_new:Nn \gaussler_command_close: { \if_false: { \else: } \fi: }
\keys_set:nn { mykeys } { command=none }
\ExplSyntaxOff
\begin{document}
\ExplSyntaxOn
\gaussler_thecommand:n{test1} % should produce test1
\keys_set:nn { mykeys } { command=\textit }
\gaussler_thecommand:n{test2} % should produce \textit{test2}
\keys_set:nn { mykeys } { command=\textbf }
\gaussler_thecommand:n{test3} % should produce \textbf{\textit{test3}}
\keys_set:nn { mykeys } { command=none }
\gaussler_thecommand:n{test4}
\ExplSyntaxOff
\end{document}