Eigener Zähler startet bei 3 in der Tabularx-Umgebung

Eigener Zähler startet bei 3 in der Tabularx-Umgebung

Das folgende Beispiel zeigt, dass der selbst definierte Zähler in der tabularxUmgebung nicht mit „1“ beginnt. Ich weiß nicht, wie ich den Befehl so definieren kann, dass er immer mit „1“ beginnt. Ideen?

\documentclass{article}

\usepackage{enumitem}
\usepackage{cleveref}
\usepackage{etoolbox}
\usepackage{booktabs}
\usepackage{ltxtable}
\usepackage{ltablex}

\makeatletter\newcommand*{%
\CreateCounterIfNotExists}[1]{\begingroup%
\@ifundefined{c@#1}{\newcounter{#1}}{}%
\endgroup}\makeatother

\def\sdef#1{\expandafter\def\csname#1\endcsname}
\newcommand\requirement[3]{%
\CreateCounterIfNotExists{my#1}%
\sdef{themy#1}{#1\arabic{my#1}}%
\refstepcounter{my#1}%
\label{#2} #1\arabic{my#1}: #3%
}

\begin{document}
\requirement{A}{req:foo:a}{Starts with 1}
\requirement{A}{req:foo:b}{continues with 2}
\requirement{B}{req:hop:a}{Starts with 1}
\requirement{B}{req:hop:b}{continues with 2}
\begin{tabularx}{\textwidth}{lX}
    \caption{Requirements}\\
    \label{tbl:reqs}\\
    \toprule
    \textbf{Area}& \textbf{Description}  \\\midrule
    \requirement{C}{req:bar:A}{Starts with 3}
    \requirement{C}{req:bar:B}{Continues with 4}
       & 
    This should start with C1    \\\midrule
\end{tabularx}
\end{document}

Antwort1

Sie können keine Zähler definieren, während tabularxes seine Arbeit erledigt.

Eine Lösung könnte darin bestehen, den Bedarf im tabularxVoraus zu definieren oder dies zeitversetzt mithilfe der .auxDatei durchzuführen.

\documentclass{article}

\usepackage{enumitem}
\usepackage{etoolbox}
\usepackage{booktabs}
\usepackage{ltxtable}
\usepackage{ltablex}
\usepackage{cleveref}

\makeatletter
\newcommand\requirement[3]{%
  \protected@write\@auxout{}{\string\CreateCounter{#1}}%
  \@ifundefined{c@my#1}
    {#1??: #3}
    {\refstepcounter{my#1}\label{#2}#1\arabic{my#1}: #3}%
}
\newcommand{\CreateCounter}[1]{%
  \@ifundefined{c@my#1}
    {\newcounter{my#1}\global\@namedef{themy#1}{#1\arabic{my#1}}}
    {}%
}
\AtEndDocument{\let\CreateCounter\@gobble}
\makeatother

\begin{document}
{\requirement{A}{req:foo:a}{Starts with 1}}
\requirement{A}{req:foo:b}{continues with 2}
\requirement{B}{req:hop:a}{Starts with 1}
\requirement{B}{req:hop:b}{continues with 2}

\noindent
\begin{tabularx}{\textwidth}{lX}
\caption{Requirements}\label{tbl:reqs}\\
\toprule
\textbf{Area}& \textbf{Description}  \\
\midrule
\requirement{C}{req:bar:A}{Starts with 1}
\requirement{C}{req:bar:B}{Continues with 2} & This should start with C1 \\
\midrule
\end{tabularx}

\end{document}

Bildbeschreibung hier eingeben

Beim ersten \requirement{X}Gebrauch wird die Nummer nicht generiert, beim nächsten LaTeX-Lauf jedoch schon.

Antwort2

Sie sollten innerhalb eines Makros fast nie einen Zähler deklarieren. Er sollte einmal in der Präambel deklariert und einfach verwendet werden. Dann tabularxwürden Sie davon wissen und seinen Wert beibehalten, wenn Sie Testbreiteneinstellungen vornehmen.

\documentclass{article}

\usepackage{enumitem}
\usepackage{cleveref}
\usepackage{etoolbox}
\usepackage{booktabs}
\usepackage{ltxtable}
\usepackage{ltablex}

\newcounter{myA}\renewcommand\themyA{A\arabic{myA}}
\newcounter{myB}\renewcommand\themyB{B\arabic{myB}}
\newcounter{myC}\renewcommand\themyC{C\arabic{myC}}



\newcommand\requirement[3]{%
\refstepcounter{my#1}%
\label{#2}#1\arabic{my#1}: #3%
}

\begin{document}
\requirement{A}{req:foo:a}{Starts with 1}
\requirement{A}{req:foo:b}{continues with 2}
\requirement{B}{req:hop:a}{Starts with 1}
\requirement{B}{req:hop:b}{continues with 2}
\begin{tabularx}{\textwidth}{lX}
    \caption{Requirements}\\
    \label{tbl:reqs}\\
    \toprule
    \textbf{Area}& \textbf{Description}  \\\midrule
    \requirement{C}{req:bar:A}{Starts with 3}
    \requirement{C}{req:bar:B}{Continues with 4}
       & 
    This should start with C1    \\\midrule
\end{tabularx}
\end{document}

Antwort3

Wie in den Lösungen von David Carlisle und egreg erwähnt: Zähler sollten nicht definiert werden on-the-fly.

Ich habe einen ähnlichen Ansatz wie David Carlisle verwendet und die Befehle \NewDocumentCounter, \DeclareRequirement, bereitgestellt DeclareRequirements, die nur in der Präambel verwendet werden können, um eine versehentliche Verwendung zu verhindern.

\documentclass{article}

\usepackage{enumitem}
\usepackage{etoolbox}
\usepackage{booktabs}
\usepackage{ltxtable}
\usepackage{ltablex}
\usepackage{cleveref}

\usepackage{xparse}

\NewDocumentCommand{\NewDocumentCounter}{m}{%
  \ifltxcounter{#1}{%
  }{%
    \typeout{Creating counter #1}
    \newcounter{#1}%
  } 
}

\NewDocumentCommand{\DeclareRequirement}{m}{%
  \NewDocumentCounter{my#1}%
  \expandafter\renewcommand\csname themy#1\endcsname{#1\arabic{my#1}}
}%

    \listgadd\reqlist{}


\NewDocumentCommand{\DeclareRequirements}{+m}{%
  \forcsvlist{\listgadd\reqlist}{#1}
  \forlistloop{\DeclareRequirement}{\reqlist}
}%



\makeatletter
\@onlypreamble{\NewDocumentCounter}
\@onlypreamble{\DeclareRequirement}
\@onlypreamble{\DeclareRequirements}
\makeatother




\DeclareRequirements{A,B,C} % Declare all requirement counters in a row ... 


\newcommand\requirement[3]{%
  \refstepcounter{my#1}%
  \label{#2} #1\arabic{my#1}: #3%
}

\begin{document}
\requirement{A}{req:foo:a}{Starts with 1}
\requirement{A}{req:foo:b}{continues with 2}
\requirement{B}{req:hop:a}{Starts with 1}
\requirement{B}{req:hop:b}{continues with 2}
\begin{tabularx}{\textwidth}{lX}
    \caption{Requirements}\\
    \label{tbl:reqs}\\
    \toprule
    \textbf{Area}& \textbf{Description}  \\\midrule
    \requirement{C}{req:bar:A}{Starts with 3 no longer} %\themyC
    \requirement{C}{req:bar:B}{Continues with 4 no longer} %\themyC
    & 
    This should start with C1    \\\midrule
  \end{tabularx}
\end{document}

verwandte Informationen