Aqui está meu 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}
Se você comentar a linha, \exy
ela funcionará conforme o esperado. Caso contrário, os erros começam com
[{
"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
}]
o que está além do meu macro-fu. Alguém pode ajudar? Deve-se notar que este não é o problema original. Para produzir um MWE sem exigir a compra de fontes, continuei retirando até que não houvesse erro e depois adicionei novamente para obter isso. O problema original era:
\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}}
substitua \test
por \Etc
com resultados semelhantes.
Responder1
Observe que você deve sempre mostrar os erros do TeX em formato multilinha, pois as quebras de linha são vitais para entender a mensagem, elas marcamqualcomando é indefinido
! Undefined control sequence.
\reserved@a ->\@nil
l.29 ...exy{one \test}{two \test s}{three \test s}
?
Você usou uma definição primitiva do TeX, resultando em um comando frágil que quebra em um contexto de expansão. Se você usar \NewDocumentCommand
, obterá \protected
comandos seguros em tais construções.
Isso funciona sem erros
\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}
Mas produz
Underfull \hbox (badness 10000) in paragraph at lines 29--30
Devido ao posicionamento incorreto \\
que nunca deve ser usado no final de um parágrafo. (Seria melhor usar \par
)
Responder2
eu não usaria xstring
. O principal problema do seu código é que o comando \Etc
é (muito) frágil.
Minha proposta é usar expl3
para contar vírgulas e pontos de interrogação, dividindo o texto neles e contando os itens (e subtraindo um).
O argumento é totalmente expandido (mas o comando robusto não é) e “purificado”, então comandos como \textnormal
desaparecem.
\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}