Как определить оболочку для команды, которая принимает необязательный аргумент?

Как определить оболочку для команды, которая принимает необязательный аргумент?

Как правильно определить обертку для команды, которая принимает необязательный аргумент (например, \chapter)? Я имею в виду обертку, а не клон, чтобы мы могли внедрить код в обертку, не вмешиваясь в команду, оберткой которой она является.

Вот как я это делаю. Есть ли способ сделать это лучше? (Условное предложение я придумал egregздесь.)

\documentclass{book}

\newcommand\mychapter[2][] {\if\relax\detokenize{#1}\relax
                             \chapter{#2}
                            \else
                             \chapter[#1]{#2}
                            \fi}

\begin{document}
\tableofcontents
\mychapter[Short title]{Long title}
\end{document}

let\mychapter\chapterклонирует его, так что это \mychapterне будет оберткой для \chapter.

решение1

Да, есть способ получше:

\usepackage{xparse}

\NewDocumentCommand{\mychapter}{som}{%
  %%% things to do before \chapter
  \IfBooleanTF{#1}
    {\chapter*{#3}}
    {\IfNoValueTF{#2}{\chapter{#3}}{\chapter[#2]{#3}}%
  %%% things to do after \chapter
}

Это поддерживает все три вызова:

\mychapter*{Title}
\mychapter{Title}
\mychapter[Short title]{Long title}

решение2

TheтрадиционныйСпособ, ранее xparseдопускавший более гибкие решения, заключается в использовании \@ifnextchar[для проверки [наличия необязательного аргумента и включения injectдругого кода в оболочку.

Помеченная звездочкой версия также включена и может иметь []текущее состояние — автору поста решать, что с ней []делать ;-)

\documentclass{book}



\newcommand\mychapter[2][]{\if\relax\detokenize{#1}%
                             \chapter{#2}
                            \else
                             \chapter[#1]{#2}
                            \fi}

\makeatletter
\newcommand{\myotherchapter@@opt}[2][]{%
  \chapter[#1]{#2}%
}

\newcommand{\myotherchapter@@noopt}[1]{%
  \chapter{#1}%
}

\newcommand{\myotherchapter@@starredopt}[2][]{%
  % Decide yourself what to do with #1 ;-)
  \chapter*{#2}
}


\newcommand{\myotherchapter@@starrednoopt}[1]{%
  \myotherchapter@@starredopt[#1]{#1}%
}

\newcommand{\myotherchapterstarred}{%
  \@ifnextchar[{\myotherchapter@@starredopt}{\myotherchapter@@starrednoopt}%
}

\newcommand{\myotherchapterunstarred}{%
  \@ifnextchar[{\myotherchapter@@opt}{\myotherchapter@@noopt}%
}

\newcommand{\myotherchapter}{%
  \@ifstar{\myotherchapterstarred}{\myotherchapterunstarred}%
}

\makeatother

\begin{document}

\tableofcontents

\mychapter[Short title]{Long title}

\myotherchapter{First}
\myotherchapter[Short Title]{Long title}

\myotherchapter*[Starred Short Title]{Starred Long title}

\myotherchapter*{Starred Long title}



\end{document}

Связанный контент