% !TeX TS-program = lualatex \documentclass[a4paper,10pt,french]{article} \usepackage{fontspec} \usepackage[bookmarks=true,bookmarksopen=true,colorlinks=true,hyperfootnotes=false,filecolor=black,linkcolor=blue,urlcolor=blue,pdfauthor={Christian Tellechea},pdftitle={xstring},pdfsubject={package pour la manipulation de chaines de tokens},pdfkeywords={xstring,latex,tex,string,chaine},pdfcreator={LaTeX}]{hyperref} \usepackage[margin=1.9cm]{geometry} \usepackage{libertine} \usepackage[scale=0.85]{noto-mono} \usepackage[bottom]{footmisc} \usepackage{amsmath,amssymb,amsfonts,xcolor,eurosym,xspace,xstring,babel} \makeatletter \newcommand\make@car@active[1]{% \catcode`#1\active \begingroup \lccode`\~`#1\relax \lowercase{\endgroup\def~}% } \newcount\code@line \newbox\box@code \newbox\box@exec \newbox\box@bkgd \newcommand*\showcode{% \begingroup \parskip\z@ \parindent\z@ \code@line\z@ \@makeother\;\@makeother\!\@makeother\?\@makeother\: \medbreak \noindent \begingroup \let\do\@makeother\dospecials \make@car@active\ { }% \make@car@active\^^M{\par\leavevmode}% \make@car@active\^^I{\space\space\space\space}% \make@car@active\,{\leavevmode\kern\z@\string,}% \make@car@active\-{\leavevmode\kern\z@\string-}% \showcode@a } \newcommand\showcode@a[2][65]{% \def\@tempa##1#2{% \xdef\the@code{##1}% \endgroup \showcode@b{#1}{##1}}% \@tempa} \newcommand\showcode@b[2]{% \vskip0.5ex \fboxsep5pt \setbox\box@code=\hbox{% $\vcenter{ \hsize\dimexpr0.#1\linewidth-\fboxsep-\fboxrule\relax \parskip0pt \ttfamily \small \everypar{\llap{\global\advance\code@line\@ne\tiny\number\code@line\kern\dimexpr\fboxsep+3pt }}% \the@code\par }$% }% \setbox\box@exec=\hbox{% $\vcenter{ \hsize\dimexpr\linewidth-0.#1\linewidth-\fboxsep-\fboxrule\relax \small \newlinechar`\^^M\everyeof{\noexpand}% \scantokens{#2}% }$% }% \fbox{% \begingroup \color{black!15}% \setbox\box@bkgd=\hbox{% fond code \vrule width\wd\box@code height\dimexpr\max@dim{\ht\box@code}{\ht\box@exec}+\fboxsep\relax depth \dimexpr\max@dim{\dp\box@code}{\dp\box@exec}+\fboxsep\relax }% \wd\box@bkgd=0pt \ht\box@bkgd=0pt \dp\box@bkgd=0pt \kern-\fboxsep \box\box@bkgd \kern\fboxsep \endgroup \setbox\box@bkgd=\hbox{% réglure verticale \vrule width.5pt height\dimexpr\max@dim{\ht\box@code}{\ht\box@exec}+\fboxsep\relax depth \dimexpr\max@dim{\dp\box@code}{\dp\box@exec}+\fboxsep\relax }% \box\box@code \wd\box@bkgd=0pt \ht\box@bkgd=0pt \dp\box@bkgd=0pt \kern-\fboxsep \box\box@bkgd \kern\fboxsep \box\box@exec }% \medbreak \endgroup } \def\max@dim#1#2{\ifdim#1>#2 #1\else#2\fi} \definecolor{violet}{rgb}{0.66,0,0.66} \makeatother \newcommand\guill[1]{\og{}#1\fg{}} \newcommand\argu[1]{$\langle$\textit{#1}$\rangle$} \newcommand\ARGU[1]{\texttt{\color{black}\{}\argu{#1}\texttt{\color{black}\}}} \newcommand\arguC[1]{\texttt{\color{black}[}\argu{#1}\texttt{\color{black}]}} \newcommand\arguCC[2]{\texttt{\color{black}[}\argu{#1}\texttt{\color{black},}\argu{#2}\texttt{\color{black}]}} \newcommand\texte[1]{\texttt{text}${}_{#1}$} \newcommand\etoile{$\langle$\texttt{[*]}$\rangle$} \newenvironment{Conditions}[1][1cm]% {\begin{list}% {$\vartriangleright$}% {\setlength{\leftmargin}{#1} \setlength{\itemsep}{0pt} \setlength{\parsep}{0pt} \setlength{\topsep}{2ptplus3ptminus2pt} }}% {\end{list}} \newcommand\US{unité syntaxique\xspace} \newcommand\USs{unités syntaxiques\xspace} \newcommand\Xstring{\textsf{xstring}\xspace} \newcommand\styleexemple{\small\vskip.4\baselineskip\relax} \newcommand\styleexercice{\footnotesize} \newcommand\colorise{\color{violet}} \def\xstringfrenchdateaux#1/#2/#3\nil{\number#3\relax\ifnum#3=1 \noexpand\ier\fi\space \ifcase #2 \or janvier\or f\'evrier\or mars\or avril\or mai\or juin\or juillet\or aout\or septembre\or octobre\or novembre\or d\'ecembre\fi\space#1} \edef\xstringfrenchdate{\expandafter\xstringfrenchdateaux\xstringdate\nil} \begin{document} \setlength{\parindent}{0pt} \begin{titlepage} \null\par\vfill \begin{center} \begin{minipage}{0.75\linewidth} \begin{center} \Huge\bfseries \Xstring\par\vspace{5pt} \small v\xstringversion\par\vspace{25pt} \normalsize Manuel de l'utilisateur \end{center} \end{minipage} \end{center} \vspace{1cm} \begin{center} Christian {\sc Tellechea}\par\small \href{mailto:unbonpetit@netc.fr}{\nolinkurl{unbonpetit@netc.fr}}\par\vspace{5pt} \xstringfrenchdate \end{center} \vfill\hrulefill \begin{center} \begin{minipage}{0.85\linewidth} \noindent \hfill\textbf{\textit{Résumé}}\hfill{}\medskip\par Cette extension fournit un ensemble de macros manipulant des chaînes de tokens. Les macros, non développables, peuvent être utilisées de façon basique dans le traitement des chaînes de caractères mais peuvent également être utiles en programmation \TeX{} pour la manipulation des tokens, c'est-à-dire du code \TeX{}. \end{minipage} \end{center} \hrulefill\vfill{} \end{titlepage} \tableofcontents % \section{Présentation} Cette extension\footnote{L'extension ne nécessite pas \LaTeX{} et peut être compilée sous Plain $\varepsilon$-\TeX{}.} regroupe des macros et des tests non développables opérant sur des chaînes de tokens, un peu comme en disposent des langages dit \guill{évolués}. On y trouve les opérations habituelles sur les chaînes, comme par exemple : test si une chaîne en contient une autre, commence ou finit par une autre, test si une chaîne est un nombre entier ou décimal, extractions de sous-chaînes, calculs de position d'une sous-chaîne, calculs du nombre d'occurrences, etc.\medskip On appelle \guill{chaîne de tokens} une suite de tokens quelconques, sachant qu'aucune supposition n'a été faite quant à leur nature, mis à part que dans les chaînes de tokens, les accolades doivent être équilibrées et que les tokens de catcode 6 et 14 (habituellement \verb|%| et \verb|#|) n'y sont pas admis. Tous les autres tokens, y compris \verb|\par|, sont \emph{à priori} permis dans n'importe quel ordre, quel que soit le code qu'ils représentent.\medskip Les arguments contenant des chaînes de tokens sont lus par \Xstring \emph{\US par \US\footnote{Sauf cas particulier, une \US est un caractère lu dans le code à ces exceptions près : une séquence de contrôle est une \US, un groupe entre accolades est aussi une \US. Voir également page~\pageref{developpementarguments}.}}, ce qui revient à les lire caractère par caractère lorsque ceux-ci contiennent des tokens \guill{normaux}, c'est-à-dire dont les catcodes sont 10, 11 et 12. On peut également utiliser \Xstring à des fins de programmation en utilisant des arguments contenant des séquences de contrôle et des tokens dont les catcodes sont moins inoffensifs. Voir le chapitre sur le mode de lecture et de développement des arguments (page~\pageref{developpementarguments}), la commande \verb|\verbtocs| (page~\pageref{verbtocs}), la commande \verb|\scancs| (page~\pageref{scancs}).\medskip Certes d'autres packages manipulant les chaînes de caractères existent (par exemple \href{http://mirror.ctan.org/tex-archive/macros/latex/contrib/substr/}{\nolinkurl{substr}} et \href{http://mirror.ctan.org/tex-archive/macros/latex/contrib/stringstrings/}{\nolinkurl{stringstrings}}), mais outre des différences notables quant aux fonctionnalités, ils ne prennent pas en charge les occurrences des sous-chaînes et me semblent soit trop limités, soit trop difficiles à utiliser pour la programmation.\medskip Comme les macros manipulent des chaînes de tokens, il peut arriver aux utilisateurs avancés de rencontrer des problèmes de \guill{catcodes\footnote{Codes de catégories, en français.}} conduisant à des comportements inattendus. Ces effets indésirables peuvent être contrôlés. Consulter en particulier le chapitre sur les catcodes des arguments page~\pageref{macrosetoilees}. \section{Les macros} Pour bien comprendre les actions de chaque macro, envisageons tout d'abord le fonctionnement et la présentation des macros dans leur mode de fonctionnement le plus simple. Pas de problème de catcode ici, ni de tokens spéciaux et encore moins de séquence de contrôle dans les arguments : les arguments contiendront des caractères alphanumériques.\medskip Dans ce chapitre, la totalité des macros est présentée selon ce plan :\par\nobreak\smallskip\parindent3em \begin{itemize} \item la syntaxe complète\footnote{L'étoile optionnelle après le nom de la macro, et l'argument optionnel entre crochet venant en dernier seront expliqués plus tard. Voir page~\pageref{macrosetoilees} pour les macros étoilées et page~\pageref{argumentoptionnel} pour l'argument optionnel en dernière position.} ainsi que la valeur d'éventuels arguments optionnels; \item une brève description du fonctionnement; \item le fonctionnement sous certaines conditions particulières. Pour chaque conditions envisagée, le fonctionnement décrit est prioritaire sur celui (ceux) se trouvant au dessous de lui; \item enfin, quelques exemples sont donnés. J'ai essayé de les trouver les plus facilement compréhensibles et les plus représentatifs des situations rencontrées dans une utilisation normale\footnote{Pour une collection plus importante d'exemples, on peut aussi consulter le fichier de test.}. Si un doute est possible quant à la présence d'espaces dans le résultat, celui-ci sera délimité par des \guill{\textbar}, étant entendu qu'une chaîne vide est représentée par \guill{\textbar\textbar}. \end{itemize}\parindent0pt\smallskip \textbf{Important} : dans les macros qui suivent, sauf cas spécifié, un \argu{nombre} est un nombre entier, un compteur, ou le résultat d'une opération arithmétique effectuée à l'aide de la primitive \verb|\numexpr|. \subsection{Les tests} \subsubsection{\ttfamily\textbackslash IfSubStr} \verb|\IfSubStr|\etoile\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\ARGU{vrai}\ARGU{faux} \smallskip L'argument optionnel \argu{nombre} vaut 1 par défaut.\par\nobreak\smallskip Teste si \argu{chaine} contient au moins \argu{nombre} fois \argu{chaineA} et exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\medskip \begin{Conditions} \item Si \argu{nombre}${}\leqslant0$, exécute \argu{faux}; \item Si \argu{chaine} ou \argu{chaineA} est vide, exécute \argu{faux}. \end{Conditions} \showcode|\IfSubStr{xstring}{tri}{vrai}{faux}\par \IfSubStr{xstring}{a}{vrai}{faux}\par \IfSubStr{a bc def }{c d}{vrai}{faux}\par \IfSubStr{a bc def }{cd}{vrai}{faux}\par \IfSubStr[2]{1a2a3a}{a}{vrai}{faux}\par \IfSubStr[3]{1a2a3a}{a}{vrai}{faux}\par \IfSubStr[4]{1a2a3a}{a}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfSubStrBefore} \verb|\IfSubStrBefore|\etoile\arguCC{nombre1}{nombre2}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}\ARGU{vrai}\ARGU{faux} \smallskip Les arguments optionnels \argu{nombre1} et \argu{nombre2} valent 1 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, la macro teste si l'occurrence \no\argu{nombre1} de \argu{chaineA} se trouve à gauche de l'occurrence \no\argu{nombre2} de \argu{chaineB}. Exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\medskip \begin{Conditions} \item Si l'une des occurrences n'est pas trouvée, exécute \argu{faux}; \item Si l'un des arguments \argu{chaine}, \argu{chaineA} ou \argu{chaineB} est vide, exécute \argu{faux}; \item Si l'un au moins des deux arguments optionnels est négatif ou nul, exécute \argu{faux}. \end{Conditions} \showcode|\IfSubStrBefore{xstring}{st}{in}{vrai}{faux}\par \IfSubStrBefore{xstring}{ri}{s}{vrai}{faux}\par \IfSubStrBefore{LaTeX}{LaT}{TeX}{vrai}{faux}\par \IfSubStrBefore{a bc def }{ b}{ef}{vrai}{faux}\par \IfSubStrBefore{a bc def }{ab}{ef}{vrai}{faux}\par \IfSubStrBefore[2,1]{b1b2b3}{b}{2}{vrai}{faux}\par \IfSubStrBefore[3,1]{b1b2b3}{b}{2}{vrai}{faux}\par \IfSubStrBefore[2,2]{baobab}{a}{b}{vrai}{faux}\par \IfSubStrBefore[2,3]{baobab}{a}{b}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfSubStrBehind} \verb|\IfSubStrBehind|\etoile\arguCC{nombre1}{nombre2}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}\ARGU{vrai}\ARGU{faux} \smallskip Les arguments optionnels \argu{nombre1} et \argu{nombre2} valent 1 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, la macro teste si l'occurrence \no\argu{nombre1} de \argu{chaineA} se trouve après l'occurrence \\\no\argu{nombre2} de \argu{chaineB}. Exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\medskip \begin{Conditions} \item Si l'une des occurrences n'est pas trouvée, exécute \argu{faux}; \item Si l'un des arguments \argu{chaine}, \argu{chaineA} ou \argu{chaineB} est vide, exécute \argu{faux}; \item Si l'un au moins des deux arguments optionnels est négatif ou nul, exécute \argu{faux}. \end{Conditions} \showcode|\IfSubStrBehind{xstring}{ri}{xs}{vrai}{faux}\par \IfSubStrBehind{xstring}{s}{i}{vrai}{faux}\par \IfSubStrBehind{LaTeX}{TeX}{LaT}{vrai}{faux}\par \IfSubStrBehind{a bc def }{ d}{a}{vrai}{faux}\par \IfSubStrBehind{a bc def }{cd}{a b}{vrai}{faux}\par \IfSubStrBehind[2,1]{b1b2b3}{b}{2}{vrai}{faux}\par \IfSubStrBehind[3,1]{b1b2b3}{b}{2}{vrai}{faux}\par \IfSubStrBehind[2,2]{baobab}{b}{a}{vrai}{faux}\par \IfSubStrBehind[2,3]{baobab}{b}{a}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfBeginWith} \verb|\IfBeginWith|\etoile\ARGU{chaine}\ARGU{chaineA}\ARGU{vrai}\ARGU{faux} \smallskip Teste si \argu{chaine} commence par \argu{chaineA}, et exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\medskip \begin{Conditions} \item Si \argu{chaine} ou \argu{chaineA} est vide, exécute \argu{faux}. \end{Conditions} \showcode|\IfBeginWith{xstring}{xst}{vrai}{faux}\par \IfBeginWith{LaTeX}{a}{vrai}{faux}\par \IfBeginWith{a bc def }{a b}{vrai}{faux}\par \IfBeginWith{a bc def }{ab}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfEndWith} \verb|\IfEndWith|\etoile\ARGU{chaine}\ARGU{chaineA}\ARGU{vrai}\ARGU{faux} \smallskip Teste si \argu{chaine} se termine par \argu{chaineA}, et exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\medskip \begin{Conditions} \item Si \argu{chaine} ou \argu{chaineA} est vide, exécute \argu{faux}. \end{Conditions} \showcode|\IfEndWith{xstring}{ring}{vrai}{faux}\par \IfEndWith{LaTeX}{a}{vrai}{faux}\par \IfEndWith{a bc def }{ef }{vrai}{faux}\par \IfEndWith{a bc def }{ef}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfInteger} \verb|\IfInteger|\ARGU{nombre}\ARGU{vrai}\ARGU{faux} \smallskip Teste si \argu{nombre} est un nombre entier relatif (c'est-à-dire dont la partie décimale est absente ou constituée d'une suite de 0), et exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\par\nobreak\smallskip Si le test est faux pour cause de caractères non autorisés, \verb|\afterinteger| contient la partie illégale de \argu{nombre}.\medskip \showcode|\IfInteger{13}{vrai}{faux}\par \IfInteger{-219}{vrai}{faux}\par \IfInteger{+9}{vrai}{faux}\par \IfInteger{3.14}{vrai}{faux}\par \IfInteger{8.0}{vrai}{faux}\par \IfInteger{0}{vrai}{faux}\par \IfInteger{49a}{vrai}{faux}\par \IfInteger{+}{vrai}{faux}\par \IfInteger{-}{vrai}{faux}\par \IfInteger{0000}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfDecimal}\label{ifdecimal} \verb|\IfDecimal|\ARGU{nombre}\ARGU{vrai}\ARGU{faux} \smallskip Teste si \argu{nombre} est un nombre décimal, et exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\par\nobreak\smallskip Les macros \verb|\integerpart| et \verb|\decimalpart| contiennent les parties entières et décimales de \argu{nombre}. Si le test est faux pour cause de caractères non autorisés, \verb|\afterdecimal| contient la partie illégale de \argu{nombre}, alors que si le test est faux parce que la partie décimale après le séparateur décimal est vide, elle contient \guill{X}.\medskip \begin{Conditions} \item Le séparateur décimal peut être un point ou une virgule; \item Si le \argu{nombre} commence par un séparateur décimal, le test est vrai (lignes 4 et 5) et la partie entière est considérée comme étant 0; \item Si le \argu{nombre} se termine par un séparateur décimal, le test est faux (lignes 9 et 10). \end{Conditions} \showcode|\IfDecimal{3.14}{vrai}{faux}\par \IfDecimal{3,14}{vrai}{faux}\par \IfDecimal{-0.5}{vrai}{faux}\par \IfDecimal{.7}{vrai}{faux}\par \IfDecimal{,9}{vrai}{faux}\par \IfDecimal{1..2}{vrai}{faux}\par \IfDecimal{+6}{vrai}{faux}\par \IfDecimal{-15}{vrai}{faux}\par \IfDecimal{1.}{vrai}{faux}\par \IfDecimal{2,}{vrai}{faux}\par \IfDecimal{.}{vrai}{faux}\par \IfDecimal{,}{vrai}{faux}\par \IfDecimal{+}{vrai}{faux}\par \IfDecimal{-}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfStrEq} \verb|\IfStrEq|\etoile\ARGU{chaineA}\ARGU{chaineB}\ARGU{vrai}\ARGU{faux} \smallskip Teste si les chaînes \argu{chaineA} et \argu{chaineB} sont égales, c'est-à-dire si elles contiennent successivement les mêmes \US dans le même ordre. Exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\smallskip \showcode|\IfStrEq{a1b2c3}{a1b2c3}{vrai}{faux}\par \IfStrEq{abcdef}{abcd}{vrai}{faux}\par \IfStrEq{abc}{abcdef}{vrai}{faux}\par \IfStrEq{3,14}{3,14}{vrai}{faux}\par \IfStrEq{12.34}{12.340}{vrai}{faux}\par \IfStrEq{abc}{}{vrai}{faux}\par \IfStrEq{}{abc}{vrai}{faux}\par \IfStrEq{}{}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfEq} \verb|\IfEq|\etoile\ARGU{chaineA}\ARGU{chaineB}\ARGU{vrai}\ARGU{faux} \smallskip Teste si les chaînes \argu{chaineA} et \argu{chaineB} sont égales, \emph{sauf} si \argu{chaineA} et \argu{chaineB} contiennent des nombres, auquel cas la macro teste si les nombres sont égaux. Exécute \argu{vrai} dans l'affirmative, et \argu{faux} dans le cas contraire.\smallskip \begin{Conditions} \item La définition de \emph{nombre} est celle évoquée dans la macro \verb|\IfDecimal| (voir page~\pageref{ifdecimal}), et donc : \item Les signes \guill{+} sont facultatifs; \item Le séparateur décimal peut être indifféremment la virgule ou le point. \end{Conditions} \showcode|\IfEq{a1b2c3}{a1b2c3}{vrai}{faux}\par \IfEq{abcdef}{ab}{vrai}{faux}\par \IfEq{ab}{abcdef}{vrai}{faux}\par \IfEq{12.34}{12,34}{vrai}{faux}\par \IfEq{+12.34}{12.340}{vrai}{faux}\par \IfEq{10}{+10}{vrai}{faux}\par \IfEq{-10}{10}{vrai}{faux}\par \IfEq{+0,5}{,5}{vrai}{faux}\par \IfEq{1.001}{1.01}{vrai}{faux}\par \IfEq{3*4+2}{14}{vrai}{faux}\par \IfEq{\number\numexpr3*4+2}{14}{vrai}{faux}\par \IfEq{0}{-0.0}{vrai}{faux}\par \IfEq{}{}{vrai}{faux}| \subsubsection{\ttfamily\textbackslash IfStrEqCase} \begin{minipage}{\textwidth} \verb|\IfStrEqCase|\etoile\ARGU{chaine}\verb|{%|\par \qquad\qquad\ARGU{chaine1}\ARGU{code1}\verb|%|\par \qquad\qquad\ARGU{chaine2}\ARGU{code2}\verb|%|\par \qquad\qquad\verb|etc...|\par \qquad\qquad\ARGU{chaineN}\ARGU{codeN}\verb|}|\arguC{code alternatif} \end{minipage} \smallskip Teste successivement si \argu{chaine} est égale à \argu{chaine1}, \argu{chaine2}, etc. La comparaison se fait au sens de \verb|\IfStrEq| (voir paragraphes précédents). Si un test est positif, le code correspondant est exécuté et la macro se termine. Si tous les tests sont négatifs, le code optionnel \argu{code alternatif} est exécuté s'il est présent.\smallskip \showcode-\IfStrEqCase{b}{{a}{AA}{b}{BB}{c}{CC}}\par |\IfStrEqCase{abc}{{a}{AA}{b}{BB}{c}{CC}}|\par \IfStrEqCase{c}{{a}{AA}{b}{BB}{c}{CC}}[autre]\par \IfStrEqCase{d}{{a}{AA}{b}{BB}{c}{CC}}[autre]\par \IfStrEqCase{+3}{{1}{un}{2}{deux}{3}{trois}}[autre]\par \IfStrEqCase{0.5}{{0}{zero}{.5}{demi}{1}{un}}[autre]- \subsubsection{\ttfamily\textbackslash IfEqCase} \begin{minipage}{\textwidth} \verb|\IfEqCase|\etoile\ARGU{chaine}\verb|{%|\par \qquad\qquad\ARGU{chaine1}\ARGU{code1}\verb|%|\par \qquad\qquad\ARGU{chaine2}\ARGU{code2}\verb|%|\par \qquad\qquad\verb|etc...|\par \qquad\qquad\ARGU{chaineN}\ARGU{codeN}\verb|}|\arguC{code alternatif} \end{minipage} \smallskip Teste successivement si \argu{chaine} est égale à \argu{chaine1}, \argu{chaine2}, etc. La comparaison se fait au sens de \verb|\IfEq| (voir paragraphes précédents). Si un test est positif, le code correspondant est exécuté et la macro se termine. Si tous les tests sont négatifs, le code optionnel \argu{code alternatif} est exécuté s'il est présent.\smallskip \showcode-\IfEqCase{b}{{a}{AA}{b}{BB}{c}{CC}}\par |\IfEqCase{abc}{{a}{AA}{b}{BB}{c}{CC}}|\par \IfEqCase{c}{{a}{AA}{b}{BB}{c}{CC}}[autre]\par \IfEqCase{d}{{a}{AA}{b}{BB}{c}{CC}}[autre]\par \IfEqCase{+3}{{1}{un}{2}{deux}{3}{trois}}[autre]\par \IfEqCase{0.5}{{0}{zero}{.5}{demi}{1}{un}}[autre]- \subsection{Les macros renvoyant une chaîne} \subsubsection{\ttfamily\textbackslash StrBefore} \verb|\StrBefore|\etoile\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\arguC{nom} \smallskip L'argument optionnel \argu{nombre} vaut 1 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, renvoie ce qui se trouve avant l'occurrence \no\argu{nombre} de \argu{chaineA}.\medskip \begin{Conditions} \item Si \argu{chaine} ou \argu{chaineA} est vide, une chaîne vide est renvoyée; \item Si \argu{nombre}${}<1$ alors, la macro se comporte comme si \argu{nombre}${}=1$; \item Si l'occurrence n'est pas trouvée, une chaîne vide est renvoyée. \end{Conditions} \showcode-\StrBefore{xstring}{tri}\par \StrBefore{LaTeX}{e}\par |\StrBefore{LaTeX}{p}|\par |\StrBefore{LaTeX}{L}|\par |\StrBefore{a bc def }{def}|\par |\StrBefore{a bc def }{cd}|\par \StrBefore[1]{1b2b3}{b}\par \StrBefore[2]{1b2b3}{b}- \subsubsection{\ttfamily\textbackslash StrBehind} \verb|\StrBehind|\etoile\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\arguC{nom} \smallskip L'argument optionnel \argu{nombre} vaut 1 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, renvoie ce qui se trouve après l'occurrence \no\argu{nombre} de \argu{chaineA}.\medskip \begin{Conditions} \item Si \argu{chaine} ou \argu{chaineA} est vide, une chaîne vide est renvoyée; \item Si \argu{nombre}${}<1$ alors, la macro se comporte comme si \argu{nombre}${}=1$; \item Si l'occurrence n'est pas trouvée, une chaîne vide est renvoyée. \end{Conditions} \showcode-\StrBehind{xstring}{tri}\par \StrBehind{LaTeX}{e}\par |\StrBehind{LaTeX}{p}|\par |\StrBehind{LaTeX}{X}|\par |\StrBehind{a bc def }{bc}|\par |\StrBehind{a bc def }{cd}|\par \StrBehind[1]{1b2b3}{b}\par \StrBehind[2]{1b2b3}{b}\par |\StrBehind[3]{1b2b3}{b}- \subsubsection{\ttfamily\textbackslash StrCut} Voici la syntaxe de cette macro :\par\nobreak\smallskip \verb|\StrCut|\etoile\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\ARGU{macroA}\ARGU{macroB} \smallskip L'argument optionnel \argu{nombre} vaut 1 par défaut.\par\nobreak\smallskip La \argu{chaine} est coupée en deux chaînes à l'occurrence \no\arguC{nombre} de la \ARGU{chaineA}. Ce qui se trouve à gauche de cette occurrence est assigné à la séquence de contrôle \argu{macroA} et ce qui se trouve à droite à \argu{macroB}. Cette macro renvoie \emph{deux chaînes} et donc \textbf{n'affiche rien}. Par conséquent, elle ne dispose pas de l'argument optionnel en dernière position.\medskip \begin{Conditions} \item Si \argu{chaine} ou \argu{chaineA} est vide, \argu{macroA} et\argu{macroB} seront vides ; \item Si \argu{nombre}${}<1$ alors, la macro se comporte comme si \argu{nombre}${}=1$; \item Si l'occurrence n'est pas trouvée, \argu{macroA} reçoit la totalité de \argu{chaine} tandis que \argu{macroB} est vide. \end{Conditions} \showcode+\def\seprouge{{\color{red}|}} \StrCut{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut[2]{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut[3]{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut[4]{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut[5]{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut[6]{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut[-4]{abracadabra}{a}\csA\csB |\csA\seprouge\csB|\par \StrCut{abracadabra}{brac}\csA\csB |\csA\seprouge\csB|\par \StrCut{abracadabra}{foo}\csA\csB |\csA\seprouge\csB|\par \StrCut{abracadabra}{}\csA\csB |\csA\seprouge\csB|+ \subsubsection{\ttfamily\textbackslash StrBetween} \verb|\StrBetween|\etoile\arguCC{nombre1}{nombre2}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}\arguC{nom} \smallskip Les arguments optionnels \argu{nombre1} et \argu{nombre2} valent 1 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, renvoie ce qui se trouve entre\footnote{Au sens strict, c'est-à-dire \emph{sans} les chaînes frontière} les occurrences \no\argu{nombre1} de \argu{chaineA} et \no \argu{nombre2} de \argu{chaineB}.\medskip \begin{Conditions} \item Si les occurrences ne sont pas dans l'ordre (\argu{chaineA} \emph{puis} \argu{chaineB}) dans \argu{chaine}, une chaîne vide est renvoyée; \item Si l'une des 2 occurrences n'existe pas dans \argu{chaine}, une chaîne vide est renvoyée; \item Si l'un des arguments optionnels \argu{nombre1} ou \argu{nombre2} est négatif ou nul, une chaîne vide est renvoyée. \end{Conditions} \showcode+\StrBetween{xstring}{xs}{ng}\par |\StrBetween{xstring}{i}{n}|\par |\StrBetween{xstring}{a}{tring}|\par |\StrBetween{a bc def }{a}{d}|\par |\StrBetween{a bc def }{a }{f}|\par \StrBetween{a1b1a2b2a3b3}{a}{b}\par \StrBetween[2,3]{a1b1a2b2a3b3}{a}{b}\par \StrBetween[1,3]{a1b1a2b2a3b3}{a}{b}\par |\StrBetween[3,1]{a1b1a2b2a3b3}{a}{b}|\par \StrBetween[3,2]{abracadabra}{a}{bra}+ \subsubsection{\ttfamily\textbackslash StrSubstitute} \verb|\StrSubstitute|\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}\arguC{nom} \smallskip L'argument optionnel \argu{nombre} vaut 0 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, la macro remplace les \argu{nombre} premières occurrences de \argu{chaineA} par \argu{chaineB}, sauf si \argu{nombre}${}=0$ auquel cas, \emph{toutes} les occurrences sont remplacées. \begin{Conditions} \item Si \argu{chaine} est vide, une chaîne vide est renvoyée; \item Si \argu{chaineA} est vide ou n'existe pas dans \argu{chaine}, la macro est sans effet; \item Si \argu{nombre} est supérieur au nombre d'occurrences de \argu{chaineA}, alors toutes les occurrences sont remplacées; \item Si \argu{nombre}${}<0$ alors la macro se comporte comme si \argu{nombre}${}=0$; \item Si \argu{chaineB} est vide, alors les occurrences de \argu{chaineA}, si elles existent, sont supprimées. \end{Conditions} \showcode|\StrSubstitute{xstring}{i}{a}\par \StrSubstitute{abracadabra}{a}{o}\par \StrSubstitute{abracadabra}{br}{TeX}\par \StrSubstitute{LaTeX}{m}{n}\par \StrSubstitute{a bc def }{ }{M}\par \StrSubstitute{a bc def }{ab}{AB}\par \StrSubstitute[1]{a1a2a3}{a}{B}\par \StrSubstitute[2]{a1a2a3}{a}{B}\par \StrSubstitute[3]{a1a2a3}{a}{B}\par \StrSubstitute[4]{a1a2a3}{a}{B}| \subsubsection{\ttfamily\textbackslash StrDel} \verb|\StrDel|\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\arguC{nom} \smallskip L'argument optionnel \argu{nombre} vaut 0 par défaut.\par\nobreak\smallskip Supprime les \argu{nombre} premières occurrences de \argu{chaineA} dans \argu{chaine}, sauf si \argu{nombre}${}=0$ auquel cas, \emph{toutes} les occurrences sont supprimées.\medskip \begin{Conditions} \item Si \argu{chaine} est vide, une chaîne vide est renvoyée; \item Si \argu{chaineA} est vide ou n'existe pas dans \argu{chaine}, la macro est sans effet; \item Si \argu{nombre} est supérieur au nombre d'occurrences de \argu{chaineA}, alors toutes les occurrences sont supprimées; \item Si \argu{nombre}${}<0$ alors la macro se comporte comme si \argu{nombre}${}=0$; \end{Conditions} \showcode+\StrDel{abracadabra}{a}\par \StrDel[1]{abracadabra}{a}\par \StrDel[4]{abracadabra}{a}\par \StrDel[9]{abracadabra}{a}\par \StrDel{a bc def }{ }\par |\StrDel{a bc def }{def}|+ \subsubsection{\ttfamily\textbackslash StrGobbleLeft} \verb|\StrGobbleLeft|\ARGU{chaine}\ARGU{nombre}\arguC{nom} \smallskip Dans \argu{chaine}, enlève les \argu{nombre} premieres \USs de gauche.\medskip \begin{Conditions} \item Si \argu{chaine} est vide, renvoie une chaîne vide; \item Si \argu{nombre}${}\leqslant0$, aucune \US n'est supprimée; \item Si \argu{nombre}${}\geqslant{}$\argu{longueurChaine}, toutes les \USs sont supprimées. \end{Conditions} \showcode+\StrGobbleLeft{xstring}{2}\par |\StrGobbleLeft{xstring}{9}|\par \StrGobbleLeft{LaTeX}{4}\par \StrGobbleLeft{LaTeX}{-2}\par |\StrGobbleLeft{a bc def }{4}|+ \subsubsection{\ttfamily\textbackslash StrLeft} \verb|\StrLeft|\ARGU{chaine}\ARGU{nombre}\arguC{nom} \smallskip Dans \argu{chaine}, renvoie la sous-chaîne de gauche de longueur \argu{nombre}.\medskip \begin{Conditions} \item Si \argu{chaine} est vide, renvoie une chaîne vide; \item Si \argu{nombre}${}\leqslant0$, aucune \US n'est retournée; \item Si \argu{nombre}${}\geqslant{}$\argu{longueurChaine}, toutes les \USs sont retournées. \end{Conditions} \showcode+\StrLeft{xstring}{2}\par \StrLeft{xstring}{9}\par \StrLeft{LaTeX}{4}\par |\StrLeft{LaTeX}{-2}|\par |\StrLeft{a bc def }{5}|+ \subsubsection{\ttfamily\textbackslash StrGobbleRight} \verb|\StrGobbleRight|\ARGU{chaine}\ARGU{nombre}\arguC{nom} \smallskip Agit comme \verb|\StrGobbleLeft|, mais enlève les \USs à droite de \argu{chaine}. \showcode+\StrGobbleRight{xstring}{2}\par |\StrGobbleRight{xstring}{9}|\par \StrGobbleRight{LaTeX}{4}\par |\StrGobbleRight{LaTeX}{-2}|\par |\StrGobbleRight{a bc def }{4}|+ \subsubsection{\ttfamily\textbackslash StrRight} \verb|\StrRight|\ARGU{chaine}\ARGU{nombre}\arguC{nom} \smallskip Agit comme \verb|\StrLeft|, mais renvoie les \USs à la droite de \argu{chaine}. \showcode+\StrRight{xstring}{2}\par \StrRight{xstring}{9}\par \StrRight{LaTeX}{4}\par |\StrRight{LaTeX}{-2}|\par \StrRight{a bc def }{5}+ \subsubsection{\ttfamily\textbackslash StrChar} \verb|\StrChar|\ARGU{chaine}\ARGU{nombre}\arguC{nom} \smallskip Renvoie l'\US à la position \argu{nombre} dans la chaîne \argu{chaine}.\medskip \begin{Conditions} \item Si \argu{chaine} est vide, aucune \US n'est renvoyée; \item Si \argu{nombre}${}\leqslant0$ ou si \argu{nombre}${}>{}$\argu{longueurChaine}, aucune \US n'est renvoyée. \end{Conditions} \showcode+\StrChar{xstring}{4}\par |\StrChar{xstring}{9}|\par |\StrChar{xstring}{-5}|\par \StrChar{a bc def }{6}+ \subsubsection{\ttfamily\textbackslash StrMid} \verb|\StrMid|\ARGU{chaine}\ARGU{nombre1}\ARGU{nombre2}\arguC{nom} \smallskip Dans \argu{chaine}, renvoie la sous chaîne se trouvant entre\footnote{Au sens large, c'est-à-dire que les chaînes \guill{frontière} sont renvoyés.} les positions \argu{nombre1} et \argu{nombre2}.\medskip \begin{Conditions} \item Si \argu{chaine} est vide, une chaîne vide est renvoyée; \item Si \argu{nombre1}${}>{}$\argu{nombre2}, alors rien n'est renvoyé; \item Si \argu{nombre1}${}<1$ et \argu{nombre2}${}<1$ alors rien n'est renvoyé; \item Si \argu{nombre1}${}>{}$\argu{longueurChaine} et \argu{nombre2}${}>{}$\argu{longueurChaine}, alors rien n'est renvoyé; \item Si \argu{nombre1}${}<1$, alors la macro se comporte comme si \argu{nombre1}${}=1$; \item Si \argu{nombre2}${}>{}$\argu{longueurChaine}, alors la macro se comporte comme si \argu{nombre2}${}={}$\argu{longueurChaine}. \end{Conditions} \showcode+\StrMid{xstring}{2}{5}\par \StrMid{xstring}{-4}{2}\par |\StrMid{xstring}{5}{1}|\par \StrMid{xstring}{6}{15}\par \StrMid{xstring}{3}{3}\par |\StrMid{a bc def }{2}{7}|+ \subsection{Les macros renvoyant des nombres} \subsubsection{\ttfamily\textbackslash StrLen} \verb|\StrLen|\ARGU{chaine}\arguC{nom} \smallskip Renvoie la longueur de \argu{chaine}. \showcode+\StrLen{xstring}\par \StrLen{A}\par \StrLen{a bc def }+ \subsubsection{\ttfamily\textbackslash StrCount} \verb|\StrCount|\ARGU{chaine}\ARGU{chaineA}\arguC{nom} \smallskip Compte combien de fois \argu{chaineA} est contenue dans \argu{chaine}.\par\medskip \begin{Conditions} \item Si l'un au moins des arguments \argu{chaine} ou \argu{chaineA} est vide, la macro renvoie 0. \end{Conditions} \showcode+\StrCount{abracadabra}{a}\par \StrCount{abracadabra}{bra}\par \StrCount{abracadabra}{tic}\par \StrCount{aaaaaa}{aa}+ \subsubsection{\ttfamily\textbackslash StrPosition} \verb|\StrPosition|\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\arguC{nom} \smallskip L'argument optionnel \argu{nombre} vaut 1 par défaut.\par\nobreak\smallskip Dans \argu{chaine}, renvoie la position de l'occurrence \no\argu{nombre} de \argu{chaineA}.\medskip \begin{Conditions} \item Si \argu{nombre} est supérieur au nombre d'occurrences de \argu{chaineA}, alors la macro renvoie 0. \item Si \argu{chaine} ne contient pas \argu{chaineA}, alors la macro renvoie 0. \end{Conditions} \medskip \showcode+\StrPosition{xstring}{ring}\par \StrPosition[4]{abracadabra}{a}\par \StrPosition[2]{abracadabra}{bra}\par \StrPosition[9]{abracadabra}{a}\par \StrPosition{abracadabra}{z}\par \StrPosition{a bc def }{d}\par \StrPosition[3]{aaaaaa}{aa}+ \subsubsection{\ttfamily\textbackslash StrCompare} \verb|\StrCompare|\etoile\ARGU{chaineA}\ARGU{chaineB}\arguC{nom} \smallskip Cette macro peut fonctionner avec 2 tolérances, la tolérance \guill{normale} qui est sélectionnée par défaut et la tolérance \guill{stricte}. \begin{itemize} \item La tolérance normale, activée par la commande \verb|\comparenormal|.\par La macro compare successivement les \US de gauche à droite des chaînes \argu{chaineA} et \argu{chaineB} jusqu'à ce qu'une différence apparaisse ou que la fin de la plus courte chaîne soit atteinte. Si aucune différence n'est trouvée, la macro renvoie 0. Sinon, la position de la 1\iere{} différence est renvoyée. \item La tolérance stricte, activée par la commande \verb|\comparestrict|.\par La macro compare les 2 chaînes. Si elles sont égales, elle renvoie 0 sinon la position de la 1\iere{} différence est renvoyée. \end{itemize} \medskip On peut également mémoriser le mode de comparaison en cours avec \verb|\savecomparemode|, le modifier par la suite et revenir à la situation lors de la sauvegarde avec \verb|\restorecomparemode|.\medskip Exemples en tolérance normale :\par\nobreak \showcode+ \comparenormal \StrCompare{abcd}{abcd}\par \StrCompare{abcd}{abc}\par \StrCompare{abc}{abcd}\par \StrCompare{a b c}{abc}\par \StrCompare{aaa}{baaa}\par \StrCompare{abc}{xyz}\par \StrCompare{123456}{123457}\par \StrCompare{abc}{}+ Exemples en tolérance stricte :\par\nobreak \showcode+\comparestrict \StrCompare{abcd}{abcd}\par \StrCompare{abcd}{abc}\par \StrCompare{abc}{abcd}\par \StrCompare{a b c}{abc}\par \StrCompare{aaa}{baaa}\par \StrCompare{abc}{xyz}\par \StrCompare{123456}{123457}\par \StrCompare{abc}{}+ \section{Modes de fonctionnement} \subsection{Développement des arguments} \label{devarg} \subsubsection{Les macros {\ttfamily \textbackslash fullexpandarg}, {\ttfamily \textbackslash expandarg} et {\ttfamily \textbackslash noexpandarg}} La macro \verb|\fullexpandarg| est appelée par défaut, ce qui fait que certains arguments (en \textcolor{violet}{violet} dans la liste ci dessous) transmis aux macros sont développés le plus possible (pour cela, un \verb|\edef| est utilisé). Ce mode de développement maximal permet dans la plupart des cas d'éviter d'utiliser des chaînes d'\verb|\expandafter|. Le code en est souvent allégé. On peut interdire le développement de ces arguments (et ainsi revenir au comportement normal de \TeX) en invoquant \verb|\noexpandarg| ou \verb|\normalexpandarg| qui sont synonymes.\medskip Il existe enfin un autre mode de développement de ces arguments que l'on appelle avec \verb|\expandarg|. Dans ce cas, le \textbf{premier token} de ces arguments est développé \emph{une fois} avant que la macro ne soit appelée. Si l'argument contient plus d'un token, les tokens qui suivent le premier ne sont pas développés (on peut contourner cette volontaire limitation et utiliser la macro \verb|\StrExpand|, voir page~\pageref{scancs}).\medskip Les commandes \verb|\fullexpandarg|, \verb|\noexpandarg|, \verb|\normalexpandarg| et \verb|\expandarg| peuvent être appelées à tout moment dans le code et fonctionnent comme des bascules. On peut rendre leur portée locale dans un groupe.\medskip On peut également mémoriser le mode de développement en cours avec \verb|\saveexpandmode|, le modifier par la suite et revenir à la situation lors de la sauvegarde avec \verb|\restoreexpandmode|.\medskip Dans la liste ci-dessous, on peut voir en \textcolor{violet}{violet} quels arguments sont soumis à l'éventuel développement pour chaque macro vue dans le chapitre précedent :\smallskip \parindent3em \begin{itemize} \item\verb|\IfSubStr|\etoile{\colorise\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfSubStrBefore|\etoile{\colorise\arguCC{nombre1}{nombre2}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfSubStrBehind|\etoile{\colorise\arguCC{nombre1}{nombre2}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfBeginWith|\etoile{\colorise\ARGU{chaine}\ARGU{chaineA}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfEndWith|\etoile{\colorise\ARGU{chaine}\ARGU{chaineA}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfInteger|{\colorise\ARGU{nombre}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfDecimal|{\colorise\ARGU{nombre}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfStrEq|\etoile{\colorise\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfEq|\etoile{\colorise\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfStrEqCase|\etoile{\colorise\ARGU{chaine}\texttt{\color{black}\{}\colorise\ARGU{chaine1}}\ARGU{code1}\par \setbox0=\hbox{{\ttfamily\string\IfStrEqCase}\etoile\ARGU{chaine}{\ttfamily\{}} \hskip\wd0{\colorise\ARGU{chaine2}}\ARGU{code2}\par \hskip\wd0\ \ldots\par \hskip\wd0{\colorise\ARGU{chaine $n$}}\ARGU{code $n$}\verb|}|\arguC{code alternatif} \item\verb|\IfEqCase|\etoile{\colorise\ARGU{chaine}\texttt{\color{black}\{}\colorise\ARGU{chaine1}}\ARGU{code1}\par \setbox0=\hbox{{\ttfamily\string\IfEqCase}\etoile\ARGU{chaine}{\ttfamily\{}} \hskip\wd0{\colorise\ARGU{chaine2}}\ARGU{code2}\par \hskip\wd0\ \ldots\par \hskip\wd0{\colorise\ARGU{chaine $n$}}\ARGU{code $n$}\verb|}|\arguC{code alternatif} \item\verb|\StrBefore|\etoile{\colorise\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrBehind|\etoile{\colorise\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrBetween|\etoile{\colorise\arguCC{nombre1}{nombre2}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\arguC{nom} \item\verb|\StrSubstitute|{\colorise\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\arguC{nom} \item\verb|\StrDel|{\colorise\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrSplit|{\colorise\ARGU{chaine}\ARGU{nombre}}\ARGU{chaineA}\ARGU{chaineB}\qquad{\small(voir page~\pageref{StrSplit} pour la macro \verb|StrSplit|)} \item\verb|\StrGobbleLeft|{\colorise\ARGU{chaine}\ARGU{nombre}}\arguC{nom} \item\verb|\StrLeft|{\colorise\ARGU{chaine}\ARGU{nombre}}\arguC{nom} \item\verb|\StrGobbleRight|{\colorise\ARGU{chaine}\ARGU{nombre}}\arguC{nom} \item\verb|\StrRight|{\colorise\ARGU{chaine}\ARGU{nombre}}\arguC{nom} \item\verb|\StrChar|{\colorise\ARGU{chaine}\ARGU{nombre}}\arguC{nom} \item\verb|\StrMid|{\colorise\ARGU{chaine}\ARGU{nombre1}\ARGU{nombre2}}\arguC{nom} \item\verb|\StrLen|{\colorise\ARGU{chaine}}\arguC{nom} \item\verb|\StrCount|{\colorise\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrPosition|{\colorise\arguC{nombre}\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrCompare|{\colorise\ARGU{chaineA}\ARGU{chaineB}}\arguC{nom} \end{itemize} \parindent0pt \subsubsection{Caractères et lexèmes autorisés dans les arguments} Tout d'abord, quelque soit le mode de développement choisi, \textbf{les tokens de catcode 6 et 14 (habituellement {\ttfamily\#} et {\ttfamily\%}) sont interdits dans tous les arguments}\footnote{Le token {\ttfamily\#} sera peut-être autorisé dans une future version !}.\bigskip Lorsque le mode \verb|\fullexpandarg| est activé, les arguments sont évalués à l'aide de la primitive \verb|\edef| avant d'être transmis aux macros. Par conséquent, sont autorisés dans les arguments : \parindent3em \begin{itemize} \item les lettres, majuscules, minuscules, accentuées\footnote{Pour pouvoir utiliser des lettres accentuées de façon fiable, il est nécessaire de charger le packages \texttt{\string\fontenc} avec l'option \texttt{[T1]}, ainsi que \texttt{\string\inputenc} avec l'option correspondant au codage du fichier tex.} ou non, les chiffres, les espaces\footnote{Selon la règle \TeX{}iènne, plusieurs espaces consécutifs n'en font qu'un.} ainsi que tout autre token de catcode 10, 11 ou 12 (signes de ponctuation, signes opératoires mathématiques, parenthèses, crochets, etc); \item les tokens de catcode 1 à 4, qui sont habituellement : \og\verb|{|\fg\quad\og\verb|}|\fg\footnote{Attention : les accolades \textbf{doivent} être équilibrées dans les arguments !}\quad\og\verb|$|\fg\quad\og\verb|&|\fg \item les tokens de catcode 7 et 8, qui sont habituellement : \og\verb|^|\fg\quad\og\verb|_|\fg \item toute séquence de contrôle si elle est purement développable\footnote{C'est-à-dire qu'elle peut être mise à l'intérieur d'un {\ttfamily\string\edef }.} et dont le développement maximal donne des caractères autorisés. \end{itemize} \parindent0pt\smallskip Lorsque les arguments ne sont plus développés (utilisation de \verb|\noexpandarg|), on peut aussi inclure dans les arguments n'importe quel token, quelque soit le code qui en résulte comme par exemple toute séquence de contrôle, même non définie ou tout token de catcode 13. On peut également inclure dans les arguments des tokens de test comme \verb|\if| ou \verb|\ifx| ou tout autre token de test, même sans leur \verb|\fi| correspondant; ou bien un \verb|\csname| sans le \verb|\endcsname| correspondant. Dans l'exemple suivant, l'argument contient un \verb|\ifx| \emph{sans} le \verb|\fi|, et l'on isole ce qui est entre le \verb|\ifx| et le \verb|\else| :\par\medskip\nobreak \showcode+\noexpandarg \StrBetween{\ifx ab faux \else vrai}{\ifx}{\else}+ \medskip Lorsqu'on utilise \verb|\expandarg|, la précaution concerne le premier token qui est développé une fois et doit donc être défini. Les autres tokens sont laissés tel quels comme avec \verb|\noexpandarg|. \subsection{Développement des macros, argument optionnel} \label{argumentoptionnel} Les macros de ce package ne sont pas purement développables et ne peuvent donc pas être mises dans l'argument d'un \verb|\edef|. L'imbrication des macros de ce package n'est pas permise non plus.\medskip C'est pour cela que les macros renvoyant un résultat, c'est-à-dire toutes sauf les tests, sont dotées d'un argument optionnel venant en dernière position. Cet argument prend la forme de \arguC{nom}, où \argu{nom} est une séquence de contrôle qui recevra (l'assignation se fait avec un \verb|\edef|) le résultat de la macro, ce qui fait que \argu{nom} est purement développable et peut donc se trouver dans l'argument d'un \verb|\edef|. Dans le cas de la présence d'un argument optionnel en dernière position, aucun affichage n'aura lieu. Cela permet donc contourner les limitations évoquées dans les exemples ci dessus.\medskip Ainsi cette construction non permise censée assigner à \verb|\Resultat| les 4 caractères de gauche de \verb|xstring| :\par\nobreak \hspace{0.2\linewidth}\verb|\edef\Resultat{\StrLeft{xstring}{4}}|\par\nobreak \qquad est équivalente à :\par\nobreak \hspace{0.2\linewidth}\verb|\StrLeft{xstring}{4}[\Resultat]|\medskip Et cette imbrication non permise censée enlever le premier et le dernier caractère de \verb|xstring| :\par\nobreak \hspace{0.2\linewidth}\verb|\StrGobbleLeft{\StrGobbleRight{xstring}{1}}{1}|\par\nobreak \qquad se programme ainsi :\par\nobreak \hspace{0.2\linewidth}\verb|\StrGobbleRight{xstring}{1}[\machaine]|\par\nobreak \hspace{0.2\linewidth}\verb|\StrGobbleLeft{\machaine}{1}|\par \subsection{Traitement des arguments} \label{developpementarguments} \subsubsection{Traitement à l'\US prés} Les macros de \Xstring traitent les arguments \US par \US. Dans le code \TeX{}, une \US\footnote{Pour les utilisateurs familiers avec la programmation \LaTeX, une \US est ce qui est supprimé par la macro \texttt{\string\@gobble} dont le code, d'une grande simplicité, est : {\ttfamily\string\def\string\@gobble\string#1\string{\string}}} est soit :\par\nobreak\smallskip\parindent3em \begin{itemize} \item une séquence de contrôle; \item un groupe, c'est à dire une suite de tokens située entre deux accolades équilibrées; \item un caractère ne faisant pas partie des 2 espèces ci dessus. \end{itemize} \medskip\parindent0pt Voyons ce qu'est la notion d'\US sur un exemple. Prenons cet argument : \og\verb|ab\textbf{xyz}cd|\fg \nobreak Il contient 6 \USs qui sont : \og\verb|a|\fg, \og\verb|b|\fg, \og\verb|\textbf|\fg, \og\verb|{xyz}|\fg, \og\verb|c|\fg{} et \og\verb|d|\fg. \nobreak\medskip Que va t-il arriver si l'on se place sous \verb|\noexpandarg| et que l'on demande à \Xstring de trouver la longueur de cet argument et d'en trouver le 4\ieme{} \guill{caractère} ?\medskip \showcode+\noexpandarg \StrLen{ab\textbf{xyz}cd}\par \StrChar{ab\textbf{xyz}cd}{4}[\mychar] \meaning\mychar+ \fullexpandarg \medskip Il est nécessaire d'utiliser \verb|\meaning| pour bien visualiser le véritable contenu de \verb|\mychar| et non pas de simplement d'appeler cette séquence de contrôle, ce qui fait perdre des informations ---~les accolades ici. On voit qu'on n'obtient pas vraiment un \guill{caractère}, mais cela était prévisible : il s'agit d'une \US. \subsubsection{Exploration des groupes} Par défaut, la commande \verb|\noexploregroups| est appelée et donc dans l'argument à examiner qui contient la chaîne de tokens, \Xstring considère les groupes entre accolades comme \USs fermées dans lesquelles \Xstring ne regarde pas. Pour certains besoins spécifiques, il peut être nécessaire de modifier le mode de lecture des arguments et d'explorer l'intérieur des groupes entre accolades. Pour cela on peut invoquer \verb|\exploregroups|.\medskip Que va donner ce nouveau mode d'exploration sur l'exemple précédent ? \Xstring ne va plus compter le groupe comme une seule \US mais va compter les \USs se trouvant à l'intérieur, et ainsi de suite s'il y avait plusieurs niveaux d'imbrication de groupes: \nobreak\smallskip \showcode+\noexpandarg \exploregroups \StrLen{ab\textbf{xyz}cd}\par \StrChar{ab\textbf{xyz}cd}{4}[\mychar] \meaning\mychar+ \fullexpandarg\noexploregroups \medskip L'exploration des groupes peut se réveler utile pour le comptage, le calcul de position ou les tests, mais comporte une limitation lorsque l'on appelle des macros renvoyant des chaînes : lorsqu'un argument est coupé à l'intérieur d'un groupe, alors \textbf{le résultat ne tient pas compte de ce qui se trouve à l'extérieur de ce groupe}. Il faut donc utiliser ce mode en connaissance de cause lorsque l'on utilise les macros renvoyant des chaînes. Voyons ce que cela signifie sur un exemple : mettons que l'on veuille renvoyer ce qui se trouve à droite de la 2\ieme{} occurrence de \verb|\a| dans l'argument \verb|\a1{\b1\a2}\a3|. Comme l'on explore les groupes, cette occurrence se trouve à l'intérieur du groupe \verb|{\b1\a2}|. Le résultat renvoyé sera donc : \verb|\b1|. Vérifions-le : \showcode+\noexpandarg \exploregroups \StrBefore[2]{\a1{\b1\a2}\a3}{\a}[\mycs] \meaning\mycs+ L'exploration des groupes\footnote{On peut consulter le fichier de test de \Xstring qui comporte de nombreux exemples et met en évidence les différences selon le mode d'exploration des groupes.} peut ainsi changer le comportement de la plupart des macros de \Xstring, à l'exception de \verb|\IfInteger|, \verb|\IfDecimal|, \verb|\IfStrEq|, \verb|\IfEq| et \verb|\StrCompare| qui sont insensibles au mode d'exploration en cours. De plus, pour des raison d'équilibrage d'accolades, 2 macros n'opèrent qu'en mode \verb|\noexploregroups|, quelque soit le mode d'exploration en cours : \verb|\StrBetween| et \verb|\StrMid|.\medskip On peut mémoriser le mode de d'exploration en cours avec \verb|\saveexploremode|, le modifier par la suite et revenir à la situation lors de la sauvegarde avec \verb|\restoreexploremode|. \subsection{Catcodes et macros étoilées} \label{macrosetoilees} Les macros de \Xstring tiennent compte des catcodes des tokens constituant les arguments. Il faut donc garder à l'esprit, particulièrement lors des tests, que les tokens \emph{et leurs catcodes} sont examinés.\medskip Par exemple, ces 2 arguments :\par\nobreak\smallskip \hfil\verb|{\string a\string b}|\qquad et\qquad\verb|{ab}|\hfil{}\par\smallskip ne se développent pas en 2 arguments égaux aux yeux de \Xstring. Dans le premier cas, à cause de l'emploi de la primitive \verb|\string|, les caractères \og\verb|ab|\fg{} ont un catcode de 12 alors que dans l'autre cas, ils ont leurs catcodes naturels de 11. Il convient donc d'être conscient de ces subtilités lorsque l'on emploie des primitives dont les résultats sont des chaînes de caractères ayant des catcodes de 12 et 10. Ces primitives sont par exemple : \verb|\string|, \verb|\detokenize|, \verb|\meaning|, \verb|\jobname|, \verb|\fontname|, \verb|\romannumeral|, etc.\medskip Pour demander aux macros de ne pas tenir compte des catcodes, on peut utiliser les macros étoilées. Après l'éventuel développement des arguments en accord avec le mode de développement, celles-ci convertissent (à l'aide d'un \verb|\detokenize|) leur arguments en chaînes de caractères dont les catcodes sont 12 et 10 pour l'espace, avant que la macro non étoilée travaille sur ces arguments ainsi modifiés. Il faut noter que les arguments optionnels ne sont pas concernés par ces modifications et gardent leur catcode.\smallskip Voici un exemple :\par\nobreak\smallskip \showcode+\IfStrEq{\string a\string b}{ab}{vrai}{faux}\par \IfStrEq*{\string a\string b}{ab}{vrai}{faux}+ \smallskip Les chaînes n'étant pas égales à cause des catcodes, le test est bien \emph{négatif} dans la version non étoilée.\bigskip \textbf{Attention} : utiliser une macro étoilée a des conséquenses ! Les arguments sont \guill{détokénisés}, il n'y a donc plus de séquence de contrôle, plus de goupes, ni aucun caractère de catcode spécial puisque tout est converti en caractères \guill{inoffensifs} ayant le même catcode.\medskip Ainsi, pour les macros renvoyant une chaîne, si on emploie les versions étoilées, le résultat sera une chaîne de caractères dont les catcodes sont 12, et 10 pour l'espace. Et donc, après un \og\verb|\StrBefore*{a \b c d}{c}[\montexte]|\fg{}, la séquence de contrôle \verb|\montexte| se développera en \og{}\verb|a|${}_{12}$\verb*| |${}_{10}$\verb|\|${}_{12}$\verb|b|${}_{12}$\verb*| |${}_{10}$\fg{}.\medskip Les macros détokenisant leur arguments par l'utilisation de l'étoile sont présentées dans la liste ci-dessous. Pour chacune d'entre elles, on peut voir en \textcolor{violet}{violet} quels arguments seront détokenisé lorsque l'étoile sera employée : \nobreak\smallskip\parindent3em \begin{itemize} \item\verb|\IfSubStr|\etoile\arguC{nombre}{\colorise\ARGU{chaine}\ARGU{chaineA}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfSubStrBefore|\etoile\arguCC{nombre1}{nombre2}{\colorise\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfSubStrBehind|\etoile\arguCC{nombre1}{nombre2}{\colorise\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfBeginWith|\etoile{\colorise\ARGU{chaine}\ARGU{chaineA}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfEndWith|\etoile{\colorise\ARGU{chaine}\ARGU{chaineA}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfStrEq|\etoile{\colorise\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfEq|\etoile{\colorise\ARGU{chaineA}\ARGU{chaineB}}\ARGU{vrai}\ARGU{faux} \item\verb|\IfStrEqCase|\etoile{\colorise\ARGU{chaine}\texttt{\color{black}\{}\colorise\ARGU{chaine1}}\ARGU{code1}\par \setbox0=\hbox{{\ttfamily\string\IfStrEqCase}\etoile\ARGU{chaine}{\ttfamily\{}} \hskip\wd0{\colorise\ARGU{chaine2}}\ARGU{code2}\par \hskip\wd0\ \ldots\par \hskip\wd0{\colorise\ARGU{chaine $n$}}\ARGU{code $n$}\verb|}|\arguC{code alternatif} \item\verb|\IfEqCase|\etoile{\colorise\ARGU{chaine}\texttt{\color{black}\{}\colorise\ARGU{chaine1}}\ARGU{code1}\par \setbox0=\hbox{{\ttfamily\string\IfEqCase}\etoile\ARGU{chaine}{\ttfamily\{}} \hskip\wd0{\colorise\ARGU{chaine2}}\ARGU{code2}\par \hskip\wd0\ \ldots\par \hskip\wd0{\colorise\ARGU{chaine $n$}}\ARGU{code $n$}\verb|}|\arguC{code alternatif} \item\verb|\StrBefore|\etoile\arguC{nombre}{\colorise\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrBehind|\etoile\arguC{nombre}{\colorise\ARGU{chaine}\ARGU{chaineA}}\arguC{nom} \item\verb|\StrBetween|\etoile\arguCC{nombre1}{nombre2}{\colorise\ARGU{chaine}\ARGU{chaineA}\ARGU{chaineB}}\arguC{nom} \item\verb|\StrCompare|\etoile{\colorise\ARGU{chaineA}\ARGU{chaineB}}\arguC{nom} \end{itemize} \parindent0pt \section{Macros avancées pour la programmation} Bien que \Xstring ait la possibilité de lire et traiter des arguments contenant du code \TeX{} ou \LaTeX{} ce qui devrait couvrir la plupart des besoins en programmation, il peut arriver pour des besoins très spécifiques que les macros décritesz précédemment ne suffisent pas. Ce chapitre présente d'autres macros qui permettent d'aller plus loin ou de contourner certaines limitations. \subsection{Recherche d'un groupe, les macros {\ttfamily\textbackslash StrFindGroup} et {\ttfamily\textbackslash groupID}} Lorsque le mode \verb=\exploregroups= est actif, la macro \verb=\StrFindGroup= permet de trouver un groupe entre accolades explicites en spécifiant son identifiant : \nobreak\smallskip \verb|\StrFindGroup|\ARGU{argument}\ARGU{identifiant}\arguC{nom}\medskip Lorsque le groupe caractérisé par l'identifiant n'existe pas, une chaîne vide sera assignée à la séquence de contrôle \argu{nom}. Si le groupe existe, ce groupe \emph{avec ses accolades} sera assigné à \argu{nom}.\smallskip Cet identifiant est une suite d'entiers séparés par des virgules caractérisant le groupe cherché dans l'argument. Le premier entier est le $n$\ieme{} groupe (d'imbrication 1) dans lequel est le groupe cherché. Puis, en se plaçant dans ce groupe, le 2\ieme{} entier est le $n$\ieme{} groupe dans lequel est le groupe cherché. Et ainsi de suite jusqu'à ce que l'imbrication du groupe soit atteinte.\bigskip Prenons par exemple l'argument suivant où l'on a 3 niveaux d'imbrication de groupes. Pour plus de clarté, les accolades délimitant les groupes sont colorées en rouge pour l'imbrication de niveau 1, en bleu pour le niveau 2 et en vert pour le niveau 3. Les groupes dans chaque imbrication sont ensuite numérotés selon la règle décrite ci-dessus : \begingroup \def\AccO#1{\text{\color{#1}\ttfamily\{}} \def\AccF#1{\text{\color{#1}\ttfamily\}}} \def\texte#1{\text{\texttt{\color{black}#1}}} \def\decalbrace{\vphantom{\underbrace{A}_A}} \[\texte{a}{\color{red}\underbrace{\AccO{red}{\color{blue}\underbrace{\decalbrace\AccO{blue}\texte{bc}\AccF{blue}}_{\color{blue}1}}\texte{d}{\color{blue}\underbrace{\decalbrace\AccO{blue}\texte{efg}\AccF{blue}}_{\color{blue}2}}\AccF{red}}_{\color{red}1}}\texte{h}{\color{red}\underbrace{\AccO{red}{\color{blue}\underbrace{\decalbrace\AccO{blue}\texte{ij}\AccF{blue}}_{\color{blue}1}}{\color{blue}\underbrace{\decalbrace\AccO{blue}\texte{k}\AccF{blue}}_{\color{blue}2}}{\color{blue}\underbrace{\AccO{blue}\texte{l}{\color{green}\underbrace{\AccO{green}\texte{m}\AccF{green}}_{\color{green}1}}{\color{green}\underbrace{\AccO{green}\texte{no}\AccF{green}}_{\color{green}2}}\AccF{blue}}_{\color{blue}3}}\texte{p}\AccF{red}}_{\color{red}2}}\] \endgroup \smallskip Dans cet exemple : \nobreak\parindent3em \begin{itemize} \item le groupe \verb={{bc}d{efg}}= a donc pour identifiant \texttt{\color{red}1}; \item le groupe \verb={ij}= a pour identifiant \texttt{{\color{red}2},\color{blue}1}; \item le groupe \verb={no}= a pour identifiant \texttt{{\color{red}2},{\color{blue}3},\color{green}2}; \item l'argument dans sa totalité \og\verb=a{{bc}d{efg}}h{{ij}{k}{l{m}{no}}p}=\fg{} a pour identifiant \verb=0=, seul cas où l'entier 0 est contenu dans l'identifiant d'un groupe. \end{itemize} \parindent0pt \medskip Voici l'exemple complet :\par\nobreak\smallskip \showcode+\exploregroups \expandarg \def\chaine{a{{bc}d{efg}}h{{ij}{k}{l{m}{no}}p}} \StrFindGroup{\chaine}{1}[\mongroupe] \meaning\mongroupe\par \StrFindGroup{\chaine}{2,1}[\mongroupe] \meaning\mongroupe\par \StrFindGroup{\chaine}{2,3,2}[\mongroupe] \meaning\mongroupe\par \StrFindGroup{\chaine}{2,5}[\mongroupe] \meaning\mongroupe+ \bigskip Le processus inverse existe, et plusieurs macros de \Xstring donnent aussi comme information l'identifiant du groupe dans lequel elles ont fait une coupure ou trouvé une recherche. Ces macros sont : \verb|\IfSubStr|, \verb|\StrBefore|, \verb|\StrBehind|, \verb|\StrSplit|, \verb|\StrLeft|, \verb|\StrGobbleLeft|, \verb|\StrRight|, \verb|\StrGobbleRight|, \verb|\StrChar|, \verb|\StrPosition|.\medskip Après l'appel à ces macros, la commande \verb|\groupID| se développe en l'identifiant du groupe dans lequel la coupure s'est faite ou la recherche d'un argument a abouti. Lorsque la coupure ne peut avoir lieu ou que la recherche n'a pas abouti, \verb|\groupID| est vide. Évidemment, l'utilisation de \verb|\groupID| n'a de sens que lorsque le mode \verb|\exploregroups| est actif, et quand les macros ne sont pas étoilées.\smallskip Voici quelques exemples avec la macro \verb|\StrChar| :\par\nobreak\smallskip \showcode+\exploregroups char 1 = \StrChar{a{b{cd}{e{f}g}h}ijkl}{1}\qquad \string\groupID = \groupID\par char 4 = \StrChar{a{b{cd}{e{f}g}h}ijkl}{4}\qquad \string\groupID = \groupID\par char 6 = \StrChar{a{b{cd}{e{f}g}h}ijkl}{6}\qquad \string\groupID = \groupID\par char 20 = \StrChar{a{b{cd}{e{f}g}h}ijkl}{20}\qquad \string\groupID = \groupID+ \subsection{Coupure d'une chaîne, la macro {\ttfamily\textbackslash StrSplit}} \label{StrSplit} Voici la syntaxe de cette macro :\par\nobreak\smallskip \verb|\StrSplit|\ARGU{chaine}\ARGU{nombre}\ARGU{chaineA}\ARGU{chaineB} \smallskip La \argu{chaine}, est coupée en deux chaînes juste après l'\US se situant à la position \argu{nombre}. La partie gauche est assigné à la séquence de contrôle \argu{chaineA} et la partie droite à \argu{chaineB}.\par Cette macro renvoie \emph{deux chaînes} et donc \textbf{n'affiche rien}. Par conséquent, elle ne dispose pas de l'argument optionnel en dernière position.\medskip \begin{Conditions} \item Si \argu{nombre}${}\leqslant0$, \argu{chaineA} sera vide et \argu{chaineB} contiendra la totalité de \argu{chaine}; \item Si \argu{nombre}${}\geqslant$\argu{longueurChaine}, \argu{chaineA} contiendra la totalité de \argu{chaine} et \argu{chaineB} sera vide; \item Si \argu{chaine} est vide \argu{chaineA} et \argu{chaineB} seront vides, quelque soit l'entier \argu{nombre}. \end{Conditions} \showcode+\def\seprouge{{\color{red}|}} \StrSplit{abcdef}{4}{\csA}{\csB}|\csA\seprouge\csB|\par \StrSplit{a b c }{2}{\csA}{\csB}|\csA\seprouge\csB|\par \StrSplit{abcdef}{1}{\csA}{\csB}|\csA\seprouge\csB|\par \StrSplit{abcdef}{5}{\csA}{\csB}|\csA\seprouge\csB|\par \StrSplit{abcdef}{9}{\csA}{\csB}|\csA\seprouge\csB|\par \StrSplit{abcdef}{-3}{\csA}{\csB}|\csA\seprouge\csB|+ Lorsque l'exploration des groupes est activée, et que l'on demande une coupure en fin de groupe, alors une chaîne contiendra la totalité du groupe tandis que l'autre sera vide comme on le voit sur cet exemple :\par\nobreak\smallskip \showcode+\exploregroups \StrSplit{ab{cd{ef}gh}ij}{6}\strA\strB \meaning\strA\par \meaning\strB+\medskip Une version étoilée de cette macro existe : dans ce cas, la coupure se fait juste avant la prochaine \US qui suit l'\US désirée. La version étoilée ne donne des résultats différents de la version normale que lorsque la $n$\ieme{} \US est à la fin d'un groupe auquel cas, la coupure intervient non pas après cette \US mais \emph{avant} la prochaine \US, que \verb|\StrSplit| atteint en fermant autant de groupes que nécessaire.\smallskip \showcode+\exploregroups Utilisation sans étoile :\par \StrSplit{ab{cd{ef}gh}ij}{6}\strA\strB \meaning\strA\par \meaning\strB\par \string\groupID\ = \groupID\par\medskip Utilisation avec \'etoile :\par \StrSplit*{ab{cd{ef}gh}ij}{6}\strA\strB \meaning\strA\par \meaning\strB\par \string\groupID\ = \groupID+ \subsection{Assigner un contenu verb, la macro \ttfamily\textbackslash verbtocs} \label{verbtocs} La macro \verb|\verbtocs| permet le lire le contenu d'un \guill{verb} qui peut contenir tous les caractères spéciaux : \verb|&|, \verb|~|, \verb|\|, \verb|{|, \verb|}|, \verb|_|, \verb|#|, \verb|$|, \verb|^| et \verb|%|. Les caractères \guill{normaux} gardent leur catcodes naturels, sauf les caractères spéciaux qui prennent un catcode de 12. Ensuite, ces caractères sont assignés à une séquence de contrôle. La syntaxe complète est :\par\nobreak\medskip \hfill\verb|\verbtocs|\ARGU{nom}|\argu{caractères}|\hfill{} \smallskip \argu{nom} est le nom d'une séquence de contrôle qui recevra à l'aide d'un \verb|\edef| les \argu{caractères}. \argu{nom} contiendra donc des caractères de catcodes 12 ou 10 pour l'espace.\medskip Par défaut, le token délimitant le contenu verb est \guill{|}, étant entendu que ce token ne peut être à la fois le délimiteur et être contenu dans ce qu'il délimite. Au cas où on voudrait lire un contenu verb contenant \guill{|}, on peut changer à tout moment le token délimitant le contenu verb par la macro :\par\medskip \hfill\verb|\setverbdelim|\ARGU{token}\hfill{}\smallskip Tout \argu{token} de catcode 12 peut être utilisé\footnote{Plusieurs tokens peuvent être utilisés au risque d'alourdir la syntaxe de \texttt{\string\verbtocs} ! Pour cette raison, avertissement sera émis si l'argument de \texttt{\string\setverbdelim} contient plusieurs tokens.}.\medskip Concernant ces arguments verb, il faut tenir compte des deux points suivants : \begin{itemize} \item tous les caractères se trouvant avant |\argu{caractères}| seront ignorés; \item à l'intérieur des délimiteurs, tous les espaces sont comptabilisés même s'ils sont consécutifs. \end{itemize} \medskip Exemple :\par\nobreak\medskip \showcode+\verbtocs{\resultat} |a & b{ c% d$ e \f| J'affiche le résultat :\par\resultat+ \subsection{Tokenisation d'un texte vers une séquence de contrôle, la macro \ttfamily\textbackslash tokenize} Le processus inverse de ce qui a été vu au dessus consiste à interpréter une suite de caractères en tokens. Pour cela, on dispose de la macro :\par\nobreak\medskip \hfill\verb|\tokenize|\ARGU{nom}\ARGU{texte}\hfill{} \smallskip \argu{texte} est développé le plus possible si l'on a invoqué \verb|\fullexpandarg|; il n'est pas développé si l'on a invoqué une des commandes \verb|\noexpandarg| ou \verb|\expandarg|. Après développement éventuel, le \argu{texte} est transformé en tokens puis assigné à l'aide d'un \verb|\def| à la séquence de contrôle \argu{nom}.\medskip Voici un exemple où l'on détokenise un argument, on affiche le texte obtenu, puis on transforme ce texte en ce que l'argument était au début; enfin, on affiche le résultat de la tokenisation :\par\nobreak\medskip \showcode+\verbtocs{\text}|\textbf{a} $\frac{1}{2}$| texte : \text \tokenize{\resultat}{\text}\par réesultat : \resultat+ \medskip Il est bien évident à la dernière ligne, que l'appel à la séquence de contrôle \verb|\resultat| est ici possible puisque les séquences de contrôle qu'elle contient sont définies.\medskip \subsection{Développement contrôlé, les macros {\ttfamily\textbackslash StrExpand} et {\ttfamily\textbackslash scancs}} \label{scancs} La macro \verb|\StrExpand| développe les tokens de la chaîne passée en argument. Voici sa syntaxe :\par\nobreak\medskip \verb|\StrExpand|\arguC{nombre}\ARGU{chaine}\ARGU{nom}\smallskip Le \argu{nombre} vaut 1 par défaut et représente le nombre de développement(s) que doit subir la \argu{chaine} de tokens. Le \argu{nom} est le nom d'une séquence de contrôle à laquelle est assigné le résultat, une fois que tous les tokens aient été développé le nombre de fois demandé.\medskip La macro opère séquentiellement et par passe : chaque token est remplacé par son 1-dévelopement, et le token suivant subit le même traitement jusqu'à ce que la chaîne ait été parcourue. Ensuite, s'il reste des niveaux de développement à faire, une nouvelle passe est initiée, et ainsi de suite jusqu'à ce que le nombre de développements voulu aient été exécutés.\medskip Voici un exemple :\par\nobreak\smallskip \showcode+\def\csA{1 2} \def\csB{a \csA} \def\csC{\csB\space} \def\csD{x{\csA y}\csB{\csC z}} Développement de \string\csD\ au\par \StrExpand[0]{\csD}{\csE} niveau 0 : \detokenize\expandafter{\csE}\par \StrExpand[1]{\csD}{\csE} niveau 1 : \detokenize\expandafter{\csE}\par \StrExpand[2]{\csD}{\csE} niveau 2 : \detokenize\expandafter{\csE}\par \StrExpand[3]{\csD}{\csE} niveau 3 : \detokenize\expandafter{\csE}\par \StrExpand[4]{\csD}{\csE} niveau 4 : \detokenize\expandafter{\csE}+ \medskip La macro agit séquentiellement et chaque token est développé isolément sans tenir compte de ce qui suit. On ne peut donc développer que des tokens qui se suffisent à eux même et dont le développement ne nécessite aucun autre token. Ainsi, \og\hbox{\verb|\iftrue A\else B\fi|}\fg, bien que se développant en \guill{A} ne peut être mis dans l'argument de \verb|\StrExpand|, et l'instruction :\par\nobreak\smallskip \hfil\verb|\StrExpand{\iftrue A\else B\fi}\resultat|\hfil{}\par\nobreak\smallskip fera échouer la compilation puisque le premier token \og\verb|\iftrue|\fg{} sera développé \emph{seul}, c'est-à-dire sans son \verb-\fi- correspondant, ce qui fâchera \TeX{} !\smallskip Les règles habituelles de lecture des arguments sont en vigueur, à savoir qu'un espace suivant une séquence de contrôle est ignoré, et plusieurs espaces consécutifs n'en font qu'un. Ces règles ne s'appliquent pas pour des espaces explicitement demandés avec \verb-\space- ou \verb*-\ -\footnote{À ce propos, \texttt{\string\space} n'a pas la même signification que \texttt{\string\ }. La première séquence de contrôle se \emph{développe} en un espace tandis que la deuxième est une primitive \TeX{} qui \emph{affiche} une espace. Comme toutes les primitives, cette dernière est son propre développement.}.\bigskip On peut détokeniser le résultat obtenu par la macro \verb-\StrExpand- avec la macro \verb-\scancs- dont la syntaxe est :\par\nobreak\medskip \verb|\scancs|\arguC{nombre}\ARGU{nom}\ARGU{chaine}\par\nobreak\smallskip Le \argu{nombre} vaut 1 par défaut et représente le nombre de développement(s) que doit subir chaque token constituant la \argu{chaine}. Le \argu{nom} est le nom d'une séquence de contrôle à laquelle est assigné le résultat, une fois que tous les tokens aient été développés le nombre de fois demandé et ensuite détokénisés.\smallskip \verb-\scancs- a été conservée pour garder une compatibilité avec des précédentes versions de \Xstring. Pour les mêmes raisons, sa syntaxe n'est pas cohérente avec la syntaxe des autres macros. Cette macro, devenue triviale, prend le résultat de \verb|\StrExpand| et lui applique simplement un \verb|\detokenize|. \subsection{À l'intérieur d'une définition de macro} Avec le verbatim, certaines difficultés surviennent lorsque l'on se trouve à l'intérieur de la définition d'une macro, c'est-à-dire entre les accolades suivant un \verb|\def\macro| ou un \verb|\newcommand\macro|.\medskip Pour les mêmes raison qu'il est interdit d'employer la commande \verb|\verb| à l'intérieur de la définition d'une macro, les arguments verb du type |\argu{caractères}| sont également interdits, ce qui disqualifie la macro \verb|\verbtocs|. Il faut donc observer la règle suivante :\par\medskip \hfill{\bfseries Ne pas utiliser la macro \verb-\verbtocs- à l'intérieur de la définition d'une macro}.\hfill{}\bigskip Mais alors, comment faire pour manipuler des arguments textuels verb et \guill{verbatimiser} dans les définitions de macro ?\bigskip Il y a la primitive \verb|\detokenize| de $\varepsilon$-\TeX, mais elle comporte des restrictions, entre autres : \begin{itemize} \item les accolades doivent être équilibrées; \item les espaces consécutifs sont ignorés; \item les signes \verb|%| sont interdits; \item une espace est ajoutée après chaque séquence de contrôle; \item le token \verb|#| devient \verb|##|. \end{itemize} \medskip Il est préférable d'utiliser la macro \verb|\scancs|, et définir avec \verb|\verbtocs| à l'extérieur des définitions de macros, des séquences de contrôle contenant des caractères spéciaux détokénisés. On pourra aussi utiliser la macro \verb|\tokenize| pour transformer le résultat final (qui est une chaîne de caractères) en une séquence de contrôle.\medskip Dans l'exemple artificiel\footnote{On peut agir beaucoup plus simplement en utilisant la commande \texttt{\string\detokenize}. Il suffit de définir la macro ainsi :\par\texttt{\string\newcommand\string\bracearg[1]\{\string\detokenize\{\{\#1\}\}\}}} qui suit, on écrit une macro qui met son argument entre accolades. Pour cela, on définit en dehors de la définition de la macro 2 séquences de contrôles \verb|\Ob| et \verb|\Cb| contenant une accolade ouvrante et une accolade fermante de catcodes 12. Ces séquences de contrôle sont ensuite développées et utilisées à l'intérieur de la macro pour obtenir le résultat voulu :\par\nobreak\medskip \showcode+\verbtocs{\Ob}|{| \verbtocs{\Cb}|}| \newcommand\bracearg[1]{% \def\text{#1}% \scancs{\result}{\Ob\text\Cb}% \result} \bracearg{xstring}\par \bracearg{\a}+ \subsection{La macro \ttfamily\textbackslash StrRemoveBraces} Pour des utilisations spéciales, on peut désirer retirer les accolades délimitant les groupes dans un argument. On peut utiliser la macro \verb|\StrRemoveBraces| dont voici la syntaxe :\par\nobreak\smallskip \verb|\StrRemoveBraces|\ARGU{chaine}\arguC{nom} \smallskip Cette macro est sensible au mode d'exploration, et retirera \emph{toutes} les accolades avec \verb|\exploregroups| alors qu'elle ne retirera que les accolades des groupes de plus bas niveau avec \verb|\noexploregroups|.\medskip \showcode+\noexploregroups \StrRemoveBraces{a{b{c}d}e{f}g}[\mycs] \meaning\mycs \exploregroups \StrRemoveBraces{a{b{c}d}e{f}g}[\mycs] \meaning\mycs+ \begin{center} $\star$\par $\star$\quad$\star$ \end{center} C'est tout, j'espère que ce package vous sera utile !\par\nobreak Merci de me signaler par \href{mailto:unbonpetit@netc.fr}{email} tout bug ou toute proposition d'amélioration\ldots\par\nobreak\bigskip Christian \textsc{Tellechea} \end{document}