Como o foreach pode ser usado para criar macros sinônimas?

Como o foreach pode ser usado para criar macros sinônimas?

Preciso de muitos sinônimos de muitas macros com referências de índices? Como uso o token \foreach como nome de \newcommand? Procuro substituir a saída ACTUAL pela saída MOCKUP gerando e usando as macros \hello, \hola e \bonjour (todas com subscrito 1) e \world, \mundo e \monde (todas com subscrito 2), porque todos os sinônimos referem-se ao mesmo número de figura. Se houver uma maneira mais simples de fazer isso, mostre-a. Soletrar tudo copiando e editando é muito tedioso.

COMPILAR:

bash$ for i in 1 2 3 4; do pdflatex MWE.tex ; done

FONTE:

% filename: MWE.tex

\documentclass[10pt,twoside]{book}

\usepackage{tikz}
\begin{document}

\begin{center}{\bf\LARGE MWE}\end{center}

\begin{figure}\tikz{\node at (0,0) {hello};}\caption{hello}\label{hello}\end{figure}
\begin{figure}\tikz{\node at (0,0) {world};}\caption{world}\label{world}\end{figure}

GOAL: produce synonymous variant macros referencing a key figure.\\\\

EXPECT:\\
{\bf key:} $hello_1$ {\bf variants:} $hola_1$ $bonjour_1$\\
{\bf key:} $world_2$ {\bf variants:} $mundo_2$ $monde_2$\\

MOCKUP:\\
% NOTICE: this exercises the loops needed, but how do I generate named macros?
% PROBLEM: \foreach defines \key and \variant so \newcommand can't reuse them.
% FAILURE: \expandafter\newcommand\csname\variant\endcsname{$\variant~\ref{\key}}
\foreach\key/\variants in {hello/{hola,bonjour},world/{mundo,monde}}{
    {\bf key:} $\key_{\ref{\key}}$ {\bf variants:}
    \foreach\variant in \variants{$\variant_{\ref{\key}}$\;} \\
}

ACTUAL:\\
TODO
% TODO uncomment these next two lines when the \newcommands work correctly.
%key: \hello variants: \hola \bonjour
%key: \world variants: \mundo \monde
\end{document}

Responder1

Usuário TNX202729. Uma solução Python funcionou perfeitamente.

Primeiro o código fonte do IDEAS.py:

#!/usr/bin/env python3

(BS, OB, CB) = ("\\", "{", "}")
IDEAS = {
    "hello": {"hola", "bonjour"},
    "world": {"mundo", "monde"}
}

def newcommand(num, key, syn):
    print(
        BS      + "newcommand" +
        BS      + syn +
        OB + BS + "large " + syn +
        BS      + "raisebox{-2pt}" +
        OB + BS + "footnotesize" +
        BS      + "ref{" + key + "}" +
        CB + CB
    )

def synonyms(num, key):
    newcommand(num + 1, key, key)
    for syn in IDEAS[key]:
        newcommand(num + 1, key, syn)

for num, key in enumerate(IDEAS):
    synonyms(num, key)

Então linha de comando:

bash$ IDEAS.py > IDEAS.tex

Então látex:

\input{IDEAS}

Responder2

Um problema é que cada \foreachiteração ocorre dentro de seu próprio escopo local.
Portanto, uma definição executada em termos de \newcommandé restrita ao escopo local que é aberto no início da \foreachiteração e fechado no final da \foreachiteração durante a qual ocorre a execução dessa definição.
Sugiro acumular \newcommand-directives dentro de um registro de token de rascunho onde as atribuições são feitas globalmente e ter o conteúdo desse registro entregue quando a iteração for concluída. Você precisa de alguns \expandaftertruques para ter as \foreachmacros -"variáveis" expandidas corretamente.

Provavelmente o código a seguir faz o que você deseja:

% filename: MWE.tex

\documentclass[10pt,twoside]{book}

\newtoks\scratchtoks

\usepackage{tikz}
\begin{document}

\begin{center}{\bf\LARGE MWE}\end{center}

\begin{figure}\tikz{\node at (0,0) {hello};}\caption{hello}\label{hello}\end{figure}
\begin{figure}\tikz{\node at (0,0) {world};}\caption{world}\label{world}\end{figure}

GOAL: produce synonymous variant macros referencing a key figure.\\\\

EXPECT:\\
{\bf key:} $hello_1$ {\bf variants:} $hola_1$ $bonjour_1$\\
{\bf key:} $world_2$ {\bf variants:} $mundo_2$ $monde_2$\\

