
答案1
與expl3
- 我們檢查;
_
的參數內部是否存在\des
- 如果裡面沒有
_
,我們只是附加到;_{\textnormal{des}}
的參數上\des
和 - 如果確實至少有一個
_
,我們用一個巨集 (\riccardo_dessb:n
) 來取代它,而巨集 ( ) 在展開時又會接受一個參數(原始下標)並轉換為_{<original subscript>,\textnormal{des}}
(基本上_{<whatever>}
轉換為_{<whatever>,\textnormal{des}}
)。
這是正確的程式碼
\documentclass{scrartcl}
\usepackage{xparse,mathtools}
\NewDocumentCommand\des{m}{\csname riccardo_des:nn\endcsname{#1}{_}}
\ExplSyntaxOn
\cs_new_protected:Npn \riccardo_des:nn #1 #2
{
\tl_set:Nn \l_tmpa_tl { #1 }
\tl_if_in:NnTF \l_tmpa_tl { #2 }
{ \tl_replace_all:Nnn \l_tmpa_tl { #2 } { \riccardo_dessb:n } }
{ \tl_put_right:Nn \l_tmpa_tl { \sb { \textnormal{des} } } }
\tl_use:N \l_tmpa_tl
}
\cs_new_protected:Npn \riccardo_dessb:n #1
{ \sb { #1 , \textnormal{des} } }
\ExplSyntaxOff
\begin{document}
$\des{a}$ and $\des{a_{b}^{c}}$
\end{document}
添加
\des{a}_{b}^{c}
如果您喜歡像而不是這樣的清晰語法\des{a_{b}^{c}}
,這裡有一個使用的解決方案一些程式碼我幾週前發過貼文。它可能看起來很麻煩,但實際的定義要短得多。
\documentclass{scrartcl}
\usepackage{xparse,mathtools}
% begin addition for new arguments
\ExplSyntaxOn
\cs_new_protected:Npn \__xparse_count_type_k:w #1
{
\__xparse_single_token_check:n { #1 }
\quark_if_recursion_tail_stop_do:Nn #1 { \__xparse_bad_arg_spec:wn }
\__xparse_count_mandatory:N
}
\cs_new_protected:Npn \__xparse_count_type_K:w #1 #2
{
\__xparse_single_token_check:n { #1 }
\quark_if_recursion_tail_stop_do:nn { #2 } { \__xparse_bad_arg_spec:wn }
\__xparse_count_mandatory:N
}
\cs_new_protected:Npn \__xparse_add_type_k:w #1
{ \exp_args:NNo \__xparse_add_type_K:w #1 { \c__xparse_no_value_tl } }
\cs_new_protected:Npn \__xparse_add_type_K:w #1 #2
{
\__xparse_flush_m_args:
\__xparse_add_grabber_optional:N K
\tl_put_right:Nn \l__xparse_signature_tl { #1 { #2 } }
\__xparse_prepare_signature:N
}
\cs_new_protected:Npn \__xparse_add_expandable_type_k:w #1
{
\exp_args:NNo \__xparse_add_expandable_type_K:w #1 { \c__xparse_no_value_tl }
}
\cs_new_protected_nopar:Npn \__xparse_add_expandable_type_K:w #1 #2
{
\__msg_kernel_error:nnx { xparse } { invalid-expandable-argument-type } { K }
\__xparse_add_expandable_type_m:w % May be create this?
}
\cs_new_protected:Npn \__xparse_grab_K:w #1 #2 #3 \l__xparse_args_tl
{
\__xparse_grab_K_aux:NnnNn #1 { #2 } { #3 } \cs_set_protected_nopar:Npn
{ _ignore_spaces }
}
\cs_new_protected:Npn \__xparse_grab_K_long:w #1 #2 #3 \l__xparse_args_tl
{
\__xparse_grab_K_aux:NnnNn #1 { #2 } { #3 } \cs_set_protected:Npn
{ _ignore_spaces }
}
\cs_new_protected:Npn \__xparse_grab_K_trailing:w #1 #2 #3 \l__xparse_args_tl
{
\__xparse_grab_K_aux:NnnNn #1 { #2 } { #3 } \cs_set_protected_nopar:Npn
{ _ignore_spaces }
}
\cs_new_protected:Npn \__xparse_grab_K_long_trailing:w #1 #2 #3 \l__xparse_args_tl
{
\__xparse_grab_K_aux:NnnNn #1 { #2 } { #3 } \cs_set_protected:Npn
{ _ignore_spaces }
}
\cs_new_protected:Npn \__xparse_grab_K_aux:NnnNn #1 #2 #3 #4 #5
{
\exp_after:wN #4 \l__xparse_fn_tl ##1
{
\__xparse_add_arg:n { ##1 }
#3 \l__xparse_args_tl
}
\use:c { peek_meaning_remove #5 :NTF } #1
{ \l__xparse_fn_tl }
{
\__xparse_add_arg:n { #2 }
#3 \l__xparse_args_tl
}
}
\prop_put:Nnn \c__xparse_shorthands_prop { a } { k \sb }
\prop_put:Nnn \c__xparse_shorthands_prop { b } { k \sp }
\prop_put:Nnn \c__xparse_shorthands_prop { A } { K \sb }
\prop_put:Nnn \c__xparse_shorthands_prop { B } { K \sp }
\ExplSyntaxOff
% end
\NewDocumentCommand\des{ma}
{#1\IfValueTF{#2}{_{#2,\textnormal{des}}}{_{\textnormal{des}}}}
\begin{document}
$\des{a}$ and $\des{a}_{b}^{c}$
\end{document}
答案2
為了應對所有的可能性,您可以使用正規表示式。我所做的唯一假設是後面沒有空間_
,但這也可以容納。
函數的名稱應該是不言自明的。有關正規表示式的語法,請參閱texdoc l3regex
。
\documentclass{article}
\usepackage{xparse,l3regex}
\ExplSyntaxOn
\NewDocumentCommand{\des}{m}
{
\riccardo_des:n { #1 }
}
\tl_new:N \l_riccardo_des_arg_tl
\cs_new_protected:Nn \riccardo_des:n
{
\regex_match:nnTF { \_ } { #1 }
{
\riccardo_des_sub:n { #1 }
}
{
\riccardo_des_nosub:n { #1 }
}
}
% syntactic sugar
\cs_new_protected:Nn \riccardo_des_replace:nnn
{
\tl_set:Nn \l_riccardo_des_arg_tl { #3 }
\regex_replace_once:nnN { #1 } { #2 } \l_riccardo_des_arg_tl
\tl_use:N \l_riccardo_des_arg_tl
}
\cs_new_protected:Nn \riccardo_des_nosub:n
{
\riccardo_des_replace:nnn { ([^\^]*) } { \1\cD\_\cB\{\c{destext}\cE\} } { #1 }
}
\cs_new_protected:Nn \riccardo_des_sub:n
{
\regex_match:nnTF { \_\{ } { #1 }
{
\riccardo_des_sub_braced:n { #1 }
}
{
\riccardo_des_sub_unbraced:n { #1 }
}
}
\cs_new_protected:Nn \riccardo_des_sub_unbraced:n
{
\riccardo_des_replace:nnn { \_(.) } { \cD\_\cB\{\1,\c{destext}\cE\} } { #1 }
}
\cs_new_protected:Nn \riccardo_des_sub_braced:n
{
\riccardo_des_replace:nnn { \_\{(.*?)\} } { \cD\_\cB\{\1,\c{destext}\cE\} } { #1 }
}
\ExplSyntaxOff
\newcommand{\destext}{\mathrm{des}}
\begin{document}
$\des{a}+\des{a^{}}+\des{a^2}$
$\des{a_1}+\des{a_1^2}+\des{a^2_1}$
$\des{a_{1}}+\des{a_{1}^2}+\des{a^2_{1}}$
$\des{a_{1}}+\des{a_{1}^{2}}+\des{a^{2}_{1}}$
\end{document}