diff --git a/config.toml b/config.toml
index 1fff6c55f..9827a9d86 100644
--- a/config.toml
+++ b/config.toml
@@ -115,6 +115,9 @@ language_name.es = "Español"
# The default setting is the light theme.
theme_switcher = true
+# Add a "copy" button to codeblocks (loads ~700 bytes of JavaScript).
+copy_button = true
+
# Date format used when listing posts (main page, /blog section, tag posts list…)
# Default is "6th July 2049" in English and "%d %B %Y" in other languages.
long_date_format = "%d %B %Y"
diff --git a/content/blog/almost-no-js.ca.md b/content/blog/almost-no-js.ca.md
deleted file mode 100644
index 2e5c75978..000000000
--- a/content/blog/almost-no-js.ca.md
+++ /dev/null
@@ -1,17 +0,0 @@
-+++
-title = "Gairebé sense JavaScript"
-date = 2023-01-06
-updated = 2023-04-28
-description = "JavaScript només s'utilitza quan HTML i CSS no són suficients."
-
-[taxonomies]
-tags = ["funcionalitat"]
-+++
-
-# JavaScript?
-
-Aquest tema gairebé no utilitza JavaScript. Inclou uns ~900 bytes de fitxers `.js` amb la lògica per al canvi de tema clar/fosc, que es pot desactivar establint `theme_switcher = false` al fitxer `config.toml`.
-
-La funcionalitat de KaTex, que requereix carregar un fitxer JavaScript de 274 KB, es pot activar per a publicacions específiques.
-
-A part d'això, és un tema ràpid amb HTML i CSS. Tal i com hauria de ser (la major part de) la web :-)
diff --git a/content/blog/almost-no-js.es.md b/content/blog/almost-no-js.es.md
deleted file mode 100644
index 3d1c64504..000000000
--- a/content/blog/almost-no-js.es.md
+++ /dev/null
@@ -1,17 +0,0 @@
-+++
-title = "Casi sin JavaScript"
-date = 2023-01-06
-updated = 2023-04-28
-description = "JavaScript solo se utiliza cuando HTML y CSS no son suficientes."
-
-[taxonomies]
-tags = ["funcionalidad"]
-+++
-
-# ¿JavaScript?
-
-Este tema casi no utiliza JavaScript. Incluye ~900 bytes de código `.js` con la lógica para el interruptor de modo claro/oscuro, el cual puede ser desactivado configurando `theme_switcher = false` en el archivo `config.toml`.
-
-El soporte de KaTeX, que requiere cargar un archivo JavaScript de 274 KB, puede ser activado para publicaciones específicas.
-
-Aparte de esto, es un tema rápido con HTML y CSS. Como debería ser (en su mayoría) la web :-)
diff --git a/content/blog/almost-no-js.md b/content/blog/almost-no-js.md
deleted file mode 100644
index 8e275965c..000000000
--- a/content/blog/almost-no-js.md
+++ /dev/null
@@ -1,17 +0,0 @@
-+++
-title = "Almost no JavaScript"
-date = 2023-01-06
-updated = 2023-04-28
-description = "JavaScript is only used when HTML and CSS aren't enough."
-
-[taxonomies]
-tags = ["showcase"]
-+++
-
-# JavaScript?
-
-This theme has almost no JavaScript. It includes ~900 bytes of `.js` with the logic for the light/dark mode switch, which can be disabled by setting `theme_switcher = false` in the `config.toml` file.
-
-KaTex support, which requires loading a 274 KB JavaScript file, can be activated for specific posts.
-
-Other than that, it's a fast site with HTML and CSS. Just the way (most of) the web should be :-)
diff --git a/content/blog/javascript.ca.md b/content/blog/javascript.ca.md
new file mode 100644
index 000000000..28eb3a646
--- /dev/null
+++ b/content/blog/javascript.ca.md
@@ -0,0 +1,22 @@
++++
+title = "Sense JavaScript obligatori"
+date = 2023-01-06
+updated = 2023-04-28
+description = "JavaScript només s'utilitza quan HTML i CSS no són suficients."
+
+[taxonomies]
+tags = ["funcionalitat"]
++++
+
+## JavaScript?
+
+Aquest tema funciona perfectament sense JavaScript. Opcionalment, pot carregar una quantitat mínima per afegir algunes funcionalitats que no són possibles utilitzant només HTML i CSS:
+
+- **Canvi de mode clar/fosc**. S'activa establint `theme_switcher = true`. (~900 bytes)
+- **Còpia de blocs de codi amb un sol clic**. S'activa establint `copy_button = true`. (~700 bytes)
+
+Aquestes dues configuracions cal aplicar-les a la secció `[extra]` del fitxer `config.toml`.
+
+La funcionalitat de KaTex, que requereix carregar un fitxer JavaScript de 274 KB, es pot activar per a publicacions específiques. Això es pot fer establint `katex = true` a la secció `[extra]` de l'encapçalament de la publicació.
+
+A part d'això, és un tema ràpid amb HTML i CSS. Tal i com hauria de ser (la major part de) la web :-)
diff --git a/content/blog/javascript.es.md b/content/blog/javascript.es.md
new file mode 100644
index 000000000..696636235
--- /dev/null
+++ b/content/blog/javascript.es.md
@@ -0,0 +1,22 @@
++++
+title = "Sin JavaScript obligatorio"
+date = 2023-01-06
+updated = 2023-04-28
+description = "JavaScript solo se utiliza cuando HTML y CSS no son suficientes."
+
+[taxonomies]
+tags = ["funcionalidad"]
++++
+
+## ¿JavaScript?
+
+Este tema funciona perfectamente sin JavaScript. Opcionalmente, puede cargar una cantidad mínima para añadir algunas funciones que son imposibles de lograr con HTML y CSS:
+
+- **El cambio de modo claro/oscuro**. Habilitado estableciendo `theme_switcher = true`. (~900 bytes)
+- **Copia de bloques de código con un clic**. Se activa configurando `copy_button = true`. (~700 bytes)
+
+Estas dos configuraciones se deben aplicar en la sección `[extra]` de tu archivo `config.toml`.
+
+El soporte de KaTex, que requiere cargar un archivo JavaScript de 274 KB, se puede activar para publicaciones específicas. Esto se puede hacer configurando `katex = true` en la sección `[extra]` del encabezado de la publicación.
+
+Aparte de esto, es un tema rápido con HTML y CSS. Como debería ser (en su mayoría) la web :-)
diff --git a/content/blog/javascript.md b/content/blog/javascript.md
new file mode 100644
index 000000000..cc8082a76
--- /dev/null
+++ b/content/blog/javascript.md
@@ -0,0 +1,22 @@
++++
+title = "No mandatory JavaScript"
+date = 2023-01-06
+updated = 2023-04-28
+description = "JavaScript is only used when HTML and CSS aren't enough."
+
+[taxonomies]
+tags = ["showcase"]
++++
+
+## JavaScript?
+
+This theme has no mandatory JavaScript. Optionally, it can load a minimal amount to add some features that are impossible to achieve with HTML and CSS:
+
+- **Light/dark mode switch**. Enabled by setting `theme_switcher = true`. (~900 bytes)
+- **One-click copy of code blocks**. Enabled by setting `copy_button = true`. (~700 bytes)
+
+These two settings can be applied in the `[extra]` section of your `config.toml` file.
+
+KaTex support, which requires loading a 274 KB JavaScript file, can be activated for specific posts. This can be done by setting `katex = true` in the post's `[extra]` section of the post's front matter.
+
+Other than that, it's a fast site with HTML and CSS. Just the way (most of) the web should be :-)
diff --git a/sass/parts/_code.scss b/sass/parts/_code.scss
index d68d207e0..2995cff97 100644
--- a/sass/parts/_code.scss
+++ b/sass/parts/_code.scss
@@ -2,68 +2,104 @@ code {
background-color: var(--bg-1);
padding: 0.1em 0.2em;
font-family: var(--code-font);
- font-size: 0.9em;
+ font-size: 0.9rem;
+
+ mark {
+ background-color: var(--codeblock-highlight);
+ color: inherit;
+ filter: brightness(110%);
+ display: block;
+ }
+
+ table {
+ width: 100%;
+ margin: 0rem;
+ border-collapse: collapse;
+ border-spacing: 0rem;
+
+ td,
+ th,
+ tr {
+ border: none;
+ padding: 0rem;
+ }
+
+ tbody td:first-child {
+ user-select: none;
+ width: 2rem;
+ text-align: left;
+ }
+
+ tbody tr:nth-child(even) {
+ background-color: inherit;
+ }
+ }
}
pre {
+ overflow: hidden;
+ position: relative;
display: block;
line-height: 1.4;
overflow-x: auto;
padding: 2rem 1rem 1rem;
- position: relative;
-webkit-overflow-scrolling: touch;
border-radius: 5px;
-}
-pre code {
- background-color: transparent;
- color: inherit;
- padding: 0;
- border: 0;
- border-radius: 4px;
-}
+ code {
+ display: block;
+ overflow-x: auto;
+ white-space: pre;
+ background-color: transparent;
+ color: inherit;
+ padding: 0rem;
+ border: 0rem;
+ border-radius: 5px;
-pre code[class*="language-"] {
- -webkit-overflow-scrolling: touch;
-}
+ &::before {
+ content: attr(data-lang);
+ display: block;
+ background-color: var(--primary-color);
+ color: var(--hover-color);
+ padding: 0.3rem;
+ padding-left: 1rem;
+ width: calc(100% - 1.3rem);
+ height: 0.9rem;
+ font-size: 0.65rem;
+ position: absolute;
+ text-align: left;
+ text-transform: uppercase;
+ top: 0;
+ left: 0;
+ }
-pre code[class*="language-"]::before {
- content: attr(data-lang);
- display: block;
- background-color: var(--primary-color);
- color: var(--hover-color);
- padding: 0.3rem;
- padding-left: 1rem;
- font-family: var(--code-font);
- width: 100%;
- font-size: 0.65rem;
- width: 100%;
- position: absolute;
- text-align: left;
- text-transform: uppercase;
- top: 0;
- left: 0;
+ &[class*="language-"] {
+ -webkit-overflow-scrolling: touch;
+ }
+ }
}
.copy-code {
z-index: 1;
- -webkit-mask: url();
- // -webkit-mask: url();
+ -webkit-mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M217.002-67.694q-37.732 0-64.02-26.288-26.287-26.287-26.287-64.019V-707.69h77.999v549.689q0 4.615 3.846 8.462 3.846 3.846 8.462 3.846h451.689v77.999H217.002Zm175.999-175.999q-37.733 0-64.02-26.287T302.694-334v-463.383q0-37.732 26.287-64.02 26.287-26.287 64.02-26.287h365.383q37.732 0 64.019 26.287 26.288 26.288 26.288 64.02V-334q0 37.733-26.288 64.02-26.287 26.287-64.019 26.287H393.001Zm0-77.998h365.383q4.615 0 8.462-3.847 3.846-3.846 3.846-8.462v-463.383q0-4.616-3.846-8.462-3.847-3.846-8.462-3.846H393.001q-4.616 0-8.462 3.846-3.847 3.846-3.847 8.462V-334q0 4.616 3.847 8.462 3.846 3.847 8.462 3.847Zm-12.309 0v-488V-321.691Z'/%3E%3C/svg%3E");
background: var(--hover-color);
cursor: pointer;
- display: inline-block;
position: absolute;
- height: 20px;
- width: 20px;
+ height: 0.9rem;
+ width: 0.9rem;
+ background-size: contain;
color: white;
- right: 0.5rem;
- top: 0.2rem;
+ right: 0.7rem;
+ top: 0.3rem;
+ align-self: center;
}
.copy-code.checked {
- -webkit-mask: url();
+ -webkit-mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M395-253 194-455l83-83 118 117 288-287 83 84-371 371Z'/%3E%3C/svg%3E");
+ height: 1rem;
+ width: 1rem;
}
.copy-code.error {
- -webkit-mask: url();
+ -webkit-mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M479.386-248Q509-248 529-267.386q20-19.386 20-49T529.614-366.5q-19.386-20.5-49-20.5T431-366.886q-20 20.114-20 49.728t19.386 49.386q19.386 19.772 49 19.772ZM416-431h128v-265H416v265Zm64.276 381q-88.916 0-167.743-33.104-78.828-33.103-137.577-91.852-58.749-58.749-91.852-137.535Q50-391.277 50-480.458q0-89.438 33.162-167.491 33.163-78.053 92.175-136.942 59.011-58.889 137.533-91.999Q391.393-910 480.458-910q89.428 0 167.518 33.093T784.94-784.94q58.874 58.874 91.967 137.215Q910-569.385 910-480.192q0 89.192-33.11 167.518-33.11 78.326-91.999 137.337-58.889 59.012-137.167 92.174Q569.447-50 480.276-50Z'/%3E%3C/svg%3E");
}
diff --git a/static/js/copyCodeToClipboard.js b/static/js/copyCodeToClipboard.js
new file mode 100644
index 000000000..ba98ba3d0
--- /dev/null
+++ b/static/js/copyCodeToClipboard.js
@@ -0,0 +1,37 @@
+const changeIcon = (copyDiv, className) => {
+ copyDiv.classList.add(className);
+ setTimeout(() => copyDiv.classList.remove(className), 2500);
+};
+
+const addCopyEventListenerToDiv = (copyDiv, block) => {
+ copyDiv.addEventListener("click", () => copyCodeAndChangeIcon(copyDiv, block));
+};
+
+const copyCodeAndChangeIcon = async (copyDiv, block) => {
+ const code = block.querySelector('table') ? getTableCode(block) : getNonTableCode(block);
+ try {
+ await navigator.clipboard.writeText(code);
+ changeIcon(copyDiv, "checked");
+ } catch (error) {
+ changeIcon(copyDiv, "error");
+ }
+};
+
+const getNonTableCode = (block) => {
+ return [...block.querySelectorAll('code')]
+ .map(code => code.textContent)
+ .join('');
+};
+
+const getTableCode = (block) => {
+ return [...block.querySelectorAll('tr')]
+ .map(row => row.querySelector('td:last-child')?.innerText ?? '')
+ .join('');
+};
+
+document.querySelectorAll("pre").forEach((block) => {
+ const copyDiv = document.createElement("div");
+ copyDiv.className = "copy-code";
+ block.prepend(copyDiv);
+ addCopyEventListenerToDiv(copyDiv, block);
+});
diff --git a/static/js/copyCodeToClipboard_min.js b/static/js/copyCodeToClipboard_min.js
new file mode 100644
index 000000000..9d144b1ec
--- /dev/null
+++ b/static/js/copyCodeToClipboard_min.js
@@ -0,0 +1 @@
+const changeIcon=(e,t)=>{e.classList.add(t),setTimeout(()=>e.classList.remove(t),2500)},addCopyEventListenerToDiv=(e,t)=>{e.addEventListener("click",()=>copyCodeAndChangeIcon(e,t))},copyCodeAndChangeIcon=async(e,t)=>{let o=t.querySelector("table")?getTableCode(t):getNonTableCode(t);try{await navigator.clipboard.writeText(o),changeIcon(e,"checked")}catch(c){changeIcon(e,"error")}},getNonTableCode=e=>[...e.querySelectorAll("code")].map(e=>e.textContent).join(""),getTableCode=e=>[...e.querySelectorAll("tr")].map(e=>e.querySelector("td:last-child")?.innerText??"").join("");document.querySelectorAll("pre").forEach(e=>{let t=document.createElement("div");t.className="copy-code",e.prepend(t),addCopyEventListenerToDiv(t,e)});
diff --git a/static/js/copy_button.js b/static/js/copy_button.js
deleted file mode 100644
index 712f7a4dc..000000000
--- a/static/js/copy_button.js
+++ /dev/null
@@ -1,19 +0,0 @@
-function changeIcon(copyDiv, className) {
- copyDiv.classList.add(className);
- setTimeout(() => copyDiv.classList.remove(className), 2500);
-}
-
-document.querySelectorAll("pre").forEach((block) => {
- const copyDiv = document.createElement("div");
- copyDiv.className = "copy-code";
- block.prepend(copyDiv);
-
- copyDiv.addEventListener("click", function () {
- const code = block.innerText;
- navigator.clipboard.writeText(code).then(() => {
- changeIcon(copyDiv, "checked");
- }, () => {
- changeIcon(copyDiv, "error");
- });
- });
-});
diff --git a/templates/base.html b/templates/base.html
index 1e37aac3b..cd09f2f84 100644
--- a/templates/base.html
+++ b/templates/base.html
@@ -22,11 +22,16 @@
{% include "partials/footer.html" %}
- {% if page.extra.katex and page.extra.katex == true %}
+ {# Add KaTeX functionality (loads CSS and JS) #}
+ {%- if page.extra.katex and page.extra.katex == true -%}
-
- {% endif %}
+ {%- endif -%}
+
+ {# Add copy button to codeblocks #}
+ {%- if config.extra.copy_button and config.extra.copy_button == true -%}
+
+ {%- endif -%}