Pen Settings

HTML

CSS

CSS Base

Vendor Prefixing

Add External Stylesheets/Pens

Any URLs added here will be added as <link>s in order, and before the CSS in the editor. You can use the CSS from another Pen by using its URL and the proper URL extension.

+ add another resource

JavaScript

Babel includes JSX processing.

Add External Scripts/Pens

Any URL's added here will be added as <script>s in order, and run before the JavaScript in the editor. You can use the URL of any other Pen and it will include the JavaScript from that Pen.

+ add another resource

Packages

Add Packages

Search for and use JavaScript packages from npm here. By selecting a package, an import statement will be added to the top of the JavaScript editor for this package.

Behavior

Auto Save

If active, Pens will autosave every 30 seconds after being saved once.

Auto-Updating Preview

If enabled, the preview panel updates automatically as you code. If disabled, use the "Run" button to update.

Format on Save

If enabled, your code will be formatted when you actively save your Pen. Note: your code becomes un-folded during formatting.

Editor Settings

Code Indentation

Want to change your Syntax Highlighting theme, Fonts and more?

Visit your global Editor Settings.

HTML

              
                <header>
	<nav>
		<ul class="menu">
			<li class="menu-item"><a href="#futhark" class="menu-link" aria-current="true">Futhark</a></li>

			<li class="menu-item"><a href="#learn" class="menu-link" aria-current="false">Learn</a></li>
		</ul>
	</nav>
</header>

<main>
	<section id="futhark" aria-hidden="false">
		<h1 class="section-overlap-box"><span>Fu&shy;th&shy;ark</span></h1>

		<ol class="grid" role="list"></ol>

		<form class="section-overlap-box">
			<input type="checkbox" id="show-alt-runes">
			<label for="show-alt-runes">&nbsp;Show alternative Runes</label>
		</form>
	</section>

	<section id="learn" aria-hidden="true">
		<h1 class="section-overlap-box"><span>Ex&shy;er&shy;cise</span></h1>

		<div id="exercise" class="content" tabindex="-1">
			<p class="text-center">Click below to get started!</p>
		</div>

		<div class="section-overlap-box">
			<button type="button" id="new-exercise">New exercise</button>
		</div>
	</section>
</main>

<footer>
	<p>Made with curiosity by <a href="https://chriskirknielsen.com" target="_top">chriskirknielsen</a>, info from <a href="https://en.wikipedia.org/wiki/Elder_Futhark">Wikipedia</a></p>
</footer>

<template id="exercise-match">
	<p data-match="glyph-to-match" class="large-letter single-flat-word">X</p>

	<p data-match="prompt" class="text-center">Which one of these match the glyph above?</p>

	<ol class="match-grid" data-match="options">
		<li data-match="option" data-index="1"><button type="button" class="stone" data-match="opt-button" aria-pressed="false">1</button></li>
		<li data-match="option" data-index="2"><button type="button" class="stone" data-match="opt-button" aria-pressed="false">2</button></li>
		<li data-match="option" data-index="3"><button type="button" class="stone" data-match="opt-button" aria-pressed="false">3</button></li>
		<li data-match="option" data-index="4"><button type="button" class="stone" data-match="opt-button" aria-pressed="false">4</button></li>
	</ol>

	<p class="msg-warning" data-match="warning" tabindex="-1" hidden></p>
	<p data-match="result" tabindex="-1" hidden></p>

	<button type="button" data-match="verify" class="large-button">Check answer</button>
</template>

<template id="exercise-spell">
	<p data-spell="word-to-spell" class="large-letter single-flat-word text-center">WORD</p>

	<p data-spell="prompt"class="text-center">Translate the {{type}} above using the {{glyphs}} below.</p>

	<ol class="glyphbank is-added" data-spell="composed"></ol>

	<ol class="glyphbank is-idle" data-spell="glyphs"></ol>

	<p class="msg-warning" data-spell="warning" tabindex="-1" hidden></p>
	<p data-spell="result" tabindex="-1" hidden></p>

	<button type="button" data-spell="verify" class="large-button">Check answer</button>
</template>
<template id="exercise-spell-tile">
	<li data-spell="glyph"><button type="button" class="tile" data-spell="opt-button" aria-pressed="false">X</button></li>
</template>

<svg aria-hidden="true" xmlns="http://www.w3.org/2000/svg" class="visually-hidden">
	<defs>
		<linearGradient id="gold" x1="0" x2="100%" y1="0" y2="100%">
			<stop stop-color="#BF953F" offset="0%" />
			<stop stop-color="#FCF6BA" offset="25%" />
			<stop stop-color="#B38728" offset="50%" />
			<stop stop-color="#FBF5B7" offset="75%" />
			<stop stop-color="#AA771C" offset="100%" />
		</linearGradient>

		<filter id="noise">
			<feGaussianBlur in="SourceGraphic" stdDeviation="0.0625" result="blur" />
			<feTurbulence result="waves" type="turbulence" baseFrequency="0.735 0.771" numOctaves="1" seed="256"></feTurbulence>
			<feDisplacementMap in="blur" in2="waves" scale="0.5" xChannelSelector="R" yChannelSelector="B" result="ripples"></feDisplacementMap>
			<feComposite in="waves" in2="ripples" operator="arithmetic" k1="1" k2="0" k3="1" k4="0"></feComposite>
		</filter>

		<filter id="noise2x">
			<feGaussianBlur in="SourceGraphic" stdDeviation="0.25" result="blur" />
			<feTurbulence result="waves" type="turbulence" baseFrequency="2.173 1.977" numOctaves="4" seed="128"></feTurbulence>
			<feDisplacementMap in="blur" in2="waves" scale="2" xChannelSelector="R" yChannelSelector="B" result="ripples"></feDisplacementMap>
			<feComposite in="waves" in2="ripples" operator="arithmetic" k1="1" k2="0" k3="1" k4="0"></feComposite>
		</filter>

		<filter id="highlight" x="0" y="0" width="200%" height="200%">
			<feDropShadow dx="0.25" dy="0.5" stdDeviation="0.1" flood-color='hsl(230 60% 60%)' flood-opacity="0.75" />
			<feDropShadow dx="0.125" dy="0.25" stdDeviation="0.05" flood-color='hsl(230 60% 60%)' flood-opacity="0.75" />
		</filter>

		<!-- https://css-tricks.com/adding-shadows-to-svg-icons-with-css-and-svg-filters/#aa-inset-shadows -->
		<filter id="innerShadow">
			<feOffset dx="1.25" dy="1.25" />
			<feGaussianBlur stdDeviation="0.75" result="offset-blur" />
			<feComposite operator="out" in="SourceGraphic" in2="offset-blur" result="inverse" />
			<feFlood flood-color="black" flood-opacity="0.95" result="color" />
			<feComposite operator="in" in="color" in2="inverse" result="shadow" />
			<feComposite operator="over" in="shadow" in2="SourceGraphic" />
		</filter>

		<!-- Petite Patterns by Bence Szabo: https://codepen.io/finnhvman/details/qBPGRgr -->
		<!-- Sorry just way too greedy for 24 elements!
		<filter id="rock">
			<feTurbulence type="fractalNoise" baseFrequency=".0125" numOctaves="1" />
			<feMorphology radius="8" operator="dilate" />
			<feConvolveMatrix kernelMatrix="9 0 -9" order="1 3" preserveAlpha="true" />
			<feColorMatrix values="
								   .4 0 0 0 0
								   .9 0 0 0 0
								   .9 0 0 0 0
								    0 0 0 0 1" />
		</filter>
		<filter id="rockMini">
			<feTurbulence type="fractalNoise" baseFrequency=".02" numOctaves="1" />
			<feMorphology radius="1" operator="dilate" />
			<feConvolveMatrix kernelMatrix="9 0 -9" order="1 3" preserveAlpha="true" />
			<feColorMatrix values="
								   .5 0 0 0 0
								   .7 0 0 0 0
								   .9 0 0 0 0
								    0 0 0 0 1" />
		</filter>
		-->
	</defs>
</svg>

