\def anidados causando error \inaccesible

\def anidados causando error \inaccesible

Probé una forma, al menos a mis ojos, muy sencilla de poner las palabras en mayúscula:

\documentclass{article}

\begin{document}
\begingroup%
\obeyspaces%\catcode`\ \active
\def {\space\MakeUppercase}%
Hello world
\endgroup
\end{document}

No hace falta decir que también intenté incluir esto en una macro como

\documentclass{article}

\newcommand{\capitalize}[1]{\begingroup\obeyspaces\def {\space\MakeUppercase}#1\endgroup}

\begin{document}
\capitalize{Hello world}
\end{document}

lo que hará que TeX se queje de la sintaxis del internal \defy lo haga \inaccessible.

¿Qué es exactamente lo que está mal y hay alguna forma de solucionarlo?

Respuesta1

Esto es lo habitual \verbque no funciona en un problema de discusión.

\obeyspacescambia el catcode del espacio, lo que significa que un espaciopersonajeen unarchivose convierte en un token activo. los cambios de catcode tienenNoefecto sobre tokens ya creados. En su caso, todo el argumento de \newcommandha sido tokenizado y, por lo tanto, no hay ningún token de espacio después de \defque sea \def{.

Debe cambiar el código cat del espacio antes de \newcommandy \capitalizenecesitaría cambiar el código cat del espacio antes de tomar su argumento. (Por esta y otras razones, no usaría un cambio de código cat para esto, sino que simplemente usaría un argumento delimitado para encontrar espacios normales)

\documentclass{article}

\newcommand{\capitalize}[1]{\xcapitalize#1 \relax}
\def\xcapitalize#1 #2{%
#1%
\ifx\relax#2%
\else
\space\MakeUppercase{#2}%
\expandafter\xcapitalize
\fi}

\begin{document}
\capitalize{Hello world and this}
\end{document}

una versión solicitada en los comentarios que también escribe en mayúscula la primera letra y permite que el argumento sea una macro:

\documentclass{article}

\newcommand{\capitalize}[1]{\ignorespaces
\expandafter\expandafter\expandafter
\xcapitalize\expandafter\space #1 \relax}
\def\xcapitalize#1 #2{%
#1%
\ifx\relax#2%
\else
\space\MakeUppercase{#2}%
\expandafter\xcapitalize
\fi}

\begin{document}
\capitalize{hello world and this}

\newcommand\zzz{hello world and this}
\capitalize{\zzz}
\end{document}

Respuesta2

Esta solución requiere una distribución TeX actualizada recientemente. Tiene la ventaja, sobre el enfoque con \obeyspaces, de que no se cambia ningún código de categoría, por lo que la macro también puede ir en el argumento de otros comandos.

\documentclass{article}
\usepackage{xparse}

\ExplSyntaxOn
\NewDocumentCommand{\capitalize}{m}
 {
  \ruben_capitalize:n { #1 }
 }

\seq_new:N \l__ruben_capitalize_words_seq
\seq_new:N \l__ruben_capitalize_out_seq

\cs_new_protected:Npn \ruben_capitalize:n #1
 {
  \seq_set_split:Nnn \l__ruben_capitalize_words_seq { ~ } { #1 }
  \seq_set_map:NNn \l__ruben_capitalize_out_seq \l__ruben_capitalize_words_seq
   {
    \tl_mixed_case:n { ##1 }
   }
  \seq_use:Nn \l__ruben_capitalize_out_seq { ~ }
 }
\ExplSyntaxOff

\begin{document}
\capitalize{Hello world}
\end{document}

Si también desea poder poner en mayúscula las cadenas que se pasan como una macro, simplemente cambie la definición de \capitalizea

\NewDocumentCommand{\capitalize}{m}
 {
  \ruben_capitalize:o { #1 }
 }

y añadir

\cs_generate_variant:Nn \ruben_capitalize:n { o }

después de la definición de \ruben_capitalize:n(es decir, justo antes de \ExplSyntaxOff).

Ejemplo completo:

\documentclass{article}
\usepackage{xparse}

\ExplSyntaxOn
\NewDocumentCommand{\capitalize}{m}
 {
  \ruben_capitalize:o { #1 }
 }

\seq_new:N \l__ruben_capitalize_words_seq
\seq_new:N \l__ruben_capitalize_out_seq

\cs_new_protected:Npn \ruben_capitalize:n #1
 {
  \seq_set_split:Nnn \l__ruben_capitalize_words_seq { ~ } { #1 }
  \seq_set_map:NNn \l__ruben_capitalize_out_seq \l__ruben_capitalize_words_seq
   {
    \tl_mixed_case:n { ##1 }
   }
  \seq_use:Nn \l__ruben_capitalize_out_seq { ~ }
 }
\cs_generate_variant:Nn \ruben_capitalize:n { o }
\ExplSyntaxOff

\newcommand{\myhello}{hello world}

\begin{document}
\capitalize{Hello world}

\capitalize{hello world}

\capitalize{\myhello}

\capitalize\myhello
\end{document}

ingrese la descripción de la imagen aquí


El enfoque clásico \obeylinesrequiere que usted lo emita.antesabsorbiendo el argumento:

% First setup obeyspace and give a meaning to active space
\newcommand{\capitalize}{\begingroup\obeyspaces\setupcapspace\docapitalize}
% Just absorb the argument and end the group
\newcommand{\docapitalize}[1]{#1\endgroup}
% Define (locally) the behavior of active space
\begingroup\lccode`~=`\ % <--- don't forget this one
  \lowercase{\endgroup\newcommand\setupcapspace{\def~{\space\MakeUppercase}}}

Las dos últimas líneas también pueden ser

{\obeyspaces\gdef\setupcapspace{\def {\space\MakeUppercase}}}

pero el \lowercaseenfoque evita \obeyspacesposibles problemas con espacios espurios.

Sin embargo, un enfoque de argumento delimitado es seguramente mejor, porque permite \capitalizeestar en el argumento de otro comando.

Respuesta3

Una solución basada en LuaLaTeX. Definimos una nueva macro llamada \capitalizeque emplea las funciones de Lua string.uppery tex.sprint. El argumento de \capitalizepuede ser una cadena codificada o una macro que, presumiblemente, genera una cadena.

ingrese la descripción de la imagen aquí

% !TEX TS-program = lualatex
\documentclass{article}
{\catcode\%=12 
 \gdef\capitalize#1{
   \directlua{ str="#1"; 
               tex.sprint ( string.gsub(" "..str, "%W%l",
                string.upper):sub(2)) } }
} 
\begin{document}

\capitalize{Once upon a time there was a princess
  who lived in a great palace that was close to the 
  edge of a dark and mysterious forest.}
\end{document}

Respuesta4

Si bien su pregunta es sobre palabras anidadas \def, la aplicación escribe palabras en mayúscula. el titlecapspaquete hace esto con la \titlecapmacro. Permite una amplia flexibilidad en el argumento, incluidos cambios de estilo y tamaño de fuente. También le permite configurar las palabras de exclusión para que no estén en mayúscula (excepto opcionalmente como la primera palabra del argumento). Puede, en gran medida, superar la puntuación inicial (como paréntesis y corchetes) al escribir palabras en mayúscula, etc. Puede escribir signos diacríticos en mayúscula, etc.

En MWE, muestro su ejemplo de "hola mundo" y luego empleo la muestra exagerada de la documentación del paquete.

\documentclass{article}
\usepackage{titlecaps}
\def\bs{$\backslash$}

\begin{document}
\titlecap{hello world}

\Addlcwords{for a is but and with of in as the etc on to if}
\titlecap{% 
to know that none of the words typed in this paragraph were initially
upper cased might be of interest to you.  it is done to demonstrate the
behavioral features of this package.  first, you should know the words
that i have pre-designated as lower case.  they are:  ``for a is but and
with of in as the etc on to if.''  you can define your own list.  note
that punctuation, like the period following the word ``if'' did not mess
up the search for lower case (nor did the quotation marks just now).
punctuation which is screened out of the lower-cased word search pattern
include . , : ; ( ) [ ] ? ! ` ' however, I cannot screen text braces;
\{for example in\} is titled, versus (for example in), since the braces
are not screened out in the search for pre-designated lower-case words
like for and in.  However, \texttt{\bs textnc} provides a workaround:
\{\textnc{for example in}\}.  titlecap will consider capitalizing
following a (, [, \{, or - symbol, such as (abc-def).  you can use your
text\textit{\relax xx} commands, like i just did here with the prior xx,
but if you want the argument of that command to not be titled, you
either need, in this example, to add \textit{xx} to the lowercase word
list, which you can see i did not.  instead, i put ``\bs relax~xx'' as
the argument, so that, in essence, the \bs relax was capitalized, not
the x.  Or you could use \texttt{\bs textnc} .  here i demonstrate that
text boldface, \textbf{as in the \bs textbf command}, also works fine,
as do \texttt{texttt}, \textsl{textsl}, \textsc{textsc},
\textsf{textsf}, \textit{etc}.  titlecap will work on diacritical marks,
such as \"apfel, \c cacao \textit{etc.}, \scriptsize fontsize \LARGE
changing commands\normalsize\unskip, as well as national symbols such as
\o laf, \ae gis, and \oe dipus.  unfortunately, i could not get it to
work on the \aa~nor the \l~symbols. the method will work with some
things in math mode, capitalizing symbols if there is a leading space,
$x^2$ can become $ x^2$, and it can process but it will not capitalize
the greek symbols, such as $\alpha$, and will choke on most macros, if
they are not direct character expansions.  Additionally,
\textsf{titlecaps} also works with font changing declarations, for
example, \bs itshape\bs sffamily. \itshape\sffamily you can see that it
works fine.  likewise, any subsequent \bs textxx command will, upon
completion, return the font to its prior state, such as this
\textbf{textbf of some text}.  you can see that i have returned to the
prior font, which was italic sans-serif. now I will return to upright
roman\upshape\rmfamily.  a condition that will not behave well is inner
braces, such as \ttfamily \bs titlecap\{blah \{inner brace material\}
blah-blah\}. \rmfamily see the section on quirks and limitations for a
workaround involving \texttt{\bs textnc}.  titlecap will always
capitalize the first word of the argument (\textbf{even if it is on the
lower-case word list}), unless \texttt{\bs titlecap} is invoked with an
optional argument that is anything other than a capital p.  in that case,
the first word will be titled \textit{unless} it is on the lowercase
word list.  for example, i will do a \bs titlecap[\relax s]\{\relax
a~big~man\} and get ``\titlecap[s]\textnc{a big man}'' with the ``a''
not titled.  i hope this package is useful to you, but as far as using
\textsf{titlecaps} on such large paragraphs\ldots \textbf{do not try
this at home!}}
\end{document}

ingrese la descripción de la imagen aquí

información relacionada