
Я использую пакет glossaries-extra для своего списка символов. По какой-то причине, не относящейся к делу, я определил новую команду
\newcommand{\symb}[3][]{%
\glsxtrnewsymbol[#1]{#2}{#3}%
}
таким образом, что записи моего глоссария будут иметь следующий вид:
\symb[description={Set of Smooth Vector Fields}]{vfs}{\ensuremath{\mathfrak{X}}}
Я хотел бы «перенести» их в шаблон диссертации моего университета, где они не используют пакет глоссариев, а вместо этого определяют среду, в которой символы должны вводиться следующим образом:
\begin{symbols}
\sym{\ensuremath{\mathfrak{X}}}{Set of Smooth Vector Fields}
\end{symbols}
Это просто среда табуляции, она напрямую печатает то, что вы записываете в аргументах команды \sym. Поэтому я планировал определить что-то вроде
\renewcommand{\symb}[3][]{%
\glsxtrnewsymbol[#1]{#2}{#3}%
\sym{#3}{\glsdesc{#2}}%
}
Это работает, за исключением того, что использование пакета glossaries каким-то образом портит другую часть файла стиля университета. Из-за этого я хотел бы определить
\renewcommand{\symb}[3][]{%
\sym{#3}{#1}%
}
Однако я не хочу иметь "description={Set of Smooth Vector Fields}" для второго аргумента, я хочу иметь "Set of Smooth Vector Fields". Есть ли способ извлечь только часть "Set of Smooth Vector Fields" из второго аргумента \symb?
Решение: Мне подходит упрощенная версия решения кота @Schrödinger:
\newcommand{\symb}[3][]{%
\def\mysplit##1=##2{\sym{#3}{##2}}%
\expandafter\mysplit#1%
}
решение1
С помощью следующего кода переопределенная процедура \symb
действует следующим образом:
Аргументы передаются \glsxtrnewsymbol
нетронутыми/как есть.
Аргументы передаются \sym
после извлечения описания из \symb
необязательного аргумента следующим образом:
В случае, если аргумент не содержит ведущую фразу «description=», он передается как есть. (Пробелы, окружающие всю фразу, и/или пробел между «description» и «=» учитываются.)
В случае, если этот аргумент содержит ведущую фразу «description=», эта фраза удаляется.
- В случае, если остаток состоит только из пробелов или ничего, результат извлечения будет пустым/вообще не будет состоять из токенов.
- В случае, если остаток состоит из одного неразделенного аргумента, т. е. либо из одного токена, не являющегося фигурными скобками, либо из набора токенов, вложенных в фигурные скобки, один уровень фигурных скобок, окружающий весь остаток, будет удален (если таковой имеется), затем будут удалены пробелы, окружающие весь остаток.
- В случае, если остаток состоит из чего-либо иного, чем просто один неразделенный аргумент, скобки не удаляются, но пробелы вокруг всего остатка удаляются.
Поскольку \sym
и \glsxtrnewsymbol
мне недоступны, я предоставил «фиктивные определения», которые не делают ничего, кроме как передают свои аргументы в детокенизированном виде, вложенные в угловые скобки.
Я все это сделал экспромтом, так что никаких гарантий. ;-)
Обратите внимание также на мой комментарий о вашей непереопределенной \symb
-процедуре и о вложении произвольного материала в фигурные скобки {
и }
при передаче его другому макросу, где он образует необязательный аргумент.
Весь «механизм» реализован без использования каких-либо \if..
... \else
... \fi
. Поэтому механизм не будет сбит с толку макроаргументами, содержащими непарные \if
или \else
или \fi
.
\documentclass{article}
\makeatletter
%%=============================================================================
%% Check whether argument is empty:
%%=============================================================================
%% \UD@CheckWhetherNull{<Argument which is to be checked>}%
%% {<Tokens to be delivered in case that argument
%% which is to be checked is empty>}%
%% {<Tokens to be delivered in case that argument
%% which is to be checked is not empty>}%
%%
%% Due to \romannumeral0-expansion the result is delivered after two
%% expansion-steps/after two "hits" by \expandafter.
%%
%% The gist of this macro comes from Robert R. Schneck's \ifempty-macro:
%% <https://groups.google.com/forum/#!original/comp.text.tex/kuOEIQIrElc/lUg37FmhA74J>
%%
\newcommand\UD@CheckWhetherNull[1]{%
\romannumeral0\expandafter\@secondoftwo\string{\expandafter
\@secondoftwo\expandafter{\expandafter{\string#1}\expandafter
\@secondoftwo\string}\expandafter\@firstoftwo\expandafter{\expandafter
\@secondoftwo\string}\@firstoftwo\expandafter{} \@secondoftwo}%
{\@firstoftwo\expandafter{} \@firstoftwo}%
}%
%%=============================================================================
%% Check whether argument is blank (empty or only spaces):
%%=============================================================================
%% -- Take advantage of the fact that TeX discards space tokens when
%% "fetching" _un_delimited arguments: --
%% \UD@CheckWhetherBlank{<Argument which is to be checked>}%
%% {<Tokens to be delivered in case that
%% argument which is to be checked is blank>}%
%% {<Tokens to be delivered in case that argument
%% which is to be checked is not blank}%
\newcommand\UD@CheckWhetherBlank[1]{%
\romannumeral\expandafter\expandafter\expandafter\@secondoftwo
\expandafter\UD@CheckWhetherNull\expandafter{\@firstoftwo#1{}.}%
}%
%%=============================================================================
%% Exchange two arguments. (From each argument an outermost level of
%% surrounding braces will be removed if present.)
%%=============================================================================
\newcommand\UD@Exchange[2]{#2#1}%
%%=============================================================================
%% Check whether argument's leading tokens form a specific
%% token-sequence that does not contain explicit character tokens of
%% category code 1 or 2:
%%=============================================================================
%% \UD@CheckWhetherLeadingTokens{<argument which is to be checked>}%
%% {<a <token sequence> without explicit
%% character tokens of category code
%% 1 or 2>}%
%% {a <single non-space token> that does
%% _not_ occur in <token sequence> >}%
%% {<internal token-check-macro>}%
%% {<tokens to be delivered in case
%% <argument which is to be checked> has
%% <token sequence> as leading tokens>}%
%% {<tokens to be delivered in case
%% <argument which is to be checked>
%% does not have <token sequence> as
%% leading tokens>}%
\newcommand\UD@CheckWhetherLeadingTokens[4]{%
\romannumeral0\UD@CheckWhetherNull{#1}%
{\UD@Exchange{ }\expandafter\@secondoftwo}%
{\expandafter\@secondoftwo\string{\expandafter
\UD@@CheckWhetherLeadingTokens#4#3#1#2}{}}%
}%
\newcommand\UD@@CheckWhetherLeadingTokens[1]{%
\expandafter\UD@CheckWhetherNull\expandafter{\@firstoftwo{}#1}%
{\UD@Exchange{\@firstoftwo}}{\UD@Exchange{\@secondoftwo}}%
{\UD@Exchange{ }{\expandafter\expandafter\expandafter\expandafter
\expandafter\expandafter\expandafter}\expandafter\expandafter
\expandafter}\expandafter\@secondoftwo\expandafter{\string}%
}%
%%=============================================================================
%% \UD@internaltokencheckdefiner{<internal token-check-macro>}%
%% {<token sequence>}%
%% Defines <internal token-check-macro> to snap everything
%% until reaching <token sequence>-sequence and spit that out
%% nested in braces.
%%=============================================================================
\newcommand\UD@internaltokencheckdefiner[2]{%
\@ifdefinable#1{\long\def#1##1#2{{##1}}}%
}%
%%=============================================================================
\UD@internaltokencheckdefiner{\UD@ExtractDescriptionEqual}{description=}%
\UD@internaltokencheckdefiner{\UD@ExtractDescriptionSpaceEqual}{description =}%
\UD@internaltokencheckdefiner{\UD@ExtractDescriptionEqualSpace}{description= }%
\UD@internaltokencheckdefiner{\UD@ExtractDescriptionSpaceEqualSpace}{description = }%
\UD@internaltokencheckdefiner{\UD@ExtractSpaceDescriptionEqual}{ description=}%
\UD@internaltokencheckdefiner{\UD@ExtractSpaceDescriptionSpaceEqual}{ description =}%
\UD@internaltokencheckdefiner{\UD@ExtractSpaceDescriptionEqualSpace}{ description= }%
\UD@internaltokencheckdefiner{\UD@ExtractSpaceDescriptionSpaceEqualSpace}{ description = }%
\UD@internaltokencheckdefiner{\UD@ExtractSpace}{ }%
%%=============================================================================
%% Trim all leading and trailing spaces:
%%=============================================================================
\newcommand\UD@RemoveSpaces[1]{%
\romannumeral0\@firstofone{\UD@TrimTrailSpaceLoop{#1}.#1\UD@Bizarre} \UD@Bizarre\relax\UD@Bizarre
}%
\@ifdefinable\UD@TrimTrailSpaceLoop{%
\long\def\UD@TrimTrailSpaceLoop#1#2 \UD@Bizarre#3\relax\UD@Bizarre{%
\UD@CheckWhetherNull{#3}{%
\UD@TrimLeadSpaceLoop{#1}%
}{%
\@firstofone{\expandafter\UD@TrimTrailSpaceLoop\expandafter{\@gobble#2}#2\UD@Bizarre} \UD@Bizarre\relax\UD@Bizarre
}%
}%
}%
\@ifdefinable\UD@gobblespace{%
\@firstofone{\def\UD@gobblespace} {}%
}%
\newcommand\UD@TrimLeadSpaceLoop[1]{%
\UD@CheckWhetherLeadingTokens{#1}{ }{.}{\UD@ExtractSpace}{\expandafter\UD@TrimLeadSpaceLoop\expandafter{\UD@gobblespace#1}}{ #1}%
}%
%%=============================================================================
%% Extract description:
%%=============================================================================
\newcommand\UD@ExtractDescription[1]{%
\romannumeral0%
\UD@CheckWhetherLeadingTokens{#1}{ description = }{.}{\UD@ExtractSpaceDescriptionSpaceEqualSpace}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractSpaceDescriptionSpaceEqualSpace#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{ description= }{.}{\UD@ExtractSpaceDescriptionEqualSpace}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractSpaceDescriptionEqualSpace#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{ description =}{.}{\UD@ExtractSpaceDescriptionSpaceEqual}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractSpaceDescriptionSpaceEqual#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{ description=}{.}{\UD@ExtractSpaceDescriptionEqual}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractSpaceDescriptionEqual#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{description = }{.}{\UD@ExtractDescriptionSpaceEqualSpace}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractDescriptionSpaceEqualSpace#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{description= }{.}{\UD@ExtractDescriptionEqualSpace}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractDescriptionEqualSpace#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{description =}{.}{\UD@ExtractDescriptionSpaceEqual}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractDescriptionSpaceEqual#1}%
}{%
\UD@CheckWhetherLeadingTokens{#1}{description=}{.}{\UD@ExtractDescriptionEqual}{%
\expandafter\UD@ExtractDescriptionCheckArgAmount\expandafter{\UD@ExtractDescriptionEqual#1}%
}{ #1}%
}%
}%
}%
}%
}%
}%
}%
}%
\newcommand\UD@ExtractDescriptionCheckArgAmount[1]{%
\expandafter\UD@CheckWhetherBlank\expandafter{\@gobble#1}{ }{%
\expandafter\UD@CheckWhetherBlank\expandafter{\@gobbletwo#1}{%
\UD@Exchange{ }{\expandafter\expandafter\expandafter\expandafter\expandafter\expandafter\expandafter}%
\expandafter\UD@RemoveSpaces\expandafter{\@secondoftwo#1}%
}{%
\UD@Exchange{ }{\expandafter\expandafter\expandafter\expandafter\expandafter\expandafter\expandafter}%
\expandafter\UD@RemoveSpaces\expandafter{\@firstoftwo{}#1}%
}%
}%
}%
%%=============================================================================
%% Your definition of \symb with a modification:
%% When passing arbitrary arguments as optional arguments, I strongly recommend
%% nesting them in braces for ensuring that nesting of square brackets won't
%% lead to problems:
%%=============================================================================
\newcommand{\symb}[3][]{%
\glsxtrnewsymbol[{#1}]{#2}{#3}% <- Here the content of #1 is arbitrary.
}% % Problems due to nested square brackets
% might occur in case #1 contains
% [ or ].
% Some day someone might do something like
% \symb[{description=optional\macro[macro's optional]}]{...}{...}.
% Therefore nest #1 in braces.
% These braces will be removed by LaTeX
% when processing \glsxtrnewsymbol's
% optional argument.
% They prevent confusion when it comes
% to nesting optional arguments within
% optional arguments.
%%=============================================================================
%% Redefinition of \symb:
%%=============================================================================
\renewcommand{\symb}[3][]{%
\romannumeral0%
\expandafter\expandafter\expandafter\UD@Exchange
\expandafter\expandafter\expandafter{%
\expandafter\expandafter\expandafter{\UD@ExtractDescription{#1}}}{ \sym{#3}}%
\glsxtrnewsymbol[{#1}]{#2}{#3}%
}%
%%=============================================================================
%% Dummy-definition for \glsxtrnewsymbol which displays the arguments verbatim
%%=============================================================================
\newcommand\glsxtrnewsymbol[3][]{%
\par\noindent
\texttt{\string\glsxtrnewsymbol}'s optional argument: $\langle$\texttt{\detokenize{#1}}$\rangle$
\par\noindent
\texttt{\string\glsxtrnewsymbol}'s mandatory argument 1: $\langle$\texttt{\detokenize{#2}}$\rangle$
\par\noindent
\texttt{\string\glsxtrnewsymbol}'s mandatory argument 2: $\langle$\texttt{\detokenize{#3}}$\rangle$
\par
}%
%%=============================================================================
%% Dummy-definition for \sym which displaysthe arguments verbatim
%%=============================================================================
\newcommand\sym[2]{%
\par\noindent
\texttt{\string\sym}'s mandatory argument 1: $\langle$\texttt{\detokenize{#1}}$\rangle$
\par\noindent
\texttt{\string\sym}'s mandatory argument 2: $\langle$\texttt{\detokenize{#2}}$\rangle$
\par
}%
\makeatother
\begin{document}
\vspace*{-1in}%
\noindent
\texttt{\detokenize{\symb[description={Set of Smooth Vector Fields}]{vfs}{\ensuremath{\mathfrak{X}}}}}:
\smallskip
\symb[description={Set of Smooth Vector Fields}]{vfs}{\ensuremath{\mathfrak{X}}}
\bigskip
\noindent\null\hrulefill\null
\bigskip
\noindent
\texttt{\detokenize{\symb[ description = { Set of Smooth Vector Fields } ]{vfs}{\ensuremath{\mathfrak{X}}}}}:
\smallskip
\symb[ description = { Set of Smooth Vector Fields } ]{vfs}{\ensuremath{\mathfrak{X}}}
\bigskip
\noindent\null\hrulefill\null
\bigskip
\noindent
\texttt{\detokenize{\symb[description = Set of Smooth Vector Fields ]{vfs}{\ensuremath{\mathfrak{X}}}}}:
\smallskip
\symb[description = Set of Smooth Vector Fields ]{vfs}{\ensuremath{\mathfrak{X}}}
\bigskip
\noindent\null\hrulefill\null
\bigskip
\noindent
\texttt{\detokenize{\symb{vfs}{\ensuremath{\mathfrak{X}}}}}:
\smallskip
\symb{vfs}{\ensuremath{\mathfrak{X}}}
\bigskip
\noindent\null\hrulefill\null
\bigskip
\noindent
\texttt{\detokenize{\symb[whatsoever]{vfs}{\ensuremath{\mathfrak{X}}}}}:
\smallskip
\symb[whatsoever]{vfs}{\ensuremath{\mathfrak{X}}}
\end{document}
решение2
Это то, что делает это. Очевидно, у меня нет вашей \sym
команды, поэтому я использую для этого что-то, что показывает, что аргументы были приняты правильно.
\documentclass{article}
\usepackage{amsfonts}
\def\forgetit{}
\newcommand{\symb}[3][]{%
\glsxtrnewsymbol[#1]{#2}{#3}%
}
\newcommand{\sym}[2]{first argument=\ensuremath{#1},second argument=#2}
\renewcommand{\symb}[3][\forgetit]{%
\def\mysplit##1=##2;{\def\mylast{##2}}%
\ifx#1\forgetit
\sym{#3}{empty}%
\else
\expandafter\mysplit#1;%
\sym{#3}{\mylast}%
\fi
}
\begin{document}
\symb[description={Set of Smooth Vector Fields}]{vfs}{\ensuremath{\mathfrak{X}}}
\symb{vfs}{\ensuremath{\mathfrak{X}}}
\end{document}
Это предполагает, что всякий раз, когда первый аргумент непустой, он содержит знак =
. Вы можете сделать его невосприимчивым к этому, но тогда я бы рекомендовал использовать какую-то систему управления ключами. Например, если вы уже загружаете, pgf
это будет очень просто.