Aller au contenu

Générer des PDF en utilisant HTML et CSS

·17 mins
Sommaire

Il existe plein de manière de générer des documents PDF. Lorsqu’on n’a pas beaucoup de connaissance technique ou bien une certaine flemme, on utilise généralement un logiciel de traitement de texte tel que LibreOffice ou OnlyOffice. Les vrais nerds ainsi que les personnes ayant besoin de générer des documents de manière automatisée auront plutôt tendance à utiliser LaTeX. Bien qu’étant dans cette dernière catégorie de personnes, j’ai toujours détesté LaTeX qui, malgré ses nombreux avantages, est très difficile à prendre en main et surtout à styliser. Depuis pas mal d’années j’ai donc totalement abandonné LaTeX au profit de langage que je connais bien et avec lesquels j’ai d’excellents résultats : HTML et CSS.

Weasyprint
#

La meilleur manière que j’ai trouvé pour générer des documents PDF à partir de HTML et CSS est Weasyprint. Il s’agit d’un projet écrit en Python, packagé dans la plupart des distributions Linux, qui propose à la fois une interface en ligne de commande et une interface en Python. Dans ce billet je me concentrerai sur l’interface en ligne de commande qui est la plus générique et la plus pratique pour débuter.

Commençons donc par créer un fichier HTML assez simple :

<!doctype html>
<html lang="fr">
	<head>
		<meta charset="utf-8">
		<title>Mon premier document PDF</title>
		<meta name="author" content="Toto">
		<meta name="description" content="Un PDF de test">
		<meta name="generator" content="super_script.sh">
		<meta name="keywords" content="pdf,weasyprint,html,css">
		<meta name="dcterms.created" content="2026-01-28T17:21:42+01:00">
		<meta name="dcterms.modified" content="2026-01-28T18:21:42+01:00">
	</head>
	<body>
		<h1>Bonjour, monde&nbsp;!</h1>
	</body>
</html>

Et convertissons le en PDF :

$ weasyprint --pdf-variant "pdf/ua-1" "test.html" "test.pdf"
Capture d'écran montrant le fichier PDF ouvert dans un lecteur. Au premier plan, une fenêtre montre les différentes métadonnées du fichier.
On commence facilement, profitez-en car ça ne vas pas durer.

Vous remarquerez que Weasyprint a extrait toutes les métadonnées des balises meta et title afin de les enregistrer dans le PDF. Notez que, même si ce n’est pas affiché, c’est également le cas de la langue du document indiquée dans la balise html.

Dans cet exemple j’ai volontairement mis un grand nombre de métadonnées, mais si vous n’en avez pas l’usage vous pouvez ne pas les mettre. Je recommande cependant de toujours indiquer au moins la langue du document, le jeu de caractères et le titre.

Notez également que j’ai utilisé --pdf-variant dans la ligne de commande afin de forcer l’utilisation de PDF/UA qui est une norme d’accessibilité. Parce que non, tous les PDF ne se valent pas niveau accessibilité, alors soyons respectueux des autres et faisons attention à ça.

Un peu de style
#

HTML c’est bien, mais niveau design ce n’est pas folichon. Ajoutons donc un peu de CSS. Pour ceci, deux solutions possibles :

  1. tout mettre dans une balise HTML style comme un gros goret ;
  2. utiliser un ou plusieurs fichiers CSS séparés.

Étant des personnes respectables, nous opterons pour la seconde solution. Cependant, contrairement à ce dont nous avons l’habitude le web classique, nous n’utiliserons pas la balise HTML link. À la place, nous passons simplement un nouvel argument à la ligne de commande :

$ weasyprint --pdf-variant "pdf/ua-1" -s "test.css" "test.html" "test.pdf"

Je vous propose de commencer avec le fichier CSS suivant :

@page {
	size: A4 portrait;
	margin: 1.5cm 1cm 1.5cm 1cm;
}

* {
	margin: 0;
	padding: 0;
}