MOCKUP:\\
% NOTICE: this exercises the loops needed, but how do I generate named macros?
% PROBLEM: \foreach defines \key and \variant so \newcommand can't reuse them.
% FAILURE: \expandafter\newcommand\csname\variant\endcsname{$\variant~\ref{\key}}
\global\scratchtoks{\global\scratchtoks{}}%
\foreach\key/\variants in {hello/{hola,bonjour},world/{mundo,monde}}{%
    %---------------------------------------------------------------------------
    \global\scratchtoks\expandafter{%
      \the\expandafter\scratchtoks
          \expandafter\newcommand
          \csname\key\expandafter\expandafter\expandafter\endcsname
                     \expandafter\expandafter\expandafter{%
                     \expandafter\expandafter\expandafter$%
                     \expandafter\key
                     \expandafter_%
                     \expandafter{%
                     \expandafter\ref
                     \expandafter{\key}}$}%
    }%
    %---------------------------------------------------------------------------
    {\bf key:} $\key_{\ref{\key}}$ {\bf variants:}
    \foreach\variant in \variants{%
      %-------------------------------------------------------------------------
      \global\scratchtoks\expandafter{%
        \the\expandafter\scratchtoks
            \expandafter\newcommand
            \csname\variant\expandafter\expandafter\expandafter\endcsname
                       \expandafter\expandafter\expandafter{%
                       \expandafter\expandafter\expandafter$%
                       \expandafter\variant
                       \expandafter_%
                       \expandafter{%
                       \expandafter\ref
                       \expandafter{\key}}$}%
      }%
      %-------------------------------------------------------------------------
      $\variant_{\ref{\key}}$\;%
    }\\
}%
\the\scratchtoks

ACTUAL:\\
% TODO uncomment these next two lines when the \newcommands work correctly.
{\bf key:} \hello { \bf variants:} \hola\; \bonjour\;\\
{\bf key:} \world { \bf variants:} \mundo\; \monde\;\\
%\show\hello
%\show\hola
%\show\bonjour
%\show\world
%\show\mundo
%\show\monde
\end{document}

insira a descrição da imagem aqui


Alternativamente, em vez de acumular em um registro de token, você pode usar expl3 \cs_new:cpxpara definir coisas globalmente:

% filename: MWE.tex

\documentclass[10pt,twoside]{book}
\usepackage{tikz}
\begin{document}

\begin{center}{\bf\LARGE MWE}\end{center}

\begin{figure}\tikz{\node at (0,0) {hello};}\caption{hello}\label{hello}\end{figure}
\begin{figure}\tikz{\node at (0,0) {world};}\caption{world}\label{world}\end{figure}

GOAL: produce synonymous variant macros referencing a key figure.\\\\

EXPECT:\\
{\bf key:} $hello_1$ {\bf variants:} $hola_1$ $bonjour_1$\\
{\bf key:} $world_2$ {\bf variants:} $mundo_2$ $monde_2$\\

MOCKUP:\\
% NOTICE: this exercises the loops needed, but how do I generate named macros?
% PROBLEM: \foreach defines \key and \variant so \newcommand can't reuse them.
% FAILURE: \expandafter\newcommand\csname\variant\endcsname{$\variant~\ref{\key}}
\ExplSyntaxOn
\foreach\key/\variants in {hello/{hola,bonjour},world/{mundo,monde}}{
    %---------------------------------------------------------------------------
    \cs_new:cpx{\key}
               {\c_math_toggle_token\exp_not:o {\key}\c_math_subscript_token{\exp_not:N\ref{\exp_not:o {\key}}}\c_math_toggle_token}
    %---------------------------------------------------------------------------
    {\bf key:}~\c_math_toggle_token\key\c_math_subscript_token{\ref{\key}}\c_math_toggle_token~{\bf variants:}~
    \foreach\variant in \variants{
      %-------------------------------------------------------------------------
      \cs_new:cpx{\variant}
                 {\c_math_toggle_token\exp_not:o {\variant}\c_math_subscript_token{\exp_not:N\ref{\exp_not:o {\key}}}\c_math_toggle_token}
      %-------------------------------------------------------------------------
      \c_math_toggle_token\variant\c_math_subscript_token{\ref{\key}}\c_math_toggle_token\;
    }\\
}
\ExplSyntaxOff

ACTUAL:\\
% TODO uncomment these next two lines when the \newcommands work correctly.
{\bf key:} \hello { \bf variants:} \hola\; \bonjour\;\\
{\bf key:} \world { \bf variants:} \mundo\; \monde\;\\
%\show\hello
%\show\hola
%\show\bonjour
%\show\world
%\show\mundo
%\show\monde
\end{document}

insira a descrição da imagem aqui

informação relacionada