aboutsummaryrefslogtreecommitdiff
path: root/src/lib/Collapsible.svelte
blob: a7b6105b751b0e61d854a51edf9fb78cecd5f6f8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
<!-- 
	@component
	
	Collapsible content that works without JS but is enhanced by it.
 -->
<script lang="ts">
	import { browser } from '$app/env'
	import { onMount } from 'svelte'
	import { cleanId } from './utils'

	let open: boolean
	export let id: string | undefined = undefined
	$: elementId = id?.replace(/_/, '-')
	export let lazy = true

	let detailsEl

	function checkHash() {
		if (elementId && elementId === location.hash.slice(1)) {
			console.log('open')
			open = true
			requestAnimationFrame(() => {
				// scroll to the element
				window.scrollTo({
					top: detailsEl.offsetTop,
					behavior: 'smooth',
				})
			})
		}
	}

	onMount(() => {
		if (browser) checkHash()
	})
	if (browser) checkHash()
</script>

<svelte:window on:hashchange={checkHash} />

<details bind:open id={elementId} bind:this={detailsEl}>
	<summary>
		<slot name="title">
			<h2>{id ? cleanId(id) : 'Details'}</h2>
		</slot>
	</summary>
	<div>
		{#if lazy}
			<!--
			We do this so images and other things inside the content are not
			loaded until it's open.
			For some reason browsers don't handle this, although they should.
		-->
			<noscript>
				<slot />
			</noscript>
			{#if open}
				<slot />
			{/if}
		{:else}
			<slot />
		{/if}
	</div>
</details>

<style>
	details {
		margin-bottom: 0.5em;
	}
	summary > :global(*) {
		display: inline;
	}

	summary {
		cursor: pointer;
		list-style: none;
	}
	summary::marker,
	summary::-webkit-details-marker {
		content: '';
		display: none;
	}

	summary:focus {
		outline: none;
		box-shadow: 0 0 0 2pt #06e7;
		border-radius: 0.1em;
		width: fit-content;
		padding-right: 0.2em;
	}

	summary::before {
		/* the background image is an arrow pointing down */
		background-image: url();
		width: 20px;
		height: 20px;
		display: inline-block;
		margin-right: 1em;
		content: '';
	}
	details[open] summary::before {
		transform: rotate(180deg);
	}
</style>