Si vous n’avez pas l’habitude de la règle CSS @page, c’est elle qui nous permet de spécifier des directives d’affichages pour nos pages. Elle va rapidement devenir votre meilleur allié. C’est par exemple grace à elle que nous pouvons utiliser le descripteur CSS size qui nous permet d’indiquer que nous créons un document A4 au format portrait.

Au niveau des marges, on commence par spécifier celles de la page elle-même. Ensuite, tel Léodagan, on crame les marges de tous les éléments afin de partir sur une base saine. Notez que, dans la mesure où nous éditons un document de taille fixe, il est possible et même recommandé d’exprimer les dimensions dans une valeur absolue tel que le centimètre (cm), le millimètre (mm) ou encore le Q (Q). Ce dernier remplace avantageusement le pixel (px) et le point (pt) car 1 Q est à peu près égal à 0,95 px soit environ 0,7 pt.

Toujours pour bien faire les choses, je recommande d’exprimer les tailles de polices en rem lorsque vous souhaitez le faire d’une manière relative. Bien entendu les valeurs absolues sont également possibles.

Quelques astuces
#

Normalement, si vous connaissez CSS vous devriez être en mesure de vous en sortir convenablement. Il ne me reste plus qu’à vous passer quelques astuces qui vous aideront à bien faire les choses.

Les variables
#

S’il y a une chose qui change la vie en CSS c’est la possibilité de définir des variables. Un cas très utile est d’utiliser :root afin de définir des valeurs que nous utiliseront régulièrement. Mon astuce ici est de définir les dimensions du document lui-même, les marges du document ainsi que les couleurs.

:root {
	--doc-width: 21cm;
	--doc-height: 29.7cm;

	--doc-margin-top: 1.5cm;
	--doc-margin-bottom: var(--doc-margin-top);
	--doc-margin-right: 1cm;
	--doc-margin-left: var(--doc-margin-right);

	--dark-grey: #444;

	--main-text-color: var(--dark-grey);
}

@page {
	size: A4 portrait;
	margin: var(--doc-margin-top) var(--doc-margin-right) var(--doc-margin-bottom) var(--doc-margin-left);
}

h1 {
	color: var(--main-text-color);
}

Importer des fichiers
#

La règle CSS @import fonctionne, ce qui nous permet de diviser notre code CSS en plusieurs fichiers. Par exemple, mettons nos définitions de variables concernant les dimensions dans un fichier et celles concernant les couleurs dans un autre. Et tant qu’à faire, mettons notre reset des marges dans encore un autre fichier.

@import url("dimensions.css");
@import url("colors.css");
@import url("reset.css");

Les calculs de dimensions
#

Vous avez pu voir que dans les exemples précédent j’ai défini les variables --doc-width et --doc-height dans les utiliser car j’ai utilisé la valeur spéciale A4 pour indiquer la taille du document. L’intérêt de faire ça est de pouvoir les utiliser pour des calculs de dimensions à l’aide de la fonction CSS calc().

Par exemple, afin d’obtenir une largeur égale à un tiers de la largeur de la page sans les marges, on peut écrire :

:root {
	--doc-width: 21cm;
	--doc-height: 29.7cm;

	--doc-margin-top: 1.5cm;
	--doc-margin-bottom: var(--doc-margin-top);
	--doc-margin-right: 1cm;
	--doc-margin-left: var(--doc-margin-right);

	--doc-inner-width: calc(var(--doc-width) - var(--doc-margin-right) - var(--doc-margin-left));
	--doc-inner-height: calc(var(--doc-height) - var(--doc-margin-top) - var(--doc-margin-bottom));
}

.trucmuche {
	width: calc(var(--doc-inner-width) / 3);
}

Si dans le web classique utiliser des dimensions absolues comme ça est une mauvaise pratique, ce n’est pas le cas dans le domaine du « print » où c’est une pratique tout à fait normale.

Différents types de pages
#

Jusque là tout va bien, alors attelons nous à la création de documents plus complexes comportant plusieurs types de pages différentes. Prenons l’exemple classique d’un rapport contenant une page de garde, un sommaire, le contenu en lui-même qui est chapitré et enfin d’éventuelles annexes.

