Programmer avec XSLT
Un article de Haypo.
La première version de ce résumé de la programmer avec XSLT a été écrite par Victor Stinner le 16 octobre 2003. Ce résumé a été repris sur developpez.com (version HTML et PDF téléchargeables).
Sommaire |
[modifier] Introduction
XSLT offre de nombreuses fonctions dignes d'un langage de haut-niveau : variables, paramètres, tests, boucles, fonctions, inclusion d'une feuille de style XSLT dans une autre, chargement de plusieurs documents XML dans une même feuille de style XSLT, recherche de balises XML selon de nombreux critères, etc.
Note : Si vous n'avez pas lu mon introduction à XSLT, allez vite la lire sous peine de ne pas comprendre certains passages.
[modifier] Fonctions XSLT
[modifier] Liste des fonctions XSLT
| Fonction XSLT | Description |
|---|---|
| xsl:for-each | "Boucle" |
| xsl:if | Si conditionnel |
| xsl:choose, xsl:when et xsl:otherwise | Suite de codes conditionnels : instruction switch en C |
| xsl:template, xsl:call-template, xsl:param et xsl:with-param | Déclarer et appler une fonction, avec ou sans paramètre(s). |
| xsl:number | Numérotation/compteur |
[modifier] Boucles avec xsl:for-each
Je vous ai menti (oh le vilain !). En fait, xsl:for-each n'est pas vraiment une boucle comme un "for" en langage C. Cette fonction va prendre tous les noeuds d'une requête XPATH, et va leur appliquer un traitement.
Voyons déjà le fichier source :
<liste> <invite>Moi</invite> <invite>Jean</invite> <invite>Michel</invite> </liste>
Puis le code XSLT qui va générer une liste HTML :
<ul>
<xsl:for-each>
<li>
<xsl:apply-templates select="." />
</li>
</for-each>
</ul>
[modifier] Fonction xsl:if
L'instruction xsl:if permet d'exécuter ou non certaines parties du code. Exemple d'utilisation de la l'instruction if :
<xsl:if test="nom == 'Victor'"> <xsl:text>Tient, on dirait mon prénom !</xsl:text> </xsl:if>
Ok, mais où est le xsl:else ? Euh ... Désolé, y'a pas :-( Il faut utiliser xsl:choose.
[modifier] Fonction xsl:choose
La fonction xsl:choose permet d'exécuter différents codes selon différentes conditions. Exemple d'utilisation de la l'instruction choose :
<xsl:choose>
<xsl:when test="nom = 'Victor'">
<xsl:text>Salut Victor, ça boume ?</xsl:text>
</xsl:when>
<xsl:when test="nom = 'Pierre'">
<xsl:text>Fait ciseau ... désolé, c'était nul.</xsl:text>
</xsl:when>
<xsl:otherwise>
<xsl:text>Message impersonnel et froid : Bonjour.</xsl:text>
</xsl:otherwise>
</xsl:choose>
On utilise souvent xsl:choose comme alternative au xsl:if pour disposer de xsl:otherwise (qui remplace xsl:else qui n'existe pas).
[modifier] Une fonction sans paramètre (xsl:template et xsl:call-template)
Pour déclarer une fonction, on utilise la fonction xsl:template. Exemple :
<xsl:template name="hello_world"> <xsl:text>Hello World !</xsl:text> </xsl:template>
On va maintenant appeler notre fonction avec xsl:call-template :
<xsl:call-template name="hello_world" />
C'est tout ... ben ouais. En fait, c'est tout bête !
[modifier] Une fonction avec paramètres (xsl:param et xsl:with-param)
Bon, passons aux choses sérieuses : passons des paramètres à notre fonction. On va par exemple afficher deux nombres ... et leur somme, yahoo ! Pour déclarer des paramètres, on utilise xsl:param. Voici le code de la fonction :
<xsl:template name="affiche_somme"> <xsl:param name="a" select="0" /> <xsl:param name="b" select="0" /> <xsl:text>a = </xsl:text> <xsl:value-of select="$a" /> <xsl:text>, b = </xsl:text> <xsl:value-of select="$b" /> <xsl:text>, et a+b = </xsl:text> <xsl:value-of select="$a + $b" /> <xsl:text>.</xsl:text> </xsl:template>
Lorsqu'on écrit <xsl:param name="a" select="0" /> : 0 est la valeur par défaut du paramètre a. Appelons ce miracle de la programmation, toujours via xsl:call-template :
<xsl:call-template name="affiche_somme"> <xsl:with-param name="a" select="173" /> <xsl:with-param name="b">9001</xsl:with-param> </xsl:call-template>
Vous savez quoi, le pire c'est que ça marche ! On obtient une sortie du style :
a = 173, b = 9001, et a+b = 9174.
[modifier] Bouts de codes utiles
[modifier] Une VRAI boucle
Je préfère vous le dire tout de suite : XSLT n'est pas fait pour faire des boucles ! C'est un langage qui utilise des algorithmes récursifs, et non pas itératifs. C'est d'ailleurs beaucoup plus propre, mais moins naturel.
Mais des fois, on a quand même besoin de boucles. Et oui, elles nous quitteront jamais, nous les programmeur :-) Je vous offre donc ce p'tit bout de code qui pourrait peut-être vous sauver la vie (qui sait ?) :
<xsl:template name="boucle">
<xsl:param name="debut" select="0" />
<xsl:param name="fin" select="0" />
<xsl:text>i = </xsl:text>
<xsl:value-of select="$debut" />
<br />
<xsl:if test="$debut < $fin">
<xsl:call-template name="boucle">
<xsl:with-param name="debut" select="($debut)+1" />
<xsl:with-param name="fin" select="$fin" />
</xsl:call-template>
</xsl:if>
</xsl:template>
Prototype de la fonction : boucle($debut, $fin). Elle va afficher les nombres de $debut à $fin. Que dire de plus ? Ah oui, ce "<" bizzare. Et bien c'est qu'on reste en XML, et "<" ne peut pas s'écrire tel quel. Il faut l'écrire dans sa forme encodée "<". Voilà tout !
[modifier] Fonctions XPATH
| Fonction XPATH | Description |
|---|---|
| position() | Position d'une balise dans l'arbre XML |
| count(xpath) | Compte le nombre d'éléments d'une requête XPATH |
[modifier] Chaîne de caractères (XPATH)
[modifier] Liste des fonctions
| Fonction XPATH | Description |
|---|---|
| concat(a,b,[c,d,...]) | Concaténation de chaînes |
| substring(a,debut[,longueur]) | Extrait une partie de la chaîne a sachant qu'une chaîne débute à la position 1 : substring('abc', 2, 1) retourne 'b' |
| substring-before(txt,token) | Ce qui précède la sous-chaîne token |
| substring-after(txt, token) | Ce qui suit la sous-chaîne token |
| string-length(txt) | longueur de la chaîne txt |
| translate(txt,avant,apres) | transforme les caractères avant par ceux après dans la chaîne txt |
[modifier] Exemple (1)
On va dire que le noeud courant (.) contient mon email : victor.stinner+haypocalc.com (lisez '@' et non '+'). Exemple d'utilisation des fonctions concat, substring-after et substring-before :
<xsl:value-of select="concat( substring-before(.,'@'), ' SUR ', substring-after(.,'@'))" />
Ben oui, on peut écrire sur plusieurs lignes si on veut, hé hé hé ! Sortie : "victor.stinner SUR haypocalc.com", cool, c'est ce que je cherchais à faire (cacher mon email) ;-)
[modifier] Exemple (2)
Disons que le noeud courant (.) contient le nom d'une image : "bibi_phoque.gif". On va remplacer l'extension '.gif' par '.png' (ben ouais, le GIF ça pue le brevet logiciel !). Exemple d'utilisation des fonctions substring et length :
<xsl:value-of select="concat(substring(.,length(.)-4),'.png')" />
Bon, la fonction xsl:string-before [b]pourrait être[/b] plus adaptée que substring. Sauf ... si le nom de l'image comporte un point ! "bibi.phoque.gif".
[modifier] Conditions (XPATH)
| Fonction XPATH | Description |
|---|---|
| true() | vrai |
| false() | faux |
| not(cond) | contraire de la condition cond |
| "a & b" ou "a and b" | a et b |
| b" ou "a or b" | a ou b |
| a == b | a égal b ? |
| a != b | a différent de b ? |
| a < b | a inférieur b ? (lire a < b, mais il faut l'écrire tel quel) |
| a <= b | a inférieur ou égal b ? (lire a <= b) |
| a > b | a supérieur b ? (lire a > b) |
| a >= b | a supérieur ou égal b ? (lire a >= b) |
[modifier] Mathématiques
| Fonction XPATH | Description |
|---|---|
| sum(xpath) | Aomme des éléments xpath |
| floor(n) | Arrondi par défaut |
| ceiling(n) | Arrondi par excès |
| round(n) | Arrondi à l'entier le plus proche |
| a + b | Somme de a et b |
| a - b | Différence de a et de b |
| a * b | Produit de a et de b |
| a div b | Quotient de a par b |
| a mod b | Reste dans la division entière de a par b |
| -x | Opposé de x |

