Skip to content

Commit 0c46102

Browse files
committed
enh: svg diagram download button
1 parent 93731e4 commit 0c46102

File tree

2 files changed

+64
-33
lines changed

2 files changed

+64
-33
lines changed
Lines changed: 52 additions & 31 deletions
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,22 @@
11
<script lang="ts">
2+
import fileSaver from 'file-saver';
3+
const { saveAs } = fileSaver;
4+
5+
import { toast } from 'svelte-sonner';
6+
7+
import panzoom, { type PanZoom } from 'panzoom';
8+
import DOMPurify from 'dompurify';
9+
210
import { onMount, getContext } from 'svelte';
311
const i18n = getContext('i18n');
412
5-
import panzoom, { type PanZoom } from 'panzoom';
13+
import { copyToClipboard } from '$lib/utils';
614
7-
import DOMPurify from 'dompurify';
815
import DocumentDuplicate from '../icons/DocumentDuplicate.svelte';
9-
import { copyToClipboard } from '$lib/utils';
10-
import { toast } from 'svelte-sonner';
1116
import Tooltip from './Tooltip.svelte';
1217
import Clipboard from '../icons/Clipboard.svelte';
1318
import Reset from '../icons/Reset.svelte';
19+
import ArrowDownTray from '../icons/ArrowDownTray.svelte';
1420
1521
export let className = '';
1622
export let svg = '';
@@ -29,12 +35,16 @@
2935
zoomSpeed: 0.065
3036
});
3137
}
32-
function resetPanZoomViewport() {
33-
console.log('Reset View');
38+
const resetPanZoomViewport = () => {
3439
instance.moveTo(0, 0);
3540
instance.zoomAbs(0, 0, 1);
3641
console.log(instance.getTransform());
37-
}
42+
};
43+
44+
const downloadAsSVG = () => {
45+
const svgBlob = new Blob([svg], { type: 'image/svg+xml' });
46+
saveAs(svgBlob, `diagram.svg`);
47+
};
3848
</script>
3949

4050
<div bind:this={sceneParentElement} class="relative {className}">
@@ -44,30 +54,41 @@
4454

4555
{#if content}
4656
<div class=" absolute top-1 right-1">
47-
<Tooltip content={$i18n.t('Copy to clipboard')}>
48-
<button
49-
class="p-1.5 rounded-lg border border-gray-100 dark:border-none dark:bg-gray-850 hover:bg-gray-50 dark:hover:bg-gray-800 transition"
50-
on:click={() => {
51-
copyToClipboard(content);
52-
toast.success($i18n.t('Copied to clipboard'));
53-
}}
54-
>
55-
<Clipboard className=" size-4" strokeWidth="1.5" />
56-
</button>
57-
</Tooltip>
58-
</div>
59-
<div class=" absolute top-1 right-10">
60-
<Tooltip content={$i18n.t('Reset view')}>
61-
<button
62-
class="p-1.5 rounded-lg border border-gray-100 dark:border-none dark:bg-gray-850 hover:bg-gray-50 dark:hover:bg-gray-800 transition"
63-
on:click={() => {
64-
resetPanZoomViewport();
65-
toast.success($i18n.t('Reset view'));
66-
}}
67-
>
68-
<Reset className=" size-4" />
69-
</button>
70-
</Tooltip>
57+
<div class="flex gap-1">
58+
<Tooltip content={$i18n.t('Download as SVG')}>
59+
<button
60+
class="p-1.5 rounded-lg border border-gray-100 dark:border-none dark:bg-gray-850 hover:bg-gray-50 dark:hover:bg-gray-800 transition"
61+
on:click={() => {
62+
downloadAsSVG();
63+
}}
64+
>
65+
<ArrowDownTray className=" size-4" />
66+
</button>
67+
</Tooltip>
68+
69+
<Tooltip content={$i18n.t('Reset view')}>
70+
<button
71+
class="p-1.5 rounded-lg border border-gray-100 dark:border-none dark:bg-gray-850 hover:bg-gray-50 dark:hover:bg-gray-800 transition"
72+
on:click={() => {
73+
resetPanZoomViewport();
74+
}}
75+
>
76+
<Reset className=" size-4" />
77+
</button>
78+
</Tooltip>
79+
80+
<Tooltip content={$i18n.t('Copy to clipboard')}>
81+
<button
82+
class="p-1.5 rounded-lg border border-gray-100 dark:border-none dark:bg-gray-850 hover:bg-gray-50 dark:hover:bg-gray-800 transition"
83+
on:click={() => {
84+
copyToClipboard(content);
85+
toast.success($i18n.t('Copied to clipboard'));
86+
}}
87+
>
88+
<Clipboard className=" size-4" strokeWidth="1.5" />
89+
</button>
90+
</Tooltip>
91+
</div>
7192
</div>
7293
{/if}
7394
</div>
Lines changed: 12 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,19 @@
11
<script lang="ts">
22
export let className = 'size-4';
3+
export let strokeWidth = '2';
34
</script>
45

5-
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512" fill="currentColor" class={className}>
6+
<svg
7+
xmlns="http://www.w3.org/2000/svg"
8+
fill="none"
9+
viewBox="0 0 24 24"
10+
stroke-width={strokeWidth}
11+
stroke="currentColor"
12+
class={className}
13+
>
614
<path
7-
d="M463.5 224l8.5 0c13.3 0 24-10.7 24-24l0-128c0-9.7-5.8-18.5-14.8-22.2s-19.3-1.7-26.2 5.2L413.4 96.6c-87.6-86.5-228.7-86.2-315.8 1c-87.5 87.5-87.5 229.3 0 316.8s229.3 87.5 316.8 0c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0c-62.5 62.5-163.8 62.5-226.3 0s-62.5-163.8 0-226.3c62.2-62.2 162.7-62.5 225.3-1L327 183c-6.9 6.9-8.9 17.2-5.2 26.2s12.5 14.8 22.2 14.8l119.5 0z"
15+
stroke-linecap="round"
16+
stroke-linejoin="round"
17+
d="M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0 3.181 3.183a8.25 8.25 0 0 0 13.803-3.7M4.031 9.865a8.25 8.25 0 0 1 13.803-3.7l3.181 3.182m0-4.991v4.99"
818
/>
919
</svg>

0 commit comments

Comments
 (0)