Commençons par le plus simple, le HTML. Pour ceci, divisons tout en sections et utilisons des id pour identifier la page de garde et le sommaire ainsi que des classes pour les différentes parties du contenu ainsi que chaque annexe.

<!doctype html>
<html lang="fr">
	<head>
		<meta charset="utf-8">
		<title>Mon premier document PDF</title>
		<meta name="author" content="Toto">
		<meta name="description" content="Un PDF de test">
		<meta name="generator" content="super_script.sh">
		<meta name="keywords" content="pdf,weasyprint,html,css">
		<meta name="dcterms.created" content="2026-01-28T17:21:42+01:00">
		<meta name="dcterms.modified" content="2026-01-28T18:21:42+01:00">
	</head>
	<body>
		<section id="first-page">
			<h1>Rapport trucmuche</h1>
			<p>28 janvier 2026</p>
		</section>

		<section id="toc">
			<h2>Sommaire</h2>
			<ol>
				<li>Lorem ipsum</li>
				<li>Etiam aliquet</li>
				<li>Ut convallis convallis lacus non pretium</li>
			</ol>
		</section>

		<section class="main-content" id="chapter-1">
			<h2>Lorem ipsum</h2>
			<p>
			Lorem ipsum dolor sit amet, consectetur adipiscing elit. Ut facilisis ultricies
			lorem nec imperdiet. Ut malesuada tellus dapibus risus elementum, a aliquet arcu
			ornare. Integer ultrices, quam nec tempus suscipit, neque erat dapibus elit, non
			vestibulum quam mi sit amet erat. Integer id lectus a nisi convallis vehicula.
			Integer lobortis velit ut nulla consectetur mollis. Mauris pulvinar cursus neque
			vel finibus. Aenean aliquam pharetra quam, sit amet semper tortor imperdiet et.
			Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos
			himenaeos. Suspendisse sagittis lacus sit amet nulla porttitor ultricies
			eu in erat. Fusce nec enim lacus. Curabitur pretium tempor rhoncus. Proin
			accumsan tortor sed euismod aliquet. Quisque sit amet aliquam leo. Duis finibus
			vestibulum orci, nec finibus sapien congue id. Integer cursus posuere felis a
			iaculis. Curabitur iaculis elit sed mauris interdum eleifend.
			</p>
			<p>
			Vestibulum metus odio, dignissim sed nunc pulvinar, sagittis mattis leo. Proin
			ornare urna in nisi ultricies maximus. Sed lacinia, neque nec rhoncus mollis,
			dolor magna interdum tortor, id ultricies nisl ex vel mauris. Etiam a elementum
			tortor. Phasellus vel aliquam orci. Nam tincidunt massa sem. Aenean porta lectus
			in facilisis bibendum. Quisque non risus et purus dignissim posuere.
			</p>
			<p>
			Aliquam ut nulla et justo aliquet viverra molestie quis tortor. Morbi massa
			eros, blandit ac ipsum eu, convallis lacinia massa. Sed finibus mattis metus,
			at efficitur odio accumsan a. Phasellus et maximus nisl. Sed nisi odio,
			commodo at vulputate ac, rutrum ac eros. Sed sit amet augue laoreet, vulputate
			arcu accumsan, laoreet tortor. Quisque aliquet massa massa, ac vulputate ex
			molestie sit amet. Nunc vestibulum sem vel consectetur lacinia. Class aptent
			taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos.
			Pellentesque ac risus id arcu placerat consequat.
			</p>
		</section>

		<section class="main-content" id="chapter-2">
			<h2>Etiam aliquet</h2>
			<p>
			Etiam aliquet, nunc sed tristique feugiat, ex tortor dapibus nibh, eget
			hendrerit eros massa ut lectus. Morbi blandit, sem quis vehicula ultrices, ipsum
			eros egestas risus, ut aliquet augue augue et dolor. Curabitur vel elit congue,
			scelerisque risus vitae, sodales est. Etiam id mauris porttitor, finibus tortor
			id, feugiat leo. Pellentesque tincidunt mattis nisi, in cursus magna. Sed eget
			ipsum dui. Pellentesque habitant morbi tristique senectus et netus et malesuada
			fames ac turpis egestas. Nulla quis justo aliquet, dictum ante vel, mattis
			nulla. Mauris eu velit libero. Donec sit amet tristique dui, mattis fringilla
			ante. Fusce maximus ultrices quam blandit interdum. Cras aliquet, tortor blandit
			vehicula dapibus, quam lectus viverra ligula, non finibus tortor arcu non neque.
			Nulla scelerisque ligula et blandit dapibus. Pellentesque non interdum purus.
			Vivamus malesuada mi nisi, nec pretium justo gravida quis. Maecenas dapibus
			accumsan magna, a pretium justo feugiat eget.
			</p>
			<p>
			Aenean volutpat felis velit, eu porttitor eros commodo sed. Suspendisse turpis
			lectus, maximus sed fermentum vitae, ullamcorper nec enim. Mauris in lobortis
			sem. Praesent viverra, libero vitae ornare accumsan, enim orci blandit odio,
			ac sollicitudin ante lectus non augue. In rhoncus mi leo, vitae luctus odio
			lobortis in. Aliquam dapibus lacus lacus. Nam sed tempor urna. Etiam eleifend
			gravida euismod. Suspendisse potenti. Donec dictum turpis quis efficitur
			suscipit. Nullam congue mauris sit amet mauris auctor vehicula. Nunc quis mauris
			id risus suscipit sodales. Quisque vitae laoreet orci.
			</p>
			<p>
			Ut tincidunt lacus id augue vestibulum cursus. Nullam ac ante id dolor
			elementum blandit. Proin ullamcorper ipsum id sagittis efficitur. Phasellus
			faucibus, arcu vitae varius consequat, purus lectus tincidunt nibh, id
			pellentesque orci felis eu nunc. Quisque ut convallis lectus. Ut pretium sapien
			dolor, vitae egestas est sagittis vel. Quisque commodo libero id elit fermentum
			consequat. Morbi imperdiet placerat felis, sit amet luctus ante lobortis
			venenatis. Suspendisse eleifend, libero et vulputate blandit, ipsum est posuere
			dolor, sit amet rutrum elit metus id nibh. Aenean pellentesque quam nunc, eget
			tincidunt purus tempus vitae. Aliquam vehicula fermentum enim, vel rhoncus arcu
			tincidunt eu.
			</p>
		</section>

		<section class="main-content" id="chapter-3">
			<h2>Ut convallis convallis lacus non pretium</h2>
			<p>
			Ut convallis convallis lacus non pretium. Sed bibendum malesuada nulla vitae
			pulvinar. Morbi mollis libero faucibus, efficitur lectus in, consequat lacus.
			Maecenas eu ultrices dolor. Donec nisl dui, viverra dignissim dictum vel,
			suscipit dictum dui. Phasellus nunc diam, facilisis et ligula et, interdum
			auctor odio. Quisque urna justo, efficitur sed tristique id, scelerisque quis
			sapien. Duis posuere, nulla quis dapibus faucibus, est dolor accumsan felis,
			ac imperdiet ipsum mi id purus. Sed porta leo non risus ullamcorper, sit
			amet feugiat magna pellentesque. Proin pretium dignissim ornare. Curabitur
			auctor augue libero, nec convallis erat congue et. Nullam et pharetra felis.
			Pellentesque magna dui, hendrerit a accumsan id, ornare id lectus. Aliquam erat
			volutpat. Fusce maximus lacus mi, non vulputate arcu cursus a.
			</p>
			<p>
			Quisque maximus orci fringilla nibh lobortis consectetur. Ut sit amet metus
			mattis, pharetra nisl id, placerat velit. Praesent eu massa vel libero iaculis
			lobortis. Mauris ligula odio, auctor eu lacus ac, elementum tempus ipsum.
			Suspendisse elit arcu, pellentesque eget tempus sit amet, egestas non quam. In
			nec felis eget nulla porta vestibulum at vitae purus. Integer non massa urna.
			</p>
			<p>
			Aliquam mattis orci quis sem iaculis, vel mollis est sodales. Pellentesque
			lobortis lacinia velit at egestas. Ut sed leo ultrices, viverra augue sit amet,
			pulvinar turpis. Maecenas gravida tortor justo, id mattis magna lacinia eu.
			Nunc eu viverra lorem. Maecenas dapibus elit nibh, non gravida orci facilisis
			at. Mauris volutpat dolor eros, at consectetur mauris gravida a. Aliquam et
			venenatis enim. Ut vel rutrum justo. Integer congue et elit ut lacinia. Donec
			varius est in venenatis mattis.
			</p>
		</section>

		<section class="appendix" id="appendix-1">
			<h2>Annexe 1</h2>
			<p>
			Suspendisse in est in nisl fermentum pharetra. Curabitur at maximus enim.
			Phasellus pharetra gravida justo nec fermentum. Donec rhoncus sodales lectus,
			a varius turpis convallis ut. Donec lobortis metus ut justo vulputate consequat
			ut at est. Maecenas id hendrerit arcu. Maecenas eu tortor ut neque scelerisque
			gravida suscipit volutpat erat.
			</p>
		</section>

		<section class="appendix" id="appendix-2">
			<h2>Annexe 2</h2>
			<p>
			Ut et odio nec nunc sollicitudin ultrices pellentesque in erat. Sed suscipit,
			orci lobortis tempus egestas, nulla massa placerat neque, ac malesuada lorem
			sapien condimentum nisi. Mauris vel ex sapien. Aliquam sed lorem ut justo
			feugiat interdum. Fusce felis erat, interdum in arcu euismod, pellentesque
			feugiat massa. Phasellus et pharetra eros. Vestibulum nec suscipit orci, vitae
			tincidunt augue. Curabitur at nulla laoreet, fermentum justo non, feugiat erat.
			Nulla sapien elit, accumsan in est eget, porttitor interdum nisl.
			</p>
		</section>
	</body>