<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 130.3 130.3" aria-hidden="true" class="visually-hidden">
	<g id="sven-badge">
		<path fill="var(--bg-fill,#fbb03b)" stroke="#39106d" stroke-miterlimit="10" stroke-width="5" d="M109.3 21 65.1 2.7 21 21 2.7 65.1 21 109.3l44.1 18.3 44.2-18.3 18.3-44.2L109.3 21z" />
		<path fill="#7c3c2f" d="M94.4 55.6 82 49.9H48.3l-12.4 5.7-6.8 11.1v34.2h72.1V66.7l-6.8-11.1z" />
		<path fill="#31393f" d="M101.2 100.9H29.1l-1.8 1.7v1.8H103v-1.8l-1.8-1.7z" />
		<path fill="#31393f" d="M29.1 94.6h72.2v3.6H29.1z" />
		<path fill="#31393f" d="M64.1 89.6h2v14.8h-2z" />
		<path fill="#f2f2f2" d="m82.1 20.8-6-5 3.2 5.8.8 4.6-6.7 5.6H56.9l-6.7-5.6.8-4.6 3.2-5.8-6 5-1.9 6.5 7.2 9.2h23.3l7.2-9.2-1.9-6.5z" />
		<path id="Face" fill="#c7b299" d="M83.3 44.6H47l-1.7 4H85l-1.7-4z" />
		<path id="Eyebrows" fill="#ed1c24" d="M51.1 47h28v-4h-28v4Z" />
		<path fill="#616f7c" d="m74.3 25.3-9.2-4.6-9.1 4.6-7.8 15.5h33.9l-7.8-15.5z" />
		<path fill="#b2151b" d="M85 48.6H45.3l-5.1 11.5L45.3 77l19.8 14.8L85 77l4.6-16.9L85 48.6z" />
		<path fill="#ed1c24" d="m74.7 48.7-9.6-1.6-9.5 1.6-4.5 8 1.1 3.7 7.3-6.2 2.3 1.4 3.3-1.7 3.4 1.7 2.3-1.4 7.3 6.2 1.1-3.7-4.5-8z" />
		<path id="Nose" fill="#c7b299" d="m65.1 45.2-4.5 2.7 4.5 4.1 4.6-4.1-4.6-2.7z" />
		<path fill="#31393f" d="m83.3 40.9-5.7-.5-12.5-.1-12.4.1-5.7.5-1 3.7 9.6.5 9.5.2 9.6-.2 9.6-.5-1-3.7z" />
		<path fill="#c6e3fc" d="M64.4 40.3h1.5V20.7l-.8-.4-.7.4v19.6z" />
		<path fill="#39106d" d="M3.5 107.1h123.3v23.2H3.5z" />
		<!-- 		<text fill="#fbb03b" font-family="Cabin, GillSans, 'Gill Sans'" font-size="15" x="50%" y="123" text-anchor="middle"></text> -->
	</g>
</svg>
              
            
!

CSS

              
                // Resets and stuff
*,
*::before,
*::after {
	box-sizing: border-box;
	margin: 0;
	padding: 0;
}

[hidden] { display: none !important; }

// Global styling variables
@property --padding {
	syntax: '<length>';
	inherit: true;
	initial-value: 0px;
}

:root {
	--font-alpha: 'Cabin', system-ui, sans-serif;
	--font-runic: "Noto Sans Runic", system-ui;
	--bg: hsl(190 8.6% 27.5%);
	--bg2: hsl(173.3 22% 8%);
	--bg-light: hsl(190 8.6% 55%);
	--frame-bg: hsl(20 67% 17%);
	--frame-bg-dark--values: 20 75% 10%;
	--frame-bg-dark: hsl(var(--frame-bg-dark--values));
	--frame-color: hsl(36 88% 75%);
	--frame-size: calc(var(--padding) * 0.75);
	--box-shadow:
		inset 1px 2px 4px 2px hsl(20 50% 5% / 0.25),
		1px 2px 4px 2px hsl(20 50% 5% / 0.5);
	--hue-warning: 50;
	--hue-wrong: 10;
	--hue-correct: 140;
	--text: white;
	--accent: gold;
	--accent-gradient: #BF953F, #FCF6BA, #B38728, #FBF5B7, #AA771C;
	--heading-font-size: clamp(1.875rem, 1.25rem + 2.25vw, 3.5rem);
	--rune-font-size: clamp(1.75rem, 1.25rem + 5vw, 3.25rem);
	--padding: 2rem;
	--transition: 200ms ease-in-out;
	
	@media (prefers-reduced-motion) {
		--transition: 0.01ms;
	}
	
	/* ⬇️ https://www.joshwcomeau.com/shadow-palette/ ⬇️ */
	--shadow-color: 20deg 25% 8%;
	--shadow-short:
		0.7px 0.5px 1px hsl(var(--shadow-color) / 0.29),
		2.1px 1.4px 2.8px -0.8px hsl(var(--shadow-color) / 0.29),
		5.1px 3.5px 7px -1.7px hsl(var(--shadow-color) / 0.29),
		12.3px 8.4px 16.8px -2.5px hsl(var(--shadow-color) / 0.29);
	--shadow-soft:
		0.6px 0.6px 0.6px hsl(var(--shadow-color) / 1),
		1px 1px 1.1px -0.8px hsl(var(--shadow-color) / 0.99),
		3.9px 3.8px 4.1px -1.7px hsl(var(--shadow-color) / 0.83),
		11.6px 11.5px 12.3px -2.5px hsl(var(--shadow-color) / 0.66),
		26.8px 26.5px 28.3px -3.3px hsl(var(--shadow-color) / 0.5),
		51.8px 51.3px 54.7px -4.2px hsl(var(--shadow-color) / 0.33),
		89px 88.1px 93.9px -5px hsl(var(--shadow-color) / 0.17);
	--shadow-elevated:
		1px 1px 1.1px hsl(var(--shadow-color) / 0.77),
		1.5px 1.5px 1.6px -0.8px hsl(var(--shadow-color) / 0.66),
		4.7px 4.7px 5px -1.7px hsl(var(--shadow-color) / 0.55),
		13.4px 13.4px 14.2px -2.5px hsl(var(--shadow-color) / 0.44),
		30.3px 30.3px 32.1px -3.3px hsl(var(--shadow-color) / 0.33),
		58.3px 58.3px 61.8px -4.2px hsl(var(--shadow-color) / 0.22),
		100px 100px 106.1px -5px hsl(var(--shadow-color) / 0.11);
}

// Viewport
html {
	width: 100%;
	height: 100%;
	overflow-x: clip;
	overflow-y: auto;
	
	font-family: var(--font-alpha);
	color: var(--text);
	accent-color: var(--accent);
	background: var(--bg) linear-gradient(135deg, var(--bg-light), var(--bg2)) fixed;
}

body {
	position: relative;
	
	display: flex;
	flex-direction: column;
    min-height: 100%;
	
	
	&::before {
		content: '';
		position: fixed;
		inset: 0;
		z-index: -1;
		
		pointer-events: none;
		
		// background: var(--bg); // Just to fill the element, the filter will replace all the pixels
		background: var(--rock-texture) repeat 50% 50% / 450px 450px;
		
		mix-blend-mode: screen;
		// filter: url(#rock) blur(2px) opacity(0.33) grayscale(1);
		filter: blur(2px) opacity(0.33) grayscale(1);
	}
}

// Default element styles
h1,
p {
	max-width: 100%;
	hyphens: auto;
}

h1 {
	--overlap-box-padding: 0.125em var(--overlap-box-padding-inline, 0.75em) 0.1625em;
	
	font-weight: 400;
	font-size: var(--heading-font-size);
	letter-spacing: 0.0625em;
	text-transform: uppercase;
	
	& > span {
		color: gold;

		filter: drop-shadow(-0.5px -1px 0.25px hsl(0 0% 90%)) drop-shadow(0.5px 1px 0.25px hsl(0 0% 10%));

		@supports (background-clip: text) {
			background: linear-gradient(120deg, var(--accent-gradient));
			-webkit-background-clip: text;
			background-clip: text;
			color: transparent;
		}
	}
}

p {
	line-height: 1.4;
}
p.runic {
	word-break: break-word;
	hyphenate-character: ''; // Avoid confusion as a new character
}

a {
	color: var(--accent);
	text-decoration-thickness: 2px;
	text-underline-offset: 2px;
	
	&:is(:hover, :focus) {
		color: var(--text);
	}
}

button {
	appearance: none;
	
	&:where(:not(.stone):not(.tile)) {
		--border-thickness: 1px;
		
		isolation: isolate;
		position: relative;
		
		padding: 0.25em 0.5em;

		font: inherit;
		text-transform: uppercase;
		letter-spacing: 0.125em;
		color: var(--frame-bg);
		text-shadow: 0 1px 0 var(--text-shadow-color, var(--accent));
		border: var(--border-thickness) solid var(--frame-bg);
		border-image: linear-gradient(120deg, var(--accent-gradient)) 1 / var(--border-thickness) round;
		background: var(--frame-color) linear-gradient(var(--bg-angle, -30deg), var(--accent-gradient));

		&:is(:hover, :focus):where(:not([aria-disabled="true"])) {
			--text-shadow-color: black;
			
			color: var(--frame-color);
			background: var(--frame-bg-dark);
		}
	
		&[aria-disabled="true"] {
			cursor: not-allowed;
			filter: grayscale(1);
		}

		&:focus-visible {
			outline: 2px solid var(--accent);
		}
		
		&::before {
			--size: calc(var(--border-thickness, 2px) * 2);
			--offset: calc(var(--size, 2px) + var(--border-thickness));
			--cut: calc(var(--offset) * 3);
			
			content: '';
			
			position: absolute;
			inset: calc(var(--offset) * -1);
			z-index: -1;
			
			background: linear-gradient(-60deg, var(--accent-gradient));
			
			filter: brightness(0.75);
			mask-image: // Only keep the pixels outside the button's edge
				linear-gradient(to top, red var(--size), transparent 0),
				linear-gradient(to bottom, red var(--size), transparent 0),
				linear-gradient(to left, red var(--size), transparent 0),
				linear-gradient(to right, red var(--size), transparent 0);
			clip-path: polygon(
				var(--cut) 0,
				calc(100% - var(--cut)) 0,
				100% var(--cut),
				100% calc(100% - var(--cut)),
				calc(100% - var(--cut)) 100%,
				var(--cut) 100%,
				0 calc(100% - var(--cut)),
				0 var(--cut)
			);
		}
	}
}

.large-button {
	--border-thickness: 2px;
	--bg-angle: 60deg;
	
	font-size: 2em;
	line-height: 1.2;
}

// Top Nav area
header {
	position: relative;
	z-index: 2;
	
	background: var(--bg2);
}

header > nav {
	max-width: 40rem;
	margin-inline: auto;
}

.menu {
	display: flex;
	flex-wrap: wrap;
	justify-content: space-evenly;
	
	list-style: none;
}

.menu-item {
	width: max-content;
}

.menu-link {
	position: relative;
	
	display: block;
	padding: calc(var(--padding) / 2);
	
	font-size: 1.5rem;
	color: var(--menu-item-color, var(--text));
	text-decoration: none;
	
	&::before {
		--size: 4px;
		
		content: '';

		position: absolute;
		inset-block-end: 0;
		inset-inline: 0;
		height: var(--size);

		clip-path: polygon(var(--size) 0, calc(100% - var(--size)) 0, 100% 100%, 0 100%);
		background: var(--menu-item-indicator, var(--bg));
	}
	
	&:where(:hover, :focus) {
		--menu-item-indicator: var(--bg-light);
	}
	
	&[aria-current="true"] {
		--menu-item-color: var(--accent);
		--menu-item-indicator: linear-gradient(150deg, var(--accent-gradient));
	}
	
	&[data-off] {
		cursor: not-allowed;
		text-decoration: line-through;
		filter: grayscale(1);
	}
}

// Content area
main {
	display: grid;
	grid-template-columns: 1fr;
	grid-template-rows: 1fr;
	justify-items: center;
	align-items: center;
	align-content: center;
	margin-block: auto;
	max-width: 100%;
	padding: var(--padding);
}

section {
	position: relative;
	isolation: isolate;
	container-name: section;
	container-type: inline-size;
	
	grid-row: 1 / -1;
	grid-column: 1 / -1;
	display: grid;
	flex-direction: column;
	align-items: center;
	gap: calc(var(--padding) / 2);
	width: min(100%, 64rem);
	max-width: 100%;
	max-height: 999vh; // Ludicrous, on purpose!
	padding: var(--padding);
	
	border: 1px solid transparent; // Add a border so Chrome renderes the image, I think?
	border-image: var(--frame-image) 58 57 / var(--frame-size) round;
	background: hsl(20deg 67% 17%);
	box-shadow: var(--shadow-soft);
	
	transition: var(--transition);
	
	@media (prefers-reduced-motion) {
		transition-duration: 0.01ms;
	}
	
	main:has(header + &) &:has(> h1:first-child) {
		margin-block-start: var(--heading-font-size);
	}
}

#futhark {
	--row: 1;
	--yeet: -100vw;
}
#learn {
	--row: 2;
	--yeet: 100vw;
}

// "Tab" "handling"
section[aria-hidden="true"] {
	z-index: -1;
	
	max-height: 50vh;
	
	user-select: none;
	pointer-events: none;
	
	opacity: 0;
	transform: translateX(var(--yeet, 0px)) scale(0.5);
}

/*/ This is slower than the other approach so… bye!
@supports selector(main:has(> #learn[aria-hidden='true'])) {
	main {
		grid-template-rows: var(--futhark-size, 1fr) var(--learn-size, 1fr);
		transition: grid-template-rows var(--transition), padding var(--transition);
	}
	
	main:has(#futhark[aria-hidden="true"]) {
		--futhark-size: 0fr;
	}
	
	main:has(#learn[aria-hidden="true"]) {
		--learn-size: 0fr;
	}
	
	section {
		grid-row: var(--row);
		min-height: 0;
	}
	
	section[aria-hidden="true"] {
		--padding: 0px;
	}
} //*/

.section-overlap-box {
	--offset: calc(-0.375em - var(--frame-size) - var(--padding) / 2);
	--border: 2px;
	
	position: relative;
	
	margin-inline: auto;
	padding: var(--overlap-box-padding, 0.5em var(--overlap-box-padding-inline, 1.5em));
	
	text-align: center;
	line-height: 1em;
		
	border: var(--border) solid var(--frame-color);
	background: var(--frame-bg);
	box-shadow: var(--box-shadow);
	
	&:first-child {
		margin-block-start: var(--offset);
		margin-block-end: 0;
	}
	
	&:last-child {
		margin-block-start: 0;
		margin-block-end: var(--offset);
	}
	
	&::after {
		content: '';
		
		position: absolute;
		inset: 0;
		
		pointer-events: none;
		
		background:
			linear-gradient(135deg, var(--frame-color) 50%, transparent 0) 0 0,
			linear-gradient(-135deg, var(--frame-color) 50%, transparent 0) 100% 0,
			linear-gradient(45deg, var(--frame-color) 50%, transparent 0) 0 100%,
			linear-gradient(-45deg, var(--frame-color) 50%, transparent 0) 100% 100%,
			;
		background-size: calc(var(--border) * 3) calc(var(--border) * 3);
		background-repeat: no-repeat;
		
		outline: 1px solid var(--frame-color);
		outline-offset: calc(var(--border) * -1.5); 
	}
	
	@container section (max-width: 16rem) {
		--overlap-box-padding-inline: 0.5em;
	}
}

.content {
	display: flex;
	flex-direction: column;
	justify-content: center;
	align-items: center;
	gap: calc(var(--padding));
	padding-block: calc(var(--padding) / 2);
	
	transition: 0s;
	
	& > * { margin-block: 0; }
}

.text-center { text-align: center; }

// Stone grid
.grid {
	display: grid;
	// grid-template-columns: repeat(auto-fill, minmax(4rem, 1fr));
	grid-template-columns: repeat(var(--c, 12), minmax(3.5rem, 1fr));
	gap: 1.5em 1em;
	width: 100%;
	
	// Art-directed: I want every row to have the same number of items. 24 glyphs divides nicely between 12, 8, 6, 4, 3, 2, and 1
	@media (max-width: 66rem) { --c: 8; }
	@media (max-width: 50rem) { --c: 6; }
	@media (max-width: 38rem) { --c: 4; }
	@media (max-width: 26rem) { --c: 3; }
	@media (max-width: 22rem) { --c: 2; }
	@media (max-width: 17.5rem) { --c: 1; }
}

.stone {
	--angle: var(--a, 0);
	
	&:hover {
		--angle: calc(-1 * var(--a, 0));
	}
	
	position: relative;
	
	display: flex;
	flex-direction: column;
	align-items: center;
	gap: 0.5em;
	padding: 1em 1.25em;
	overflow: hidden;
	
	list-style: none;
	text-align: center;
	
	color: inherit;
	background-color: hsl(var(--h, 240) var(--s, 60%) var(--l, 30%));
	border: 0;
	border-radius: var(--r, 25%);
	box-shadow:
		// Inner shadows
		inset -0.125em -0.125em 2px hsl(var(--stone-shadow-hue, var(--h, 240)) 75% 20% / 0.9),
		inset -0.25em -0.25em 0.5em hsl(var(--stone-shadow-hue, var(--h, 240)) 82% 15% / 0.75),
		inset -0.5em -1em 2em hsl(var(--stone-shadow-hue, var(--h, 240)) 90% 10% / 0.5),
		
		// Inner highlights
		inset 1px 1px 1px hsl(var(--stone-shadow-hue, var(--h, 240)) 75% 90% / 0.9),
		inset 0.25em 0.25em 0.75em hsl(var(--stone-shadow-hue, var(--h, 240)) 82% 85% / 0.75),
		inset 0.5em 0.5em 3em hsl(var(--stone-shadow-hue, var(--h, 240)) 90% 80% / 0.5),
		
		// Drop shadow
		var(--shadow, var(--shadow-soft));
	
	transform: rotate(calc(var(--angle, 0) * 1deg)) translateX(calc(var(--x, 0) * 1px)) translateY(calc(var(--y, 0) * 1px));
	transition: var(--transition);
	
	&.has-alt :is([data-rune-default], [data-rune-alt]) { transition: var(--transition); }
	
	&.is-alt {
		--angle: calc(var(--a, 0) * -1);
		--l: 45%;
	}
	
	&.has-alt:not(.is-alt) [data-rune-alt],
	&.has-alt.is-alt [data-rune-default] {
		opacity: 0;
		user-select: none;
	}
	
	& > * {
		position: relative;
	}
	
	&::before {
		--stone-bg: var(--rock-texture) repeat calc(1%*var(--px, 50)) calc(1%*var(--py, 50)) / 200% 200%;
		content: '';

		position: absolute;
		inset: -20px;
		aspect-ratio: 1;

		background: var(--stone-bg), var(--stone-bg);
		background-blend-mode: screen, normal;

		mix-blend-mode: soft-light;
		// filter: url(#rockMini) blur(0.5px) opacity(0.9);
		filter: contrast(90%) opacity(0.8);
		transform: scale(2) rotate(calc(var(--ta, 0) * 1deg));
	}
}

.tile {
	--tile-border: var(--text);
	--tile-bg: var(--frame-color);
	--tile-text: var(--bg2);
	--tile-depth: 3px;
	--tile-shadow: 3px;
	
	padding-block: calc(var(--p, 0.5rem) / 2);
	padding-inline: var(--p, 0.5rem);
	
	font-size: 1.25em;
	color: var(--tile-text);
	border-radius: 4px;
	border: 1px solid var(--tile-border);
	background: var(--tile-bg);
	box-shadow: 0 var(--tile-shadow, var(--tile-depth)) 0 var(--tile-border);
	
	transition: var(--transition);
	
	&[aria-pressed="false"] {
		&:is(:hover, :focus) {
			--tile-shadow: 0px;
			transform: translateY(var(--tile-depth));
		}
	}
	
	&[aria-pressed="true"] {
		--tile-border: var(--bg);
		--tile-bg: var(--bg2);
		--tile-text: var(--tile-border);
		--tile-shadow: 0px;
		
		cursor: not-allowed;
	}
	
	[data-answer='wrong'] & {
		--tile-border: hsl(var(--hue-wrong) 80% 50%);
		--tile-text: var(--tile-border);
		--tile-bg: hsl(var(--hue-wrong) 50% 15%);
		
		animation: failWave 500ms cubic-bezier(.66,0,.5,.5) 0ms 1;
	}
	
	[data-answer='correct'] & {
		--tile-border: hsl(var(--hue-correct) 80% 50%);
		--tile-text: var(--tile-border);
		--tile-bg: hsl(var(--hue-correct) 50% 15%);
		
		animation: successWave 800ms cubic-bezier(.66,0,.5,.5) calc(75ms * var(--tile-index)) 1;
	}
}

@keyframes successWave {
	// I realised after the fact this is the same animation as Wordle but I like it so I'm keeping it
	0% { transform: translateY(0); }
	33% { transform: translateY(-50%); animation-timing-function: linear; }
	67% { transform: translateY(50%); animation-timing-function: linear; }
	100% { transform: translateY(0); animation-timing-function: cubic-bezier(.5,.5,.33,1); }
}

@keyframes failWave {
	0% { transform: translateX(0); }
	20%, 60% { transform: translateX(-1rem); animation-timing-function: ease-in-out; }
	40%, 80% { transform: translateX(1rem); animation-timing-function: ease-in-out; }
	100% { transform: translateX(0); animation-timing-function: cubic-bezier(.5,.5,.33,1); }
}

.runic { font-family: var(--font-runic); }
.alpha { text-transform: uppercase; }
.large-letter { font-size: calc(1.5 * var(--rune-font-size, 3.25em)); }
.single-flat-word.large-letter { font-size: calc(1.25 * var(--rune-font-size, 3.25em)); }

.glyph-rune {
	width: 1em;
	height: 1em;
	overflow: visible;
	
	font-size: var(--rune-font-size, 3.25em);
	filter: url(#innerShadow);
	
	& > text {
		fill: url(#gold);
		
		filter: url(#noise) url(#highlight) Grayscale(var(--stone-glyph-gray, 0)) brightness(calc(1 + 0.25 * var(--stone-glyph-gray, 0)));
		transition: var(--transition);
		
		&::selection {
			filter: none;
		}
	}
}

.single-flat-glyph { // Not used
	--shadow-color: 21deg 100% 8%;
	--fill: hsl(35 50% 34%);
	
	display: flex;
	justify-content: center;
	align-items: center;
	width: 1.75em;
	aspect-ratio: 1;
	padding: 0.25em;
	
	line-height: 1;
	text-align: center;

	color: var(--bg2);
	border: 2px solid var(--accent);
	border-radius: 50%;
	background: linear-gradient(-120deg, var(--accent-gradient)), linear-gradient(var(--fill), var(--fill));
	background-blend-mode: soft-light, normal;
	box-shadow: inset 3px 3px 6px hsl(var(--shadow-color) / 0.25), var(--shadow-short);
	
	& > span {
		--shadow-color: 47deg 77% 26%;
		
		line-height: 1.1; // Ensures taller characters don't get cut off
		
		filter:
			drop-shadow(-1px -1px 0 white)
			drop-shadow(1px 1px 0 black)
			drop-shadow(1px 1px 0 black)
			drop-shadow(1px 1px 0 black)
			drop-shadow(0.7px 0.5px 1.3px hsl(var(--shadow-color) / 0.25))
			drop-shadow(5px 3.4px 9.1px hsl(var(--shadow-color) / 0.50))
			drop-shadow(14.5px 9.9px 26.3px hsl(var(--shadow-color) / 0.75));
		
		@supports (background-clip: text) {
			color: transparent;
			background: linear-gradient(120deg, var(--accent-gradient));
			-webkit-background-clip: text;
			background-clip: text;
		}
	}
	
	& > .smol {
		transform-origin: 50% 30%;
	 	transform: scale(1.5);
	}
}

.single-flat-word {
	color: var(--accent);
	text-shadow:
		1px 1px 0 #936900,
		0px 1px 0px hsl(var(--frame-bg-dark--values) / 1),
		1px 2px 0.5px hsl(var(--frame-bg-dark--values) / 0.8),
		2px 4px 1px hsl(var(--frame-bg-dark--values) / 0.7),
		3px 6px 2px hsl(var(--frame-bg-dark--values) / 0.6),
		4px 8px 4px hsl(var(--frame-bg-dark--values) / 0.5),
		5px 12px 6px hsl(var(--frame-bg-dark--values) / 0.25),
}

// Exercise-related styles
.match-grid {
	display: grid;
	grid-template-columns: repeat(var(--c, 4), minmax(2rem, 1fr));
	gap: calc(var(--padding) / 1.5);
	list-style: none;
	
	@media (max-width: 32rem) { --c: 2; }
	@media (max-width: 20rem) { --c: 1; }
}

button.stone {
	transform: scale(var(--scale, 1.0000001));
	transition: 200ms ease-in-out;
	
	&:not([aria-disabled="true"]):where(:hover, :focus) {
		--shadow: var(--shadow-elevated);
		--l: 45%;
		--scale: 1.1;	
	}
	
	&[aria-pressed="true"] {
		--h: 42 !important; // Overwrites the inline custom property
		--stone-shadow-hue: 30;
		--l: 35%;
		--shadow: var(--shadow-elevated);
		--stone-glyph-gray: 1;
		
		--scale: 1.2;
	}
	
	&[aria-disabled="true"]:not([data-answer]) {
		--stone-shadow-hue: var(--h);
		--s: 0%;
		--l: 20;
		--stone-glyph-gray: 1;
		--scale: 0.9;
	}
	&[data-answer] {
		--stone-shadow-hue: var(--h);
		--stone-glyph-gray: 0;
	}
	&[data-answer="wrong"] {
		--h: 0 !important; // Overwrites the inline custom property
		--l: 35%;
		--scale: initial;
	}
	&[data-answer="correct"] {
		--h: 140 !important; // Overwrites the inline custom property
		--scale: 1.25;
	}
}

.glyphbank {
	--p: 0.5rem;
	--lh: calc(2.5em + var(--p));
	
	display: flex;
	flex-wrap: wrap;
	justify-content: center;
	min-width: 8rem;
	
	min-height: var(--lh);
	width: 100%;
	gap: 0 0.5em;
	
	line-height: var(--lh);
}

.glyphbank.is-added {
	max-width: min(100%, 24rem);
	padding-inline: 0.5em;
	
	background: linear-gradient(to bottom,
		transparent 4px,
		rgb(0 0 0 / 0) 0,
		67%,
		rgb(0 0 0 / 0.25) calc(var(--lh) - 2px),
		currentColor 0
	) repeat 0 0 / 100% var(--lh);
}

// .glyphbank.is-idle {}

.sven-info {
	--bg-fill: #fbb03b;
	--grow: clamp(0px, 3vw, 20px);
	--sven-offset: calc(var(--padding) * -0.5 - var(--grow));
	--sven-size: calc(var(--frame-size) + var(--padding) * 1.5 + var(--grow));
	
	position: absolute;
	inset-block-start: var(--sven-offset);
	inset-inline-end: var(--sven-offset);
	
	width: var(--sven-size);
	height: var(--sven-size);
	
	opacity: 1;
	filter: drop-shadow(2px 2px 2px hsl(0 0% 0% / 0.6)) drop-shadow(4px 4px 4px hsl(0 0% 0% / 0.3));
	transform: scale(1) rotate(15deg);
	animation: scaleIn 600ms cubic-bezier(.2, 1.8, 0, 1) 200ms backwards;
	
	@media (prefers-reduced-motion) {
		animation: none;
	}
	
	& > text {
		font-family: var(--font-alpha);
		fill: var(--accent);
	}
}

@keyframes scaleIn {
	from {
		opacity: 0;
		transform: scale(0) rotate(0deg);
	}
}

.msg-warning,
.msg-wrong,
.msg-correct {
	width: 100%;
	max-width: min(100%, 20rem);
	padding: calc(var(--padding) / 2);
	
	text-align: center;
	color: hsl(var(--h, 240) 100% 50%);
	background: hsl(var(--h, 240) 10% 10% / 0.25);
	border: 2px solid hsl(var(--h, 240) 30% 20%);
}

.msg-warning { --h: var(--hue-warning); }
.msg-wrong { --h: var(--hue-wrong); }
.msg-correct { --h: var(--hue-correct); }

// Footer styles
footer {
	position: relative;
	z-index: 2;
	
	color: var(--frame-color);
	background: var(--frame-bg);
}

footer > p {
	max-width: 40rem;
	margin-inline: auto;
	padding: calc(var(--padding) / 4) calc(var(--padding) / 2);
	
	text-align: center;
}

// Hide stuff
.visually-hidden {
  position: absolute;
  width: 1px;
  height: 1px;
  margin: -1px;
}

// Inlined image for the content frame — tucking it at the end since it's so dang long
:root {
	// Repurposed this frame from my poster: https://chriskirknielsen.com/designs/god-of-war/
	--frame-image:   url('');
}
:root {
	--rock-texture: url("");
}
              
            
!

JS

              
                // Helpers
const _NS = {
  svg: 'http://www.w3.org/2000/svg',
  html: 'http://www.w3.org/1999/xhtml',
  xml: 'http://www.w3.org/XML/1998/namespace',
  xlink: 'http://www.w3.org/1999/xlink',
  xmlns: 'http://www.w3.org/2000/xmlns/'
}
const createSvgEl = (tagName) => {
	const el = document.createElementNS(_NS.svg, tagName);
	if (tagName === 'svg') {
		el.setAttribute('xmlns', 'http://www.w3.org/2000/svg');
	}
	return el;
};
const setSvgAttr = (svgEl, attrs) => {
	const keyValPairs = Object.entries(attrs);
	for (let pair of keyValPairs) {
		const key = pair[0];
		const value = pair[1];

		if (key.indexOf(':') > -1 && svgEl.tagName.toUpperCase() !== 'SVG') {
			const ns = key.split(':').pop()[0].trim().toLowerCase();
			svgEl.setAttributeNS(_NS[ns], key, String(value));
		} else {
			svgEl.setAttribute(key, String(value));
		}
	}
};
const getRandom = (min = 0, max = 1) => Math.floor(Math.random() * (max - min + 1) + min);
const minRadius = 33;
const maxRadius = 75;
const getRandomRadius = () => getRandom(minRadius, maxRadius);
const randomItem = (array) => array[Math.floor(Math.random() * array.length)];
const randomSlice = (array, size = -1) => {
	if (size === -1) { size = array.length; } // Randomise order
    let shuffled = array.slice(0), i = array.length, min = i - size, temp, index;
    while (i-- > min) {
        index = Math.floor((i + 1) * Math.random());
        temp = shuffled[index];
        shuffled[index] = shuffled[i];
        shuffled[i] = temp;
    }
    return shuffled.slice(min);
}

// Data
const elderFuthark = [ // Thanks to Wikipedia's contributors for this neat list
    { rune: 'ᚠ', alphabet: 'f', meaning: `"cattle; wealth"` },
    { rune: 'ᚢ', alphabet: 'u', meaning: `"aurochs", Wild ox (or *ûram "water/slag"?)` },
    { rune: 'ᚦ', alphabet: 'þ', altAlphabet: 'th', meaning: `"Thurs" (see Jötunn) or *þunraz ("the god Thunraz")` },
    { rune: 'ᚨ', alphabet: 'a', meaning: `"god"` },
    { rune: 'ᚱ', alphabet: 'r', meaning: `"ride, journey"` },
    { rune: 'ᚲ', alphabet: 'k', meaning: `"ulcer"? (or *kenaz "torch"?)`, isSmol: true },
    { rune: 'ᚷ', alphabet: 'g', meaning: `"gift"` },
    { rune: 'ᚹ', alphabet: 'w', meaning: `"joy"` },
    { rune: 'ᚺ', altRune: 'ᚻ', alphabet: 'h', meaning: `"hail" (the precipitation)` },
    { rune: 'ᚾ', alphabet: 'n', meaning: `"need"` },
    { rune: 'ᛁ', alphabet: 'i', meaning: `"ice"` },
    { rune: 'ᛃ', alphabet: 'j', meaning: `"year, good year, harvest"`, isSmol: true },
    { rune: 'ᛇ', alphabet: 'ï', altAlphabet: 'æ', meaning: `"yew-tree"` },
    { rune: 'ᛈ', alphabet: 'p', meaning: `meaning unknown; possibly "pear-tree".` },
    { rune: 'ᛉ', alphabet: 'z', meaning: `"elk" (or "protection, defence")` },
    { rune: 'ᛊ', altRune: 'ᛋ', alphabet: 's', meaning: `"sun"` },
    { rune: 'ᛏ', alphabet: 't', meaning: `"the god Tiwaz"` },
    { rune: 'ᛒ', alphabet: 'b', meaning: `"birch"` },
    { rune: 'ᛖ', alphabet: 'e', meaning: `"horse"` },
    { rune: 'ᛗ', alphabet: 'm', meaning: `"man"` },
    { rune: 'ᛚ', alphabet: 'l', meaning: `"water, lake" (or possibly *laukaz "leek")` },
    { rune: 'ᛜ', alphabet: 'ŋ', meaning: `"the god Ingwaz"`, isSmol: true },
    { rune: 'ᛟ', alphabet: 'o', meaning: `"heritage, estate, possession"` },
    { rune: 'ᛞ', alphabet: 'd', meaning: `"day"` }
];
const words = [ // Objects represented as { RunicString:EnglishString }
    { word: 'odin', trap: ['u'], level: 1 },
    { word: 'þor', spellings: ['thor'], level: 1 },
    { word: 'loks', spellings: ['lox'], trap: ['g'], level: 1 },
	{ word: 'midgard', hyphenated: ['mid','gard'], level: 2 },
	{ word: 'ragnarok', hyphenated: ['rag','na','rok'], level: 2 },
	{ word: 'freja', spellings: ['freya', 'freja'], level: 2 },
	{ word: 'jormungandr', hyphenated: ['jor','mun','gandr'], level: 3 },
	{ word: 'ratatoskr', hyphenated: ['rata','toskr'], level: 2 },
	{ word: 'fenrir' },
];
let previousWord = false;
const correctMessages = [
	'By Odin’s beard, great job!', 'We got a winner!', 'Someone’s been studying!', 'Excellent!', 'That is indeed correct!', 'You’re crushing it!', 'Are you a Viking? Too easy for you!'
];
const incorrectMessages = [
	'Sorry, that wasn’t it.', 'Ah, you almost had it!', 'You’ll get it next time!', 'Study the stones again and you’ll ace this!', 'Incorrect, but give it another go!'
];

function generateStone(glyphData) {
	const index = parseInt(glyphData.index, 10);
	const hideTranslation = glyphData.hideTranslation || false;
	const swapSystems = glyphData.swapSystems || false;
	const useAltAlpha = glyphData.useAltAlpha || false;
	let hue = getRandom(210, 250);
	let angle = (!glyphData.altRune || swapSystems) ? getRandom(-3, 3) : (3 * (Math.random() < 0.5 ? -1 : 1)); // Runes with alts should always been runed to the max on either side so the toggle to an alt run is more obvious
	let textureAngle = (index * 73) % 360;
	let x = getRandom(-4, 4);
	let y = getRandom(-4, 4);
	let px = getRandom(0, 100);
	let py = getRandom(0, 100);
	let radius = `${getRandomRadius()}% ${getRandomRadius()}% ${getRandomRadius()}% ${getRandomRadius()}%`;
	const defaultFontSize = 16;
	const defaultTextY = 14;

	let stone = glyphData._element || document.createElement(glyphData._tag || 'li');
	stone.classList.add('stone');
	stone.style.setProperty('--i', index);
	stone.style.setProperty('--a', angle);
	stone.style.setProperty('--x', x);
	stone.style.setProperty('--y', y);
	stone.style.setProperty('--px', px);
	stone.style.setProperty('--py', py);
	stone.style.setProperty('--h', hue);
	stone.style.setProperty('--r', radius);
	stone.style.setProperty('--ta', textureAngle);
	if (glyphData.isSmol && !swapSystems) {
		stone.classList.add('smol');
	}

	let rune = createSvgEl('svg');
	rune.classList.add('glyph-rune');
	rune.classList.add((swapSystems) ? 'alpha' : 'runic');
	rune.setAttribute('viewBox', '0 0 16 16');
	rune.setAttribute('width', '16');
	rune.setAttribute('height', '16');

	let runeText = createSvgEl('text');
	let textY = (!swapSystems && glyphData.isSmol) ? 16 : (swapSystems && glyphData.alphabet == 'ŋ' ? defaultTextY*0.925 : defaultTextY);
	let textSize = (!swapSystems && glyphData.isSmol) ? 20 : defaultFontSize;
	rune.append(runeText);
	runeText.setAttribute('text-anchor', 'middle');
	runeText.setAttribute('x', '50%');
	runeText.setAttribute('y', textY.toString());
	runeText.setAttribute('font-size', textSize.toString());
	if (glyphData.altRune && !swapSystems) {
		stone.classList.add('has-alt');
		runeText.setAttribute('data-rune-default', '');
	}
	runeText.innerHTML = swapSystems ? (useAltAlpha ? glyphData.altAlphabet : glyphData.alphabet) : glyphData.rune;
	stone.append(rune);
	
	if (glyphData.altRune && !swapSystems) {
		let altRuneText = createSvgEl('text');
		let altTextY = 16;
		let textSize = (!swapSystems && glyphData.isSmol) ? 20 : 16;
		rune.append(altRuneText);
		altRuneText.setAttribute('text-anchor', 'middle');
		altRuneText.setAttribute('x', '50%');
		altRuneText.setAttribute('y', defaultTextY);
		altRuneText.setAttribute('font-size', defaultFontSize);
		altRuneText.setAttribute('data-rune-alt', '');
		altRuneText.innerHTML = glyphData.altRune;
	}

	if (!hideTranslation) {
		let alpha = document.createElement('span');
		alpha.classList.add('alpha');
		alpha.innerText = swapSystems ? glyphData.rune : glyphData.alphabet;
		stone.append(alpha);
	}
	
	return stone;
}

function generateSven(text, bgOverride = null) {
	const svg = createSvgEl('svg');
	setSvgAttr(svg, {
		'xmlns:xlink': 'http://www.w3.org/1999/xlink',
		'width': 130,
		'height': 130,
		'viewBox': '0 0 130 130',
	});
	
	const sven = createSvgEl('use');
	setSvgAttr(sven, {
		'width': '100%',
		'height': '100%',
		'x': '0',
		'y': '0',
		'href': '#sven-badge',
	});
	svg.append(sven);
	
	const textEl = createSvgEl('text');
	textEl.setAttribute('font-size','15');
	textEl.setAttribute('x', '50%');
	textEl.setAttribute('y', '123');
	textEl.setAttribute('text-anchor', 'middle');
	textEl.innerHTML = text;
	svg.append(textEl);
	
	if (bgOverride) {
		svg.style.setProperty('--bg-fill', bgOverride);
	}
	
	return svg;
}

function getRuneFromTranslit(letter) {
		let matchingRunic = elderFuthark.find(f => f.alphabet === letter || f.altAlphabet === letter);
		return matchingRunic.rune;
}

// Show the Futhark stones in a grid
function populateFutharkGrid(){
	const grid = document.querySelector('.grid');
	const glyphs = document.createDocumentFragment();
	
	for (let index = 0; index < elderFuthark.length; index++) {
		const runeData = Object.assign({ index }, elderFuthark[index]);
		const glyph = generateStone(runeData);
		glyphs.append(glyph);
	}
	grid.append(glyphs);
}
populateFutharkGrid();

// Toggle between classic and alternative rune style
const glyphToggler = document.getElementById('show-alt-runes');
function setGlyphStyle() {
	const isAltStyle = glyphToggler.checked;
	const runesWithAlt = Array.from(document.querySelectorAll('#futhark .stone.has-alt'));
	runesWithAlt.forEach(function (glyphStone) {
		glyphStone.classList.toggle('is-alt', isAltStyle);
	})
}
setGlyphStyle(); // Ensure state is properly defined on page load
glyphToggler.addEventListener('change', setGlyphStyle);

// Generate a random exercise
const newExerciseButton = document.getElementById('new-exercise');
const exerciseContent = document.getElementById('exercise');

function makeMatchExercise() {
	const template = document.getElementById('exercise-match').content.cloneNode(true);
	const targetSystem = Math.random() < 0.5 ? 'FUTHARK' : 'ALPHA';
	const allAnswers = randomSlice(elderFuthark, 4);
	const correctAnswer = randomItem(allAnswers);
	let correctOption;
	const optionItems = Array.from(template.querySelectorAll(`[data-match="options"] > [data-match="option"] > button`));
	
	const warningEl = template.querySelector('[data-match="warning"]');
	const resultEl = template.querySelector('[data-match="result"]');
	
	const optionClickHandler = function (e) {
		const wasPressed = e.target.closest('[data-match="option"] > button').getAttribute('aria-pressed') === 'true';
		optionItems.forEach(opt => opt.setAttribute('aria-pressed', 'false'));
		e.target.closest('[data-match="option"] > button').setAttribute('aria-pressed', 'true');

		// Reset warning
		warningEl.innerText = '';
		warningEl.hidden = true;
	};
	const optionKeydownHandler = function (e) {
		// Implement radio behaviour
		const key = e.key;
		const options = e.target.closest('[data-match="options"]');
		
		// Skip to next item
		if (key === 'Tab') {
			e.preventDefault();
			if (e.shiftKey) {
				e.target.closest('.content').focus();
			} else {
				checkAnswerButton.focus();
			}
			return false;
		}
		
		const itemIndex = parseInt(e.target.parentElement.getAttribute('data-index'), 10);
		let newItemIndex = itemIndex;
		
		if (['ArrowRight', 'ArrowDown'].includes(key)) {
			newItemIndex++;
		} else if (['ArrowLeft', 'ArrowUp'].includes(key)) {
			newItemIndex--;
		} else if (key !== 'Space') {
			return;
		}
		
		if (newItemIndex > optionItems.length) { newItemIndex = 1; }
		else if (newItemIndex <= 0) { newItemIndex = optionItems.length; }
		const optionItem = options.querySelector(`[data-match="option"][data-index="${newItemIndex}"] > button`);

		optionItem.focus();
		optionItem.click();
	};
	for (let a = 0; a < allAnswers.length; a++) {
		const answer = allAnswers[a];
		const optionItem = template.querySelector(`[data-match="options"] > [data-match="option"][data-index="${a + 1}"] > button`);
		optionItem.innerText = '';
		generateStone(Object.assign({ _element: optionItem, index: a, hideTranslation: true, swapSystems: targetSystem === 'FUTHARK', useAltAlpha: answer.altAlphabet === 'th' && targetSystem === 'FUTHARK' }, answer));
		if (answer.rune === correctAnswer.rune) { correctOption = optionItem; }
		optionItem.addEventListener('click', optionClickHandler);
		optionItem.addEventListener('keydown', optionKeydownHandler);
	}
	
	const itemToMatch = template.querySelector('[data-match="glyph-to-match"]');
	const glyphToMatch = targetSystem === 'ALPHA' ? (correctAnswer.altAlphabet === 'th' ? correctAnswer.altAlphabet : correctAnswer.alphabet).toUpperCase() : correctAnswer.rune;
	const glyphCssClass = [];
	if (targetSystem === 'FUTHARK') { glyphCssClass.push('runic'); }
	if (targetSystem === 'FUTHARK' && correctAnswer.isSmol) { glyphCssClass.push('smol'); }
	itemToMatch.innerHTML = `<span class="${glyphCssClass.join(' ')}">${glyphToMatch}</span>`
	
	const checkAnswerButton = template.querySelector('[data-match="verify"]');
	const checkAnswerHandler = function (e) {
		const selectedOption = optionItems.find(opt => opt.getAttribute('aria-pressed') === 'true');
		if (!selectedOption) {
			warningEl.hidden = false;
			warningEl.innerText = 'Please select an answer first!';
			warningEl.focus();
			return;
		}
		
		const isCorrect = (correctOption === selectedOption);
		
		checkAnswerButton.setAttribute('aria-disabled', 'true');
		checkAnswerButton.removeEventListener('click', checkAnswerHandler);
		optionItems.forEach(opt => {
			opt.setAttribute('aria-disabled', 'true');
			opt.removeEventListener('click', optionClickHandler);
		});
		
		checkAnswerButton.hidden = true;
		warningEl.hidden = true;
		resultEl.hidden = false;
		
		if (isCorrect) {
			resultEl.classList.add('msg-correct');
			resultEl.innerText = randomItem(correctMessages);
		} else {
			resultEl.classList.add('msg-wrong');
			resultEl.innerText = randomItem(incorrectMessages);
			selectedOption.setAttribute('data-answer', 'wrong');
		}
		
		correctOption.setAttribute('data-answer', 'correct');
		newExerciseButton.focus();
	}
	checkAnswerButton.addEventListener('click', checkAnswerHandler)
	return template;
}

function makeSpellExercise() {
	const template = document.getElementById('exercise-spell').content.cloneNode(true);
	const targetSystem = Math.random() < 0.5 ? 'FUTHARK' : 'ALPHA';
	const wordShown = template.querySelector('[data-spell="word-to-spell"]');
	const prompt = template.querySelector('[data-spell="prompt"]');
	const wordToSpell = randomItem(words.filter(w => w.word !== previousWord)); // Don't show the same twice in a row
	if (!wordToSpell.spellings) { wordToSpell.spellings = [wordToSpell.word]; } // If there is no specified spelling, use the same as the rune transliteration
	if (!Array.isArray(wordToSpell.spellings)) { wordToSpell.spellings = [wordToSpell.spellings]; } // In case I forget to wrap it in a string…
	previousWord = wordToSpell.word;
	
	if (wordToSpell.level > 1) {
		const levelText = wordToSpell.level === 3 ? 'Hard' : 'Medium' ;
		const sven = generateSven(`${levelText} Difficulty`, wordToSpell.level === 3 ? 'hsl(5, 100%, 50%)' : null);
		template.append(sven);
		sven.classList.add('sven-info')
	}
	
	const warningEl = template.querySelector('[data-spell="warning"]');
	const resultEl = template.querySelector('[data-spell="result"]');
	
	const hasTraps = wordToSpell.trap && targetSystem === 'FUTHARK';
	const trapList = hasTraps ? wordToSpell.trap.filter(_ => Math.random() > 0.333) : []; // Keep a random amount of traps (67% chance of keeping one of the traps)
	const trapCount = trapList.length;
	const wordFuthark = wordToSpell.word; // Latin transliteration
	const wordAlpha = wordToSpell.spellings[0]; // Pick the first spelling as the common answer, or prompt
	const wordFutharkInRunic = wordFuthark.split('').map(getRuneFromTranslit).join('');
	wordToSpell.runic = wordFutharkInRunic; // Store the word in runic
	const wordFutharkInRunicHyphenated = (wordToSpell.hasOwnProperty('hyphenated') ? wordToSpell.hyphenated : [wordToSpell.word]).map(h => h.split('').map(getRuneFromTranslit).join('')).join('&shy;');
	const wordAlphaHyphenated = wordToSpell.hasOwnProperty('hyphenated') ? wordToSpell.hyphenated.join('&shy;') : wordAlpha;
	prompt.innerText = prompt.innerText
		.replace('{{type}}', targetSystem === 'FUTHARK' ? 'word' : 'runes')
		.replace('{{glyphs}}', targetSystem === 'FUTHARK' ? 'runes' : 'letters');
	const extraGlyphCount = Math.max(0, getRandom(0, 3) - trapCount) + trapCount; // Randomly add zero to three items
	const extraGlyphData = [];
	for (let g = 1; g <= extraGlyphCount; g++) {
		// Don't add a random 'th' option as it has 2 characters, and the single character is basically a rune in itself
		extraGlyphData.push(randomItem(elderFuthark.filter(l => l.altAlphabet !== 'th')));
	}
	for (let t = 0; t < trapCount ; t++) {
		// Add the trap letter to the list of available letters
		const trapLetter = trapList[t];
		extraGlyphData.push(elderFuthark.find(l => l.alphabet === trapLetter));
	}

	const extraGlyphs = extraGlyphData.map(d => (targetSystem === 'FUTHARK') ? d.rune : d.alphabet);
	const composed = template.querySelector('[data-spell="composed"]');
	const glyphBank = template.querySelector('[data-spell="glyphs"]');
	let glyphBankOptions = [];
	if (targetSystem === 'FUTHARK') {
		glyphBankOptions = glyphBankOptions.concat(wordFutharkInRunic.split(''));
	} else {
		glyphBankOptions = glyphBankOptions.concat(wordAlpha.split(''));
	}
	const randomisedGlyphBankOptions = randomSlice(glyphBankOptions.concat(randomSlice(extraGlyphs)));
	for (let g = 0; g < randomisedGlyphBankOptions.length; g++) {
		let glyphOption = randomisedGlyphBankOptions[g];
		const glyphTileTemplate = document.getElementById('exercise-spell-tile').content.cloneNode(true);
		const glyphTile = glyphTileTemplate.querySelector('[data-spell="glyph"]');
		const tileButton = glyphTile.querySelector('[data-spell="opt-button"]');
		tileButton.classList.add(targetSystem === 'FUTHARK' ? 'runic' : 'alpha');
		glyphTile.setAttribute('data-option-index', g);
		tileButton.innerText = glyphOption;
		glyphBank.append(glyphTile);
	}
	
	const glyphBankClickHandler = function (e) {
		const tileItem = e.target.closest('[data-spell="glyph"]');
		if (!tileItem) { return; }
		const tileButton = tileItem.querySelector('[data-spell="opt-button"]');
		if (tileButton.getAttribute('aria-pressed') === 'true') { return; }
		const clonedItem = tileItem.cloneNode(true);
		tileButton.setAttribute('aria-pressed', 'true');
		tileButton.inert = true;
		console.log(tileButton.inert);
		clonedItem.querySelector('[data-spell="opt-button"]').removeAttribute('aria-pressed');
		composed.append(clonedItem);
	};
	glyphBank.addEventListener('click', glyphBankClickHandler);
	
	const composedClickHandler = function (e) {
		const tileItem = e.target.closest('[data-spell="glyph"]');
		if (!tileItem) { return; }
		const tileIndex = tileItem.getAttribute('data-option-index');
		const glyphBankItem = glyphBank.querySelector(`[data-option-index="${tileIndex}"]`);
		const glyphButtonItem = glyphBankItem.querySelector('[data-spell="opt-button"]');
		glyphButtonItem.setAttribute('aria-pressed', 'false');
		glyphButtonItem.inert = false;
		console.log(glyphButtonItem.inert);
		composed.removeChild(tileItem);
	};
	composed.addEventListener('click', composedClickHandler)
	
	wordShown.innerHTML = targetSystem === 'FUTHARK' ? wordAlphaHyphenated : wordFutharkInRunicHyphenated;
	wordShown.classList.add(targetSystem === 'FUTHARK' ? 'alpha' : 'runic');
	
	const checkAnswerButton = template.querySelector('[data-spell="verify"]');
	const checkAnswerHandler = function (e) {
		if (composed.children.length === 0) {
			warningEl.hidden = false;
			warningEl.innerText = 'Please add your answer first!';
			warningEl.focus();
			return;
		}
		
		let isCorrect = false;
		const userInputTiles = Array.from(composed.children);
		let userInput = userInputTiles.map(c => c.innerText).join('');
		
		if (targetSystem === 'FUTHARK') {
			isCorrect = userInput.trim() === wordToSpell.runic.trim();
		} else {
			isCorrect = wordToSpell.spellings.includes(userInput.trim().toLowerCase());
		}
		
		userInputTiles.forEach((item, index) => {
			const tile = item.querySelector('[data-spell="opt-button"]');
			tile.style.setProperty('--tile-index', index);
		})
		
		glyphBank.removeEventListener('click', glyphBankClickHandler);
		composed.removeEventListener('click', composedClickHandler);
		
		checkAnswerButton.setAttribute('aria-disabled', 'true');
		checkAnswerButton.removeEventListener('click', checkAnswerHandler);
		
		checkAnswerButton.hidden = true;
		warningEl.hidden = true;
		resultEl.hidden = false;
		
		if (isCorrect) {
			resultEl.classList.add('msg-correct');
			resultEl.innerText = randomItem(correctMessages);
			composed.setAttribute('data-answer', 'correct');
		} else {
			resultEl.classList.add('msg-wrong');
			resultEl.innerText = randomItem(incorrectMessages);
			composed.setAttribute('data-answer', 'wrong');
		}
		
		newExerciseButton.focus();
	}
	checkAnswerButton.addEventListener('click', checkAnswerHandler)
	
	return template;
}

function generateExercise() {
	const prefersReducedMotion = window.matchMedia('(prefers-reduced-motion: reduce)').matches
	const timing = prefersReducedMotion ? 1 : 200;
	const type = Math.random() < 0.5 ? 'match' : 'spell';
	let height = 'auto';
	
	const setOpacity = async (opacity) => {
		return new Promise(function (resolve, _) {
			exerciseContent
				.animate({ opacity: Math.max(0, Math.min(1, parseFloat(opacity))) }, { duration: timing, iterations: 1, easing: 'ease-out' })
   				.onfinish = (e) => {
					e.target.effect.target.style.opacity = opacity;
					resolve(true);
			   };
		});
	}
	const removeContent = async () => {
		// Remove any existing content
		while (exerciseContent.firstChild) {
			exerciseContent.removeChild(exerciseContent.lastChild);
		}
		return true;
	};
	setOpacity(0)
		.then(async () => { height = window.getComputedStyle(exerciseContent).height; return true; })
		.then(removeContent)
		.then(() => {
			const prevHeight = height;
			const exercise = type === 'spell' ? makeSpellExercise() : makeMatchExercise();
			exerciseContent.append(exercise);
			const newHeight = window.getComputedStyle(exerciseContent).height;
			exerciseContent.style.height = prevHeight;
			height = newHeight;
			return new Promise((resolve, _) => setTimeout(resolve, 0));
		})
		.then(async () => {
			return new Promise(function (resolve, _) {
				exerciseContent
					.animate({ height: height }, { duration: Math.ceil(timing/2), iterations: 1, easing: 'ease-out' })
					.onfinish = (e) => {
						e.target.effect.target.style.height = height;
						resolve(true);
					};
			});
		})
		.then(() => setOpacity(1))
		.then(() => {
			height = 'auto';
			exerciseContent.style.height = 'auto';
			exerciseContent.focus();
		});
}
newExerciseButton.addEventListener('click', generateExercise);

// Prevent form submissions
Array.from(document.querySelectorAll('form')).forEach(function(form) {
	form.addEventListener('submit', function (e) {
		e.preventDefault();
		return false;
	});
});

// Lo-fi tab navigation…
document.addEventListener('click', function(e) {
	const menuLink = e.target.closest('.menu-link');
	if (!menuLink) { return; }
	const targetSection = document.querySelector(menuLink.getAttribute('href'));
	Array.from(document.querySelectorAll('.menu-link')).forEach(m => {
		const linkTarget = document.querySelector(m.getAttribute('href'));
		m.setAttribute('aria-current', 'false');
		linkTarget.inert = true;
		linkTarget.setAttribute('aria-hidden', 'true');
	});
	menuLink.setAttribute('aria-current', 'true');
	targetSection.addEventListener('transitionend', function(e){
		targetSection.focus(); // Focus once transition is over
	}, { once: true });
	targetSection.inert = false;
	targetSection.setAttribute('aria-hidden', 'false');
	
	e.preventDefault();
	return false;
});
              
            
!
999px

Console