Как правильно определить обертку для команды, которая принимает необязательный аргумент (например, \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}