</html>

Si vous générez le PDF, vous vous apercevrez que tout est collé ensemble, c’est assez moche. Nous pourrions y aller tout schuss à styliser tout ça, mais pensons donc d’abord à la sémantique de chaque élément. En effet, la page de garde est une page à part, le sommaire également. D’ailleurs, même s’il peut s’étaler sur plusieurs pages, le contenu est également une page, de même que les annexes. Nous pouvons matérialiser ceci en CSS à l’aide de la propriété page.

#first-page {
	page: first-page;
}

#toc {
	page: toc;
}

.main-content {
	page: main-content;
	break-after: page;
}

.appendix {
	page: appendix;
	break-after: page;
}

Définir des pages différentes permet, au delà du sens sémantique, d’insérer un saut de page. Dans la mesure où tous les chapitres sont regroupés en une seule page sémantique, il n’y a pas de saute de page entre chaque chapitre. Pour les ajouter, nous utilisons break-after: page;. Faisons de même pour les annexes.

Avoir ainsi sémantiquement défini des pages nous permet d’utiliser @page pour leur appliquer un affichage spécifique. Par exemple, si les annexes ont besoin d’être en paysage alors que tout le reste doit rester en portrait :

@page appendix {
	size: A4 landscape;
}

Un vrai sommaire
#

C’est bien d’avoir un sommaire statique, mais c’est encore mieux lorsque l’on peut cliquer dessus pour directement accéder à la page en question. C’est exactement pour ça que nous avons utilisé des id pour chaque chapitre et annexe. Tout comme dans le web classique, les liens sur les ancres fonctionnent.

<section id="toc">
	<h2>Sommaire</h2>
	<ol>
		<li><a href="#chapter-1">Lorem ipsum</a></li>
		<li><a href="#chapter-2">Etiam aliquet</a></li>
		<li><a href="#chapter-3">Ut convallis convallis lacus non pretium</a></li>
	</ol>
</section>

C’est pas mal, mais en général dans un sommaire on affiche également le numéro de la page en question.

<section id="toc">
	<h2>Sommaire</h2>
	<ol>
		<li><a href="#chapter-1">Lorem ipsum</a><a class="page-nb" href="#chapter-1"></a></li>
		<li><a href="#chapter-2">Etiam aliquet</a><a class="page-nb" href="#chapter-2"></a></li>
		<li><a href="#chapter-3">Ut convallis convallis lacus non pretium</a><a class="page-nb" href="#chapter-3"></a></li>
	</ol>
</section>
#toc a.page-nb::before {
	content: target-counter(attr(href), page);
}

Oui je sais, là on entre dans la magie noire du CSS, à tel point qu’au jour de l’écriture de cet article target-counter n’est pas documenté dans la documentation MDN. Ça nous permet de récupérer le compteur, ici page dont le but est de contenir un numéro de page, qui s’applique à une cible, ici attr(href) donc l’attribut href du lien.

En-tête et pied de page
#

Le plus souvent il est souhaitable de numéroter les pages d’un document. En revanche, nous ne souhaitons numéroter que les pages du contenu, donc certainement pas la page de garde par exemple. Comme vous pouvez vous en douter, nous allons donc retrouver à la fois @page et les compteurs.

@page main-content {
	@bottom-right {
		content: "Page " counter(page) " / " counter(pages);
	}
}

Oui, c’est intégralement du CSS, nous n’avons eu besoin d’ajouter aucun élément HTML. Je le répète : le CSS c’est magique. Notons que si page est le compteur indiquant le numéro de la page courante, pages est le compteur indiquant le nombre total de pages. Notons également l’utilisation de la règle de marge @bottom-right mais il en existe beaucoup d’autres.

Lorsque l’on imprime un livre, on souhaite généralement avoir le numéro de page d’un côté puis de l’autre afin qu’il soit toujours vers l’extérieur. C’est là qu’entrent en jeu les sélecteurs de page :left et :right.

@page main-content :left {
	@bottom-left {
		content: counter(page);
	}
}

@page main-content :right {
	@bottom-right {
		content: counter(page);
	}
}

Bon, des numéros de page c’est bien gentil, mais comment fait-on lorsqu’on a besoin d’insérer quelque chose de plus complexe ? Pour ça, commençons par créer un footer en HTML que nous allons placer directement tout en haut (j’insiste) du body.

<body>
		<footer id="main-content-footer">
			Société <b>Exemple</b> SARL<br>
			Email&nbsp;: contact@example.com
		</footer>

		<!--
			Le reste du HTML.
		-->
</body>

Et là on utilise le CSS pour le mettre en position: running() tout en lui passant un nom qui servira d’identifiant de référence. Cette opération va l’enlever du flux courant et nous pouvons donc l’utiliser dans @page.

@page main-content {
	@bottom-center {
		content: element(main-content-footer);
	}
}

#main-content-footer {
	position: running(main-content-footer);
}

Et voilà, nous avons un joli pied de page écrit en HTML qui s’affiche en bas et au centre de chaque page du contenu principal, tout ça juste en le mettant en position de running.

Détournement se voulant humoristique de deux captures d'écran du film « Forest Gump » l'une au dessus l'autre. La première montre Jenny en train de crier « Run footer, run! ». La seconde montre Forest en train de courir avec l'inscription « <footer></footer> » sur son visage.
Run footer, run!

Des icônes
#

Dans la mesure où nous avons un document HTML et CSS, nous pouvons utiliser toutes les ressources auxquelles l’on a généralement accès pour du développement web. Par exemple, des jeux d’icônes. Personnellement j’aime beaucoup Remix Icon mais il en existe beaucoup d’autres.

Intégrer des trucs
#

S’il est généralement possible d’inclure des éléments externes tels que des images ou des polices d’écriture à l’aide d’un lien relatif ou absolu pointant vers le fichier concerné, parfois ce n’est malheureusement pas possible. Je pense par exemple à une application qui devrait automatiquement générer un PDF contenant des images stockées dans une base de donnée.

Dans ce genre de cas, il est possible d’utiliser une URL spéciale :

  • le « protocole », en l’occurrence data ;
  • le séparateur : ;
  • le type MIME du fichier, par exemple image/png ;
  • le séparateur ; ;
  • base64 afin de dire qu’on va passer le fichier en base46 ;
  • le séparateur , ;
  • en enfin le contenu du fichier en base64.

Par exemple, voici comment insérer une image PNG en fond d’un élément sans sortir de CSS :

.img-bg {
	background-image: url("");
}

Ça fonctionne partout où la fonction CSS url est supportée, donc c’est bien plus large que les images. C’est également utilisable directement en HTML, par exemple dans l’attribut src d’une balise img ou autre.

Notons que les fichiers SVG n’ont pas besoin d’être mis en base64 pour être directement inclus, c’est tout à fait possible de les insérer directement dans du HTML.

Styliser les compteurs
#

Les compteurs CSS servent à bien plus de choses qu’à compter ou afficher des nombres. En fait ce sont plus des suites de symboles qu’autre chose. Il est donc possible d’utiliser @counter-style afin de personaliser tout ça, par exemple avoir un « compteur » qui utilise les émojis représentant les différents cycles de la lune afin de boucler dessus.

@counter-style moon-cycle {
	system: cyclic;
	symbols: "🌑" "🌒" "🌓" "🌔" "🌕️" "🌖" "🌗" "🌘";
}

Pour revenir dans le domaine des nombres, il est par exemple possible de les afficher sur 2 chiffres avec un 0 devant si besoin.

@counter-style decimal-pad-2 {
	system: numeric;
	symbols: "0" "1" "2" "4" "5" "6" "7" "8" "9";
	pad: 2 "0";
}

Pour utiliser votre style personnalisé, précisez-le simplement dans l’appel à counter. Les listes intégrant un compteur pour leurs éléments, vous pouvez également utiliser votre style personnalisé dans list-style-type.

@page main-content {
	@bottom-right {
		content: counter(page, decimal-pad-2);
	}
}

ol {
	list-style-type: moon-cycle;
}

Créer ses propres compteurs
#

Les compteurs pré-définis c’est bien, mais parfois il est utile de créer les siens. Comme ça nous pouvons par exemple enfin arrêter d’utiliser des listes ordonnées dans votre table des matières. À la place on va se faire un petit style sympa à base de grid.

<section id="toc">
	<h2>Sommaire</h2>

	<div class="toc-elem">
		<a class="toc-elem-title" href="#chapter-1">Lorem ipsum</a>
		<span></span>
		<a class="toc-elem-page-nb" href="#chapter-1"></a>
	</div>

	<div class="toc-elem">
		<a class="toc-elem-title" href="#chapter-2">Etiam aliquet</a>
		<span></span>
		<a class="toc-elem-page-nb" href="#chapter-2"></a>
	</div>

	<div class="toc-elem">
		<a class="toc-elem-title" href="#chapter-3">Ut convallis convallis lacus non pretium</a>
		<span></span>
		<a class="toc-elem-page-nb" href="#chapter-3"></a>
	</div>
</section>

De là c’est assez simple : on commence par initialiser un compteur à 0 et à lui donner un nom à l’aide de counter-reset. Ensuite, avant chaque nouvel élément du sommaire, on l’incrémente de 1 à l’aide de counter-increment puis on l’affiche.

#toc {
	counter-reset: toc-lvl-1;
}

.toc-elem::before {
	counter-increment: toc-lvl-1;
	content: counter(toc-lvl-1);
	margin-right: 2mm;
}

.toc-elem {
	display: grid;
	grid-template-columns: max-content max-content auto max-content;
}

.toc-elem > span {
	border-bottom: 1Q dotted black;
}

.toc-elem-page-nb::before {
	content: target-counter(attr(href), page);
}

Les grids c’est le petit bonus. Je ne voulais pas faire une section dédiée car malheureusement Weasyprint est encore en train de progresser sur le sujet et vous risquez de rapidement en rencontrer les limites.