Вот мой mwe.tex
\documentclass[12pt]{report}
\usepackage{xifthen}
\usepackage{xstring}
\def\test{\textnormal{test}}
\newcounter{wordCount}
\setcounter{wordCount}{0}
\newcounter{AllWord}
\setcounter{AllWord}{0}
\NewDocumentCommand{\exy}{ m m m}{%
\StrCount{#2}{,}[\comma]
\StrCount{#2}{?}[\qmark]
\ifthenelse{\qmark > 0}{%
\addtocounter{wordCount}{\comma + \qmark}
\addtocounter{AllWord}{\comma + \qmark}
}{%
\addtocounter{wordCount}{\comma + \qmark + 1}
\addtocounter{AllWord}{\comma + \qmark + 1}
}
\textbf{#2} (#3), ``#1''\\%
}
\begin{document}
\test\par
\exy{one \test}{two \test s}{three \test s}
\end{document}
Если вы закомментируете строку с \exy
этим, то все работает как и ожидалось. В противном случае ошибки начинаются с
[{
"resource": "/c:/Users/hsmye/LaTeX/Gaelic/vocab/mwe.tex",
"owner": "LaTeX",
"severity": 8,
"message": "Undefined control sequence.\n\\reserved@a ->\\@nil",
"source": "LaTeX",
"startLineNumber": 27,
"startColumn": 1,
"endLineNumber": 27,
"endColumn": 65536
}]
что выходит за рамки моего макро-фу. Может кто-нибудь помочь? Следует отметить, что это не изначальная проблема. Чтобы создать MWE без необходимости покупки шрифта, я продолжал вычитать, пока не исчезла ошибка, а затем добавил обратно, чтобы получить это. Изначальная проблема была:
\newfontfamily{\EtGoudy}{P22 Goudy Ampersands}
\DeclareTextFontCommand{\GoudyEt}{\EtGoudy}
\newfontfamily{\cVirgi}{P22 Virginian}
\DeclareTextFontCommand{\Virgic}{\cVirgi}
\def\dispEt{\GoudyEt{c}\kern -3pt{\vspace{-0.01ex}\huge\Virgic{c}}.}
\def\Etc{\textnormal{\dispEt}}
заменить \test
на \Etc
с аналогичным результатом.
решение1
Обратите внимание, что ошибки TeX всегда следует отображать в многострочной форме, поскольку переносы строк имеют решающее значение для понимания сообщения, они отмечаюткоторыйкоманда не определена
! Undefined control sequence.
\reserved@a ->\@nil
l.29 ...exy{one \test}{two \test s}{three \test s}
?
Вы использовали примитивное определение TeX, что привело к хрупкой команде, которая ломается в контексте расширения. Если вы используете \NewDocumentCommand
you get \protected
command safe в таких конструкциях.
Это выполняется без ошибок.
\documentclass[12pt]{report}
\usepackage{xifthen}
\usepackage{xstring}
\NewDocumentCommand\test{}{\textnormal{test}}
\newcounter{wordCount}
\setcounter{wordCount}{0}
\newcounter{AllWord}
\setcounter{AllWord}{0}
\NewDocumentCommand{\exy}{ m m m}{%
\StrCount{#2}{,}[\comma]%%%%%%%%%%%%%%
\StrCount{#2}{?}[\qmark]%%%%%%%%%%%%%%
\ifthenelse{\qmark > 0}{%
\addtocounter{wordCount}{\comma + \qmark}%%%%%%%%%%%%%%
\addtocounter{AllWord}{\comma + \qmark}%%%%%%%%%%%%%%
}{%
\addtocounter{wordCount}{\comma + \qmark + 1}%%%%%%%%%%%%%%
\addtocounter{AllWord}{\comma + \qmark + 1}%%%%%%%%%%%%%%
}%%%%%%%%%%%%%%
\textbf{#2} (#3), ``#1''\\%
}
\begin{document}
\test\par
\exy{one \test}{two \test s}{three \test s}
\end{document}
Но производит
Underfull \hbox (badness 10000) in paragraph at lines 29--30
Из-за неправильного расположения \\
, которое никогда не следует использовать в конце абзаца. (Было бы лучше использовать \par
)
решение2
Я бы не использовал xstring
. Основная проблема в вашем коде в том, что команда \Etc
(очень) хрупкая.
Я предлагаю использовать expl3
для подсчета запятых и вопросительных знаков разделение текста на них и подсчет элементов (и вычитание одного).
Аргумент полностью развернут (но не надежные команды) и «очищен», поэтому такие команды \textnormal
исчезают.
\documentclass[12pt]{report}
\newcommand\test{\textnormal{test}}
\NewDocumentCommand\dispEt{}{\GoudyEt{c}\kern -3pt{\huge\Virgic{c}}.}
\NewDocumentCommand\Etc{}{\textnormal{\dispEt}}
\newcommand\GoudyEt{}% just for testing
\newcommand{\Virgic}{}% just for testing
\newcounter{wordCount}
\newcounter{AllWord}
\ExplSyntaxOn
\NewDocumentCommand{\exy}{m m m}
{
\hsmyers_xyz_exy:nnen { #1 } { #2 } { \text_purify:n { \text_expand:n { #2 } } } { #3 }
}
\int_new:N \l_hsmyers_xyz_comma_int
\int_new:N \l_hsmyers_xyz_qmark_int
\cs_new_protected:Nn \hsmyers_xyz_exy:nnnn
{
\hsmyers_xyz_count:nnN { , } { #3 } \l_hsmyers_xyz_comma_int
\hsmyers_xyz_count:nnN { ? } { #3 } \l_hsmyers_xyz_qmark_int
\int_compare:nTF { \l_hsmyers_xyz_qmark_int > 0 }
{
\addtocounter{wordCount}
{
\int_eval:n { \l_hsmyers_xyz_comma_int + \l_hsmyers_xyz_qmark_int }
}
\addtocounter{AllWord}
{
\int_eval:n { \l_hsmyers_xyz_comma_int + \l_hsmyers_xyz_qmark_int }
}
}{%
\addtocounter{wordCount}
{
\int_eval:n { \l_hsmyers_xyz_comma_int + \l_hsmyers_xyz_qmark_int + 1}
}
\addtocounter{AllWord}
{
\int_eval:n { \l_hsmyers_xyz_comma_int + \l_hsmyers_xyz_qmark_int + 1}
}
}
\textbf{#2}~#4,~``#1''
}
\cs_generate_variant:Nn \hsmyers_xyz_exy:nnnn { nne }
\cs_new_protected:Nn \hsmyers_xyz_count:nnN
{
\seq_set_split:Nnn \l_tmpa_seq { #1 } { #2 }
\int_set:Nn #3 { \seq_count:N \l_tmpa_seq - 1 }
}
\ExplSyntaxOff
\begin{document}
\test\par
\exy{one \Etc}{two \test \Etc{} s}{three \Etc{} s}
wordCount=\the\value{wordCount}
AllWord=\the\value{AllWord}
\exy{one \test}{two, three?}{three \test s}
wordCount=\the\value{wordCount}
AllWord=\the\value{AllWord}
\end{document}