% \iffalse %% File: nameref.dtx %% Copyright (C) 1995-1999 Sebastian Rahtz %% 2000 Sebastian Rahtz, Heiko Oberdiek %% 2001-2010 Heiko Oberdiek %% %% This file is part of the `Hyperref Bundle'. %% ------------------------------------------- %% %% This work may be distributed and/or modified under the %% conditions of the LaTeX Project Public License, either version 1.3 %% of this license or (at your option) any later version. %% The latest version of this license is in %% http://www.latex-project.org/lppl.txt %% and version 1.3 or later is part of all distributions of LaTeX %% version 2005/12/01 or later. %% %% This work has the LPPL maintenance status `maintained'. %% %% The Current Maintainer of this work is Heiko Oberdiek. %% %% The list of all files belonging to the `Hyperref Bundle' is %% given in the file `manifest.txt'. % %<*dtx> \ProvidesFile{nameref.dtx} % %\NeedsTeXFormat{LaTeX2e} %\ProvidesPackage{nameref} %\ProvidesFile{nameref.drv} % \fi % \ProvidesFile{nameref.dtx} [2010/01/25 v2.36 Cross-referencing by name of section]% % % \CheckSum{580} % % \iffalse %<*driver> \documentclass{ltxdoc} \usepackage[numbered]{hypdoc} \usepackage{bmhydoc} \EnableCrossrefs \CodelineIndex \begin{document} \GetFileInfo{nameref.sty}% \title{% Section name references in \LaTeX \thanks{% This file has version number \fileversion, % last revised \filedate.% }% }% \author{Sebastian Rahtz}% \date{\filedate}% \maketitle \makeatletter \@ifundefined{HyperrefOverview}{}{% \HyperrefOverview*% }% \tableofcontents \DocInput{nameref.dtx}% \PrintIndex \end{document} % % \fi % % \StopEventually{} % % \section{Introduction} % Cross-referencing to include the \emph{name} of the section, rather than % just the number or page. This works by redefining some of the inside % \LaTeX\ macros, so it is vulnerable to future changes. % \section{History} % Sebastian Rahtz, for Lou Burnard, March 15th 1994;\\ % SPQR CERN July 1994;\\ % Last mod. Sept. 3th MG;\\ % Sept. 19th SPQR;\\ % April 11th 1995 SPQR (added section formatting hook);\\ % April 14th 1995 SPQR (compatibility with hyperref);\\ % June 22 1995 SPQR (removed typeout from Sectionformat);\\ % September 22 1997 added varioref hacks by Corey Minyard;\\ % September 28th changes by David Carlisle.\\ % More recent changes are recorded in ChangeLog. % % The versions before 2009/11/27 v2.32 did not expand the % title strings (of \cs{section}, \cs{caption}, \dots). % Then the behaviour switched to expanding, because this % allows to remove unwanted macros (\cs{label}, \dots). % Despite this method is used in package \textsf{titleref} % and class \textsf{memoir}, the first feedbacks let me conclude % that this is not robust enough. Therefore I wrote package % \textsf{gettitlestring} that uses the non-expanding % method as default. It also allows me to use this package % in package \textsf{zref-titleref}. % See the package documentation of package \textsf{gettitlestring} % how to change the method. It also provides a hook to remove % additional unwanted macros, not supported by the package. % Example: %\begin{quote} %\begin{verbatim} %\usepackage{gettitlestring} %\GetTitleStringDisableCommands{% % \let\mylabel\@gobble % \renewcommand\myindex[2]{}% %} %\end{verbatim} %\end{quote} % % \section{Implementation} % \begin{macrocode} %<*package> % \end{macrocode} % % \begin{macrocode} \RequirePackage{refcount}[2006/02/12] \RequirePackage{gettitlestring}[2009/12/18] % \end{macrocode} % % We redefine |\label| so that it also writes the name of the % current section to the .aux file; if the name ends in a dot, % we zap it. To allow for the hyperref package, also write % fourth and fifth fields (empty by default; % used for cross-ref name, and file). % \begin{macrocode} \def\strip@period#1.\relax#2\@@@{#1} % \end{macrocode} % % \subsection{\cs{label} stuff} % % Here is tested, whether package showkeys is loaded. % But if option final is used, the macros of showkeys % are not defined, so we have to check an additional % test. % \begin{macrocode} \newif\ifNR@showkeys \NR@showkeysfalse \@ifpackageloaded{showkeys}{% \begingroup \@ifundefined{SK@@label}{% }{% \global\NR@showkeystrue }% \endgroup }{} % \end{macrocode} % \begin{macrocode} \def\label#1{% \@bsphack \begingroup \@onelevel@sanitize\@currentlabelname \edef\@currentlabelname{% \expandafter\strip@period\@currentlabelname\relax.\relax\@@@% }% \protected@write\@auxout{}{% \string\newlabel{#1}{% {\@currentlabel}% {\thepage}% {\@currentlabelname}% {\@currentHref}{}% }% }% \endgroup \@esphack }% \ifNR@showkeys \def\label#1{% \@bsphack \SK@\SK@@label{#1}% \begingroup \@onelevel@sanitize\@currentlabelname \edef\@currentlabelname{% \expandafter\strip@period\@currentlabelname\relax.\relax\@@@% }% \protected@write\@auxout{}{% \string\newlabel{#1}{% {\@currentlabel}% {\thepage}% {\@currentlabelname}% {\@currentHref}{}% }% }% \endgroup \@esphack }% \fi % \end{macrocode} % \begin{macrocode} \let\ltx@label\label % \end{macrocode} % Needed for the \emph{subeqnarray} package. % \begin{macrocode} \@ifundefined{slabel}{}{% \def\slabel#1{% \@bsphack \if@filesw {% \@onelevel@sanitize\@currentlabelname \edef\@currentlabelname{% \expandafter\strip@period\@currentlabelname\relax.\relax\@@@% }% \let\thepage\relax \def\protect{\noexpand\noexpand\noexpand}% \edef\@tempa{% \write\@auxout{% \string\newlabel{#1}{% {\thesubequation}% {\thepage}% {\@currentlabelname}% {\@currentHref}{}% }% }% }% \expandafter }% \@tempa \if@nobreak\ifvmode\nobreak\fi\fi \fi \@esphack }% } % \end{macrocode} % Overload an AMS \LaTeX\ command, which uses |\newlabel|. Sigh! % \begin{macrocode} \def\@writetocindents@{% \begingroup \@for\@tempa:=-1,0,1,2,3\do{% \immediate\write\@auxout{% \string\newlabel{tocindent\@tempa}{% \csname r@tocindent\@tempa\endcsname{}{}{}{}% }% }% }% \endgroup } % \end{macrocode} % % \subsection{Changes of section macros} % % Add to the underlying section heading macros so that they % note the section name for use by label. % % If a section heading or the like has a |\label| in it, % we need to extract it, or subsequent processing breaks. % % \begin{macrocode} \def\NR@gettitle#1{% \GetTitleString{#1}% \let\@currentlabelname\GetTitleStringResult } % \end{macrocode} % % First the numbered sections. While we are about it, put in % a useful section formatting macro. % \begin{macrocode} \let\NR@sect\@sect \newcounter{section@level} \def\@sect#1#2#3#4#5#6[#7]#8{% \setcounter{section@level}{#2}% \NR@gettitle{#7}% \NR@sect{#1}{#2}{#3}{#4}{#5}{#6}[{#7}]{\Sectionformat{#8}{#2}}% } % \end{macrocode} % and now the unnumbered ones % \begin{macrocode} \let\NR@ssect\@ssect \def\@ssect#1#2#3#4#5{% \NR@gettitle{#5}% \NR@ssect{#1}{#2}{#3}{#4}{\Sectionformat{#5}{#1}}% } % \end{macrocode} % % Parts and chapters are treated differently. sigh. % % \begin{macrocode} \let\NR@part\@part \def\@part[#1]#2{% \NR@gettitle{#1}% \NR@part[{#1}]{#2}% } \let\NR@chapter\@chapter \def\@chapter[#1]#2{% \NR@gettitle{#1}% \NR@chapter[{#1}]{#2}% } \let\NR@schapter\@schapter \def\@schapter#1{% \NR@gettitle{#1}% \NR@schapter{#1}% } % \end{macrocode} % % Captions % % \begin{macrocode} \AtBeginDocument{% \@ifpackageloaded{hyperref}{% \@ifpackagelater{hyperref}{2009/12/05}\@gobble\@firstofone }\@firstofone {% \let\NR@@caption\@caption \long\def\@caption#1[#2]{% \NR@gettitle{#2}% \NR@@caption{#1}[{#2}]% }% }% } % \end{macrocode} % % \begin{macrocode} \AtBeginDocument{% \@ifpackageloaded{longtable}{% \let\NRorg@LT@c@ption\LT@c@ption \def\LT@c@ption#1[#2]#3{% \NRorg@LT@c@ption{#1}[{#2}]{#3}% \def\@tempa{#2}% \ifx\@tempa\@empty \else \NR@gettitle{#2}% \fi }% }{}% } % \end{macrocode} % % Compatibility for package titlesec. % \begin{macrocode} \AtBeginDocument{% \@ifpackageloaded{titlesec}{% \let\NRorg@ttl@sect@i\ttl@sect@i \def\ttl@sect@i#1#2[#3]#4{% \NR@gettitle{#4}% \NRorg@ttl@sect@i{#1}{#2}[{#3}]{#4}% }% }{}% } % \end{macrocode} % % \subsubsection{Class beamer} % % \begin{macrocode} \@ifclassloaded{beamer}{% \def\NR@PatchBeamer#1{% \expandafter\let\csname NRorg@beamer@#1\expandafter\endcsname \csname beamer@#1\endcsname \begingroup \edef\x{\endgroup \long\def\expandafter\noexpand\csname beamer@#1\endcsname [####1]{% hash-ok \noexpand\NR@gettitle{####1}% \expandafter\noexpand\csname NRorg@beamer@#1\endcsname [{####1}]% }% }% \x }% \NR@PatchBeamer{section}% \NR@PatchBeamer{subsection}% \NR@PatchBeamer{subsubsection}% }{} % \end{macrocode} % % \subsection{Referencing stuff} % % We default the label and hypertext reference to be empty. % \begin{macrocode} \def\@currentlabelname{} \def\@currentHref{} % \end{macrocode} % Since the second part of the preserved reference now has five parts % (number, page, name, hypertext reference and file), we need extra utility % macros: % \begin{macrocode} \long\def\@firstoffive#1#2#3#4#5{#1} \long\def\@secondoffive#1#2#3#4#5{#2} \long\def\@thirdoffive#1#2#3#4#5{#3} \long\def\@fourthoffive#1#2#3#4#5{#4} \long\def\@fifthoffive#1#2#3#4#5{#5} % \end{macrocode} % We have to redefine |\ref| and |\pageref| to know about extra % reference elements. Make them robust, and compatible with Babel. % The Babel switch is mimicked if not already defined. % Some support for package |showkeys| is implemented. If % option |notref| of that package is set, |\SK@ref| has the % meaning of |\@empty|. % \begin{macrocode} \providecommand*\@safe@activestrue{}% \providecommand*\@safe@activesfalse{}% \def\NR@setref#1{% \begingroup \@safe@activestrue \expandafter \endgroup \expandafter\NR@@setref\csname r@#1\endcsname } \def\NR@@setref#1{% \ifx\@undefined#1% \let#1\relax \fi \@setref#1% } \def\T@ref#1{% \NR@setref{#1}\@firstoffive{#1}% }% \def\T@pageref#1{% \NR@setref{#1}\@secondoffive{#1}% }% \ifNR@showkeys \ifx\SK@ref\@empty \else \def\T@ref#1{% \@safe@activestrue \SK@\SK@@ref{#1}% \@safe@activesfalse \NR@setref{#1}\@firstoffive{#1}% }% \def\T@pageref#1{% \@safe@activestrue \SK@\SK@@ref{#1}% \@safe@activesfalse \NR@setref{#1}\@secondoffive{#1}% }% \fi \fi % \end{macrocode} % Package |hyperref| uses its own definitions of |\ref| and % |\pageref| because it extends the syntax by the star form. % |\DeclareRobustCommand| is used by hyperref. % Package |nameref| can be loaded before, after hyperref or after % hyperref's definitions in |\AtBeginDocument|. % \begin{macrocode} \@ifpackageloaded{hyperref}{% }{% \DeclareRobustCommand{\ref}{\T@ref}% \DeclareRobustCommand{\pageref}{\T@pageref}% } % \end{macrocode} % % \subsection{Usage and frontend} % Access the (third) name part with |\nameref|; % \begin{macrocode} \def\T@nameref#1{% \begingroup \let\label\@gobble \NR@setref{#1}\@thirdoffive{#1}% \endgroup }% \ifNR@showkeys \ifx\SK@ref\@empty \else \def\T@nameref#1{% \@safe@activestrue \SK@\SK@@ref{#1}% \@safe@activesfalse \begingroup \let\label\@gobble \NR@setref{#1}\@thirdoffive{#1}% \endgroup }% \fi \fi \def\nameref{\protect\T@nameref} % \end{macrocode} % An extended form which gives title and page number. % \begin{macrocode} \def\Nameref#1{`\nameref{#1}' on page~\pageref{#1}} % \end{macrocode} % % The default for |\Sectionformat| % \begin{macrocode} \providecommand\Sectionformat[2]{#1} % \end{macrocode} % % \subsection{Package ifthen support} % % Package ifthen's \cmd{\isodd} does not work with % \cmd{\pageref} because of the extra link and the % star form. Therefore we need an expandable variant. % % Unfortunately there is no hook in \cmd{\ifthenelse} in order % to provide expandable versions without link of \cmd{\ref} % and \cmd{\pageref}. As possible target I could only find % \cmd{\begingroup}: % \begin{macrocode} \@ifpackageloaded{ifthen}{% \let\NROrg@ifthenelse\ifthenelse \def\ifthenelse{% \let\begingroup\NR@TE@begingroup \NROrg@ifthenelse }% \let\NROrg@begingroup\begingroup \def\NR@TE@begingroup{% \let\begingroup\NROrg@begingroup \begingroup \def\ref##1{\getrefbykeydefault{##1}{}{??}}% \def\pageref##1{\getrefbykeydefault{##1}{page}{0}}% }% }{} % \end{macrocode} % % \subsection{Compatibility with varioref} % % [This section was derived by Corey Minyard \texttt{minyard@acm.org} % from the varioref package] % \begin{macrocode} \AtBeginDocument{% \ifx\csname @@vpageref\endcsname\relax \else \def\@@vpageref#1[#2]#3{% \begingroup \csname @safe@activestrue\endcsname \edef\x{#3}% \@onelevel@sanitize\x \toks@{% \NR@@vpageref{#1}[{#2}]% }% \edef\x{\endgroup \the\toks@{\x}% }% \x }% \def\NR@@vpageref#1[#2]#3{% \leavevmode\unskip \global\advance\c@vrcnt\@ne \@ifundefined{r@\the\c@vrcnt @vr}% {\@namedef{r@\the\c@vrcnt @vr}{{??}{??}{}{}{}}}{}% \edef\@tempa{% \expandafter\expandafter\expandafter \@secondoffive\csname r@\the\c@vrcnt @vr\endcsname }% \@ifundefined{r@\the\c@vrcnt @xvr}% {\@namedef{r@\the\c@vrcnt @xvr}{{??}{??}{}{}{}}}{}% \edef\@tempb{% \expandafter\expandafter\expandafter \@secondoffive\csname r@\the\c@vrcnt @xvr\endcsname }% {% \let\@currentlabel\@empty \expandafter\label\expandafter{\the\c@vrcnt @xvr}% }% \ifx\@tempa\@tempb \else \vref@err{% \noexpand\vref at page boundary % \@tempb-\@tempa\space (may loop)% }% \fi \@ifundefined{r@#3}{\@namedef{r@#3}{{??}{??}{}{}{}}}{}% \edef\thevpagerefnum{% \expandafter\expandafter\expandafter \@secondoffive\csname r@#3\endcsname }% \ifx\vref@space\@undefined \space \else \vref@space \fi \ifx\@tempa\thevpagerefnum #1% \else #2% \is@pos@number\thevpagerefnum{% \is@pos@number\@tempa{% \@tempcnta\@tempa \advance\@tempcnta\@ne }{% \@tempcnta\maxdimen }% \ifnum \thevpagerefnum =\@tempcnta \ifodd\@tempcnta \if@twoside \reftextfaceafter \else \reftextafter \fi \else \reftextafter \fi \else \advance\@tempcnta-2 % \ifnum \thevpagerefnum =\@tempcnta \ifodd\@tempcnta \reftextbefore \else \if@twoside \reftextfacebefore \else \reftextbefore \fi \fi \else \reftextfaraway{#3}% \fi \fi }{% \reftextfaraway{#3}% }% \fi {% \let\@currentlabel\@empty \expandafter\label\expandafter{\the\c@vrcnt @vr}% }% }% \DeclareRobustCommand\vnameref[1]{% \unskip~\nameref{#1}% \@vpageref[\unskip]{#1}% }% \fi } % \end{macrocode} % \begin{macrocode} \@ifundefined{@Refstar}{% \def\T@Ref#1{% \NR@setref{#1}\NR@MakeUppercaseFirstOfFive{#1}% }% \def\NR@MakeUppercaseFirstOfFive#1#2#3#4#5{% \MakeUppercase#1% }% }{}% \AtBeginDocument{% \@ifpackageloaded{varioref}{% \@ifundefined{@Refstar}{% \DeclareRobustCommand*{\Ref}[1]{% \T@Ref{#1}% }% }{}% }{}% }% % \end{macrocode} % % \subsection{Support for listings} % % \begin{macrocode} \AtBeginDocument{% \@ifpackageloaded{listings}{% \let\NROrg@lst@MakeCaption\lst@MakeCaption \def\lst@MakeCaption{% \ifx\lst@label\@empty \else \let\@currentlabelname\lst@@caption \fi \NROrg@lst@MakeCaption }% }{}% } % \end{macrocode} % % \begin{macrocode} % % \end{macrocode} % % \Finale \endinput