
次の例は、環境内の自己定義カウンターがtabularx
「1」で始まらないことを示しています。コマンドを常に「1」で始まるように定義する方法がわかりません。アイデアはありますか?
\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}
答え1
tabularx
が機能している間は、カウンターを定義することはできません。
解決策としては、必要なものをtabularx
事前に定義するか、ファイルを使用して遅延方式で定義することが考えられます.aux
。
\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}
初めて\requirement{X}
使用する場合、番号は生成されませんが、次回の LaTeX 実行時には生成されます。
答え2
マクロ内でカウンターを宣言することはほとんどなく、プリアンブルで一度宣言して使用するだけです。そうすれば、tabularx
試行幅の設定時にカウンターを認識し、その値を保持します。
\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}
答え3
David Carlisle と egreg によるソリューションで述べられているように、カウンターは定義されるべきではありませんon-the-fly
。
私は David Carlisle と同様のアプローチを採用し、誤って使用されるのを防ぐために、プリアンブルでのみ使用できる\NewDocumentCounter
、\DeclareRequirement
、コマンドを提供しました。DeclareRequirements
\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}