A place to cache linked articles (think custom and personal wayback machine)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

index.html 25KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. <!doctype html><!-- This is a valid HTML5 document. -->
  2. <!-- Screen readers, SEO, extensions and so on. -->
  3. <html lang="fr">
  4. <!-- Has to be within the first 1024 bytes, hence before the `title` element
  5. See: https://www.w3.org/TR/2012/CR-html5-20121217/document-metadata.html#charset -->
  6. <meta charset="utf-8">
  7. <!-- Why no `X-UA-Compatible` meta: https://stackoverflow.com/a/6771584 -->
  8. <!-- The viewport meta is quite crowded and we are responsible for that.
  9. See: https://codepen.io/tigt/post/meta-viewport-for-2015 -->
  10. <meta name="viewport" content="width=device-width,initial-scale=1">
  11. <!-- Required to make a valid HTML5 document. -->
  12. <title>Ajout d’un module de recherche pour Hugo (archive) — David Larlet</title>
  13. <meta name="description" content="Publication mise en cache pour en conserver une trace.">
  14. <!-- That good ol' feed, subscribe :). -->
  15. <link rel="alternate" type="application/atom+xml" title="Feed" href="/david/log/">
  16. <!-- Generated from https://realfavicongenerator.net/ such a mess. -->
  17. <link rel="apple-touch-icon" sizes="180x180" href="/static/david/icons2/apple-touch-icon.png">
  18. <link rel="icon" type="image/png" sizes="32x32" href="/static/david/icons2/favicon-32x32.png">
  19. <link rel="icon" type="image/png" sizes="16x16" href="/static/david/icons2/favicon-16x16.png">
  20. <link rel="manifest" href="/static/david/icons2/site.webmanifest">
  21. <link rel="mask-icon" href="/static/david/icons2/safari-pinned-tab.svg" color="#07486c">
  22. <link rel="shortcut icon" href="/static/david/icons2/favicon.ico">
  23. <meta name="msapplication-TileColor" content="#f7f7f7">
  24. <meta name="msapplication-config" content="/static/david/icons2/browserconfig.xml">
  25. <meta name="theme-color" content="#f7f7f7" media="(prefers-color-scheme: light)">
  26. <meta name="theme-color" content="#272727" media="(prefers-color-scheme: dark)">
  27. <!-- Documented, feel free to shoot an email. -->
  28. <link rel="stylesheet" href="/static/david/css/style_2021-01-20.css">
  29. <!-- See https://www.zachleat.com/web/comprehensive-webfonts/ for the trade-off. -->
  30. <link rel="preload" href="/static/david/css/fonts/triplicate_t4_poly_regular.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: light), (prefers-color-scheme: no-preference)" crossorigin>
  31. <link rel="preload" href="/static/david/css/fonts/triplicate_t4_poly_bold.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: light), (prefers-color-scheme: no-preference)" crossorigin>
  32. <link rel="preload" href="/static/david/css/fonts/triplicate_t4_poly_italic.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: light), (prefers-color-scheme: no-preference)" crossorigin>
  33. <link rel="preload" href="/static/david/css/fonts/triplicate_t3_regular.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: dark)" crossorigin>
  34. <link rel="preload" href="/static/david/css/fonts/triplicate_t3_bold.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: dark)" crossorigin>
  35. <link rel="preload" href="/static/david/css/fonts/triplicate_t3_italic.woff2" as="font" type="font/woff2" media="(prefers-color-scheme: dark)" crossorigin>
  36. <script>
  37. function toggleTheme(themeName) {
  38. document.documentElement.classList.toggle(
  39. 'forced-dark',
  40. themeName === 'dark'
  41. )
  42. document.documentElement.classList.toggle(
  43. 'forced-light',
  44. themeName === 'light'
  45. )
  46. }
  47. const selectedTheme = localStorage.getItem('theme')
  48. if (selectedTheme !== 'undefined') {
  49. toggleTheme(selectedTheme)
  50. }
  51. </script>
  52. <meta name="robots" content="noindex, nofollow">
  53. <meta content="origin-when-cross-origin" name="referrer">
  54. <!-- Canonical URL for SEO purposes -->
  55. <link rel="canonical" href="https://lord.re/posts/206-recherche-pour-un-blog-statique/">
  56. <body class="remarkdown h1-underline h2-underline h3-underline em-underscore hr-center ul-star pre-tick" data-instant-intensity="viewport-all">
  57. <article>
  58. <header>
  59. <h1>Ajout d’un module de recherche pour Hugo</h1>
  60. </header>
  61. <nav>
  62. <p class="center">
  63. <a href="/david/" title="Aller à l’accueil"><svg class="icon icon-home">
  64. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-home"></use>
  65. </svg> Accueil</a> •
  66. <a href="https://lord.re/posts/206-recherche-pour-un-blog-statique/" title="Lien vers le contenu original">Source originale</a>
  67. </p>
  68. </nav>
  69. <hr>
  70. <p>Régulièrement j'ai des gens qui ne retrouvent pas un article que j'ai écrit.
  71. Et moi le premier, je cherche souvent pour savoir où j'ai bien pu parler d'un truc.
  72. Et c'est vrai qu'étant donné qu'il y a de plus en plus de contenu sur mon blog, c'est forcément c'est de plus en plus complexe de se rappeler ou de retrouver une page précise.</p>
  73. <p>Étant donné que j'ai déjà ouvert toutes les pages dans mon navigateur, je me base sur l'autocomplétion du navigateur mais c'est loin d'etre parfait.</p>
  74. <p>Jusqu'à présent je recommandai du coup de se rendre sur la <a href="https://lord.re/mono/">monopage</a>.
  75. Cette page contient absolument tout le contenu du site, il suffit donc d'attendre qu'elle charge puis utiliser la fonction recherche du navigateur (<kbd>Ctrl-f</kbd>) mais c'est un usage qui est au final marginal.
  76. Ça fonctionne mais c'est peu plaisant et puis la page est de plus en plus longue à charger vu qu'elle pèse désormais près de 3Mo de HTML (bon une fois gzippé ça tombe à 800Ko).</p>
  77. <p>L'autre technique est tout simplement de chercher via un moteur de recherche conventionnel comme le fameux <strong>DuckDuckGo</strong> ou autre moteur alternatif.
  78. Mon site est pas mal crawlé et donc plutôt bien indexé donc ça fonctionne pas trop mal.
  79. Mais c'est quand même dommage de dépendre de la bonne volonté des moteurs de recherche pour une fonctionnalité que j'aimerai offrir moi-même dans ma quête d'autonomie et d'indépendance.</p>
  80. <h2 id="cahier-des-charges">Cahier des charges</h2>
  81. <p>Donc je voulais un truc adapté à mes contraintes.</p>
  82. <ul>
  83. <li>
  84. <p>La première c'est d'être <strong>statique</strong>, c'est-à-dire que je n'ai rien à installer sur le serveur hébergeant le site.
  85. Pas de PHP, pas de node, pas …</p>
  86. </li>
  87. <li>
  88. <p>La seconde c'est que ce soit du <strong>logiciel Libre</strong> simple à utiliser.
  89. Encore que la simplicité est très négociable si ça fait bien le taf.
  90. Le but étant de ne pas avoir à installer cinquante trucs pour que ça fonctionne.</p>
  91. </li>
  92. <li>
  93. <p>La troisième c'est que pour l'utilisateur ça soit <strong>pratique</strong> donc si c'est plus complexe ou moins performant que de fouiller manuellement la monopage ça sert à rien.</p>
  94. </li>
  95. </ul>
  96. <h2 id="solution-sélectionnée">Solution sélectionnée</h2>
  97. <p>Il y a trois semaines, lors de ma navigation habituelle sur Hacker News je tombe sur un ptit projet qui semble fait pour me titiller : <strong><a href="https://github.com/tinysearch/tinysearch">Tinysearch</a></strong></p>
  98. <p><em>C'est codé en rust, nécessite un bout de javascript + un bout de webassembly et c'est tout.</em>
  99. Il est prévu pour être mis en place dans mon cas d'usage (les sites statiques) donc je n'aurai pas besoin de tordre un outil pour transformer une pince coupante en pince croco…</p>
  100. <p>Après je n'ai pas cherché plus, je suis tombé par hasard sur ça et ça matchait vraiment bien.
  101. Si ça se trouve il existe mieux ailleurs mais pour le moment je saurai m'en contenter.</p>
  102. <h2 id="donc-la-marche-à-suivre-">Donc la marche à suivre ?</h2>
  103. <p>Alors d'un point de vue utilisateur c'est simplement un petit bout de javascript, qui va lancer un bout de webassembly qui contient le moteur de recherche + son index.</p>
  104. <p>Il faut donc au préalable (après la génération du site) créer (tout du moins le remettre à jour) cet index.</p>
  105. <p>Il faut également intégrer dans les pages (ou dans une seule comme j'ai préféré) ajouter un peu de html+js pour charger ça et voilà.
  106. C'est donc vraiment pas intrusif pour le site.</p>
  107. <h2 id="création-de-lindex">Création de l'index</h2>
  108. <p>Alors ce cher <strong>Tinysearch</strong> a besoin d'un index mais ce n'est pas lui qui va le créer.
  109. Enfin il va se créer son index à lui en touillant tout à sa sauce mais il faut lui fournir les données brutes.</p>
  110. <p>Ici, les données brutes, c'est un fichier json contenant la liste de tous les articles.
  111. Pour chaque article il veut un <strong>titre</strong>, une <strong>url</strong> et le <strong>contenu</strong>.</p>
  112. <p>Il faut donc demander à notre cher <strong>Hugo</strong> de nous générer un fichier json avec la bonne syntaxe.
  113. Bon alors sur le <a href="https://github.com/tinysearch/tinysearch">github de Tinysearch</a> on trouve un peu de doc dont une toute fraîche détaillant la marche à suivre pour Hugo mais je l'ai quelque peu modifiée (<del>je la proposerai ptet en retour si j'ai pas de mauvais retours</del> c'est fait).</p>
  114. <p>On va donc devoir créer 1 fichier définissant la structure du fichier.</p>
  115. <details><summary>layout/_default/list.json.json (oui oui, deux fois .json)</summary>
  116. <div class="highlight"><pre tabindex="0"><code class="language-golang" data-lang="golang"><span><span>[
  117. </span></span><span><span>{{ <span>range</span> <span>$</span><span>index</span> , <span>$</span><span>e</span> <span>:=</span> .<span>Site</span>.<span>RegularPages</span> }}{{ <span>if</span> <span>$</span><span>index</span> }}, {{<span>end</span>}}{{ <span>dict</span> <span>"title"</span> .<span>Title</span> <span>"url"</span> .<span>Permalink</span> <span>"body"</span> .<span>Plain</span> | <span>jsonify</span> }}{{<span>end</span>}}
  118. </span></span><span><span>]</span></span></code></pre></div>
  119. </details>
  120. <p>Il faut maintenant dire à Hugo de générer ce fichier pour la home uniquement, on a pas besoin de générer un json pour chacune des pages/liste.</p>
  121. <p>Il faut donc éditer la config globale :</p>
  122. <details><summary>extrait du config.toml</summary>
  123. <div class="highlight"><pre tabindex="0"><code class="language-toml" data-lang="toml"><span><span>[<span>outputs</span>]
  124. </span></span><span><span> <span>home</span> = [<span>"html"</span>,<span>"rss"</span>,<span>"json"</span>]</span></span></code></pre></div>
  125. </details>
  126. <p>Donc là, pour la home, il générera le html habituel, le rss associé mais également le json.</p>
  127. <p>Voilà maintenant à la racine de votre site ouaib vous trouverez votre fichier index.json</p>
  128. <details><summary>extrait du index.json généré</summary>
  129. <div class="highlight"><pre tabindex="0"><code class="language-json" data-lang="json"><span><span>[
  130. </span></span><span><span> {
  131. </span></span><span><span> <span>"body"</span>: <span>"blabla"</span>,
  132. </span></span><span><span> <span>"title"</span>: <span>"Recherche"</span>,
  133. </span></span><span><span> <span>"url"</span>: <span>"https://lord.re/recherche/"</span>
  134. </span></span><span><span> },
  135. </span></span><span><span> {
  136. </span></span><span><span> <span>"body"</span>: <span>"encore du blabla"</span>,
  137. </span></span><span><span> <span>"title"</span>: <span>"Event Horizon"</span>,
  138. </span></span><span><span> <span>"url"</span>: <span>"https://lord.re/visionnages/event-horizon/"</span>
  139. </span></span><span><span> }
  140. </span></span><span><span>]</span></span></code></pre></div>
  141. </details>
  142. <p>Voilà pour la partie création d'index.
  143. Vous remarquerez que ce fichier json peut vite être assez gros.
  144. Dans mon cas, il pèse 1.9Mo (710Ko gzippé) à comparer à la monopage qui fait 2.9Mo (835Ko gzippée).</p>
  145. <h2 id="utilisation-de-tinysearch">Utilisation de Tinysearch</h2>
  146. <p>Bon je zappe la partie installation : c'est un logiciel très jeune en rust qui n'est probablement dans aucune distribution linux pour le moment.
  147. Je l'ai direct git cloné depuis github et j'ai même touillé une variable dans le code pour le faire fonctionner mais ça devrait être amélioré très bientôt cette partie-là.</p>
  148. <p>Bref, la seule chose à faire est de lui donner à manger votre <em>index.json</em>.
  149. Mettez-vous dans un dossier vierge créé pour l'occasion car il va pondre quelques fichiers.</p>
  150. <p><kbd>tinysearch /tmp/www/public/fr/index.json</kbd>.
  151. Et là ça va voltiger dans tous les sens.
  152. Pour le moment, à chaque fois que vous l'invoquerez il va récupérer des paquets rust, compiler quelques morceaux, lire votre index compiler un binaire en rust, le transpiler en webassembly et vous générer donc un fichier <em>js</em>, un <em>wasm</em> (contenant entre autre l'index dans son format à lui) quelques fichiers en plus qui ne seront pas utiles (sauf peut-être le demo.html pour tester vite fait).</p>
  153. <p>Donc vous pouvez y récupérer le fichier js et wasm et le coller dans votre Hugo où bon vous semble, j'ai choisi de foutre ça dans <em>static/js/</em> alors que bon le fichier wasm en vrai n'est pas statique car il changera à chaque nouvelle génération d'index.</p>
  154. <p>Ces deux fichiers peuvent être gzippés (et c'est très fortement recommandé).
  155. Si votre site est assez conséquent le fichier wasm peut rapidement devenir un peu gros mais il se compresse vraiment très bien.
  156. Pour info, mon fichier <em>tinysearch_engine_bg.wasm</em> pèose 2Mo (mais 330Ko gzippé).</p>
  157. <h2 id="création-dune-page-de-recherche">Création d'une page de recherche</h2>
  158. <p>J'aurai pu foutre la recherche direct dans la sidebar mais ça aurait alourdi toutes les pages du site alors que la recherche ne sera pas utilisée à tous les coups.</p>
  159. <p>J'ai donc créé une page grâce à <kbd>hugo new recherche.md</kbd> et à l'intérieur les trois quarts sont du html.</p>
  160. <details><summary>extrait de content/recherche.md</summary>
  161. <div class="highlight"><pre tabindex="0"><code class="language-html" data-lang="html"><span><span>&lt;<span>section</span> <span>class</span><span>=</span><span>"ideas"</span>&gt;
  162. </span></span><span><span>&lt;<span>article</span>&gt;
  163. </span></span><span><span>Vous êtes tout tristouille en train de chercher une page en particulier dans mon ptit bordel ?
  164. </span></span><span><span>
  165. </span></span><span><span>Allez on va tenter de la trouver ensemble !
  166. </span></span><span><span>
  167. </span></span><span><span> &lt;<span>script</span> <span>type</span><span>=</span><span>"module"</span>&gt;
  168. </span></span><span><span> <span>import</span> { <span>search</span>, <span>default</span> <span>as</span> <span>init</span> } <span>from</span> <span>'https://lord.re/js/tinysearch_engine.js'</span>;
  169. </span></span><span><span> window.<span>search</span> <span>=</span> <span>search</span>;
  170. </span></span><span><span>
  171. </span></span><span><span> <span>async</span> <span>function</span> <span>run</span>() {
  172. </span></span><span><span> <span>await</span> <span>init</span>(<span>'https://lord.re/js/tinysearch_engine_bg.wasm'</span>);
  173. </span></span><span><span> }
  174. </span></span><span><span>
  175. </span></span><span><span> <span>run</span>();
  176. </span></span><span><span> &lt;/<span>script</span>&gt;
  177. </span></span><span><span>
  178. </span></span><span><span> &lt;<span>script</span>&gt;
  179. </span></span><span><span> <span>function</span> <span>doSearch</span>() {
  180. </span></span><span><span> <span>let</span> <span>value</span> <span>=</span> document.<span>getElementById</span>(<span>"recherche"</span>).<span>value</span>;
  181. </span></span><span><span> <span>const</span> <span>arr</span> <span>=</span> <span>search</span>(<span>value</span>, <span>21</span>);
  182. </span></span><span><span> <span>let</span> <span>ul</span> <span>=</span> document.<span>getElementById</span>(<span>"results"</span>);
  183. </span></span><span><span> <span>ul</span>.<span>innerHTML</span> <span>=</span> <span>""</span>;
  184. </span></span><span><span>
  185. </span></span><span><span> <span>for</span> (<span>i</span> <span>=</span> <span>0</span>; <span>i</span> <span>&lt;</span> <span>arr</span>.<span>length</span>; <span>i</span><span>++</span>) {
  186. </span></span><span><span> <span>var</span> <span>li</span> <span>=</span> document.<span>createElement</span>(<span>"li"</span>);
  187. </span></span><span><span>
  188. </span></span><span><span> <span>let</span> <span>elem</span> <span>=</span> <span>arr</span>[<span>i</span>];
  189. </span></span><span><span> <span>let</span> <span>elemlink</span> <span>=</span> document.<span>createElement</span>(<span>'a'</span>);
  190. </span></span><span><span> <span>elemlink</span>.<span>innerHTML</span> <span>=</span> <span>elem</span>[<span>0</span>];
  191. </span></span><span><span> <span>elemlink</span>.<span>setAttribute</span>(<span>'href'</span>, <span>elem</span>[<span>1</span>]);
  192. </span></span><span><span> <span>li</span>.<span>appendChild</span>(<span>elemlink</span>);
  193. </span></span><span><span>
  194. </span></span><span><span> <span>ul</span>.<span>appendChild</span>(<span>li</span>);
  195. </span></span><span><span> }
  196. </span></span><span><span> }
  197. </span></span><span><span> &lt;/<span>script</span>&gt;
  198. </span></span><span><span>
  199. </span></span><span><span> &lt;<span>input</span> <span>type</span><span>=</span><span>"text"</span> <span>id</span><span>=</span><span>"recherche"</span> <span>onkeyup</span><span>=</span><span>"doSearch()"</span> <span>style</span><span>=</span><span>"margin:1em;padding:1em;font-size:2rem;background-color:#222;color:#ddd;border-radius:0.3em;border:none;width:90%;box-shadow:inset 0 0 1em #111;"</span> <span>placeholder</span><span>=</span><span>"recherche"</span>&gt;
  200. </span></span><span><span> &lt;<span>ul</span> <span>id</span><span>=</span><span>"results"</span>&gt;
  201. </span></span><span><span> &lt;/<span>ul</span>&gt;</span></span></code></pre></div>
  202. </details>
  203. <p>Donc on voit qu'il y a le js et le wasm qui sont chargés (faudra que vous adaptiez les url), j'ai un chouilla stylisé la boite d'input et voilà.</p>
  204. <h2 id="profit-">Profit !</h2>
  205. <p>Normailement c'est tout bon.</p>
  206. <p>Enfin normalement si vous avez pas joué les ptits malins avec des <abbr title="Content Security Policy">CSP</abbr>.
  207. Visiblement le webassembly (wasm) nécessite d'avoir <kbd>script-src 'unsafe-eval'</kbd> pour accepter de tourner sinon vous aurez une erreur étrange dans la console.</p>
  208. <p>En gros <strong>Tinysearch</strong> se base sur votre index exhaustif et utilise un <em>bloom filter</em> (j'y connais rien dans ce domaine) ce qui lui permet d'avoir une corresponsdance entre un mot et du contenu.
  209. L'avantage c'est que potentiellement ce nouvel index peut-être vraiment petit par rapport à la taille de données indexées.
  210. L'inconvénient c'est que c'est assez approximatif, certains termes peuvent donner des résultats faux-positifs et aussi quelques faux-négatifs (mais plus rare).</p>
  211. <p>Vu que cet index est transféré au navigateur web et que c'est également le navigateur qui doit s'en dépatouiller lors d'une recherche, on ne peut pas se permettre d'avoir un fichier trop lourdingue.
  212. Du coup c'est un compromis à trouver, pour l'instant c'est pas configurable (tout du moins il faut changer le code et recompiler <strong>Tinysearch</strong>) et c'est fourni avec une valeur ridiculement petite (tout du moins pour le contenu que j'ai).</p>
  213. <p>J'ai passé le <em><abbr title="une valeur qui se trouve dans bin/src/storage.rs à la ligne 68">magic number</abbr></em> de 10 par défaut à 1024.
  214. Le fichier wasm généré est désormais de 2Mo cependant il se gzip à environ 350Ko ce qui est de suite bien plus raisonnable.</p>
  215. <p>Si vous voulez des explications plus en détails sur l'aspect technique de Tinysearch, un ptit tour sur <a href="https://endler.dev/2019/tinysearch/">le blog du créateur du soft</a> où il explique un peu tout.
  216. C'est intéressant mais très technique et en anglais.</p>
  217. <h2 id="pensées-concernant-tinysearch">Pensées concernant Tinysearch</h2>
  218. <p>Le logiciel est <em>vraiment jeune pour le moment</em> et s'oriente d'ailleurs vers une première sortie en version 1.
  219. Du coup ça implique que le code bouge pas mal et que les devs sont vraiment très à l'écoute et réactif.</p>
  220. <p>Il est très probable que son fonctionnement change dans les semaines à venir.
  221. Pour l'instant, on a dépassé le stade du prototype mais on est loin d'un logiciel fini et mature.
  222. Ils sont conscients que le fonctionnement actuel n'est pas optimal.</p>
  223. <p>Il faut pour l'instant modifier le code à la main et recompiler le soft afin de gérer le compromis d'efficacité/poids de l'index.</p>
  224. <p>Ils savent que télécharger et compiler tout un tas de truc lors de son utilisation est pas user-friendly, pas optimisé du tout.
  225. Bref, ce que je raconte aujourd'hui ne sera ptet plus du tout d'actualité d'ici quelque temps.</p>
  226. <h2 id="mettre-à-jour-lindex">Mettre à jour l'index</h2>
  227. <p>Bon maintenant ça veut dire qu'à chaque fois que je rajoute du nouveau contenu je dois désormais recréer l'index.
  228. Ça mériterait d'être placé dans le hook git qui m'automatise la publication du blog, cela dit le logiciel bougeant encore pas mal, je ne vais pas l'optimiser tout de suite.</p>
  229. <h2 id="à-vous-cognacq-jay--à-vous-les-studios-">À vous Cognacq Jay ! À vous les studios !</h2>
  230. </article>
  231. <hr>
  232. <footer>
  233. <p>
  234. <a href="/david/" title="Aller à l’accueil"><svg class="icon icon-home">
  235. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-home"></use>
  236. </svg> Accueil</a> •
  237. <a href="/david/log/" title="Accès au flux RSS"><svg class="icon icon-rss2">
  238. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-rss2"></use>
  239. </svg> Suivre</a> •
  240. <a href="http://larlet.com" title="Go to my English profile" data-instant><svg class="icon icon-user-tie">
  241. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-user-tie"></use>
  242. </svg> Pro</a> •
  243. <a href="mailto:david%40larlet.fr" title="Envoyer un courriel"><svg class="icon icon-mail">
  244. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-mail"></use>
  245. </svg> Email</a> •
  246. <abbr class="nowrap" title="Hébergeur : Alwaysdata, 62 rue Tiquetonne 75002 Paris, +33184162340"><svg class="icon icon-hammer2">
  247. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-hammer2"></use>
  248. </svg> Légal</abbr>
  249. </p>
  250. <template id="theme-selector">
  251. <form>
  252. <fieldset>
  253. <legend><svg class="icon icon-brightness-contrast">
  254. <use xlink:href="/static/david/icons2/symbol-defs-2021-12.svg#icon-brightness-contrast"></use>
  255. </svg> Thème</legend>
  256. <label>
  257. <input type="radio" value="auto" name="chosen-color-scheme" checked> Auto
  258. </label>
  259. <label>
  260. <input type="radio" value="dark" name="chosen-color-scheme"> Foncé
  261. </label>
  262. <label>
  263. <input type="radio" value="light" name="chosen-color-scheme"> Clair
  264. </label>
  265. </fieldset>
  266. </form>
  267. </template>
  268. </footer>
  269. <script src="/static/david/js/instantpage-5.1.0.min.js" type="module"></script>
  270. <script>
  271. function loadThemeForm(templateName) {
  272. const themeSelectorTemplate = document.querySelector(templateName)
  273. const form = themeSelectorTemplate.content.firstElementChild
  274. themeSelectorTemplate.replaceWith(form)
  275. form.addEventListener('change', (e) => {
  276. const chosenColorScheme = e.target.value
  277. localStorage.setItem('theme', chosenColorScheme)
  278. toggleTheme(chosenColorScheme)
  279. })
  280. const selectedTheme = localStorage.getItem('theme')
  281. if (selectedTheme && selectedTheme !== 'undefined') {
  282. form.querySelector(`[value="${selectedTheme}"]`).checked = true
  283. }
  284. }
  285. const prefersColorSchemeDark = '(prefers-color-scheme: dark)'
  286. window.addEventListener('load', () => {
  287. let hasDarkRules = false
  288. for (const styleSheet of Array.from(document.styleSheets)) {
  289. let mediaRules = []
  290. for (const cssRule of styleSheet.cssRules) {
  291. if (cssRule.type !== CSSRule.MEDIA_RULE) {
  292. continue
  293. }
  294. // WARNING: Safari does not have/supports `conditionText`.
  295. if (cssRule.conditionText) {
  296. if (cssRule.conditionText !== prefersColorSchemeDark) {
  297. continue
  298. }
  299. } else {
  300. if (cssRule.cssText.startsWith(prefersColorSchemeDark)) {
  301. continue
  302. }
  303. }
  304. mediaRules = mediaRules.concat(Array.from(cssRule.cssRules))
  305. }
  306. // WARNING: do not try to insert a Rule to a styleSheet you are
  307. // currently iterating on, otherwise the browser will be stuck
  308. // in a infinite loop…
  309. for (const mediaRule of mediaRules) {
  310. styleSheet.insertRule(mediaRule.cssText)
  311. hasDarkRules = true
  312. }
  313. }
  314. if (hasDarkRules) {
  315. loadThemeForm('#theme-selector')
  316. }
  317. })
  318. </script>
  319. </body>
  320. </html>