open-webui/src/lib/components/common/SVGPanZoom.svelte

133 lines
3.4 KiB
Svelte
Raw Normal View History

2024-10-06 19:51:29 +00:00
<script lang="ts">
2025-02-18 02:50:23 +00:00
import fileSaver from 'file-saver';
const { saveAs } = fileSaver;
import { toast } from 'svelte-sonner';
import panzoom, { type PanZoom } from 'panzoom';
import DOMPurify from 'dompurify';
2024-10-11 07:10:00 +00:00
import { onMount, getContext } from 'svelte';
const i18n = getContext('i18n');
2025-02-18 02:50:23 +00:00
import { copyToClipboard } from '$lib/utils';
2024-10-06 19:51:29 +00:00
2024-10-11 07:10:00 +00:00
import DocumentDuplicate from '../icons/DocumentDuplicate.svelte';
import Tooltip from './Tooltip.svelte';
import Clipboard from '../icons/Clipboard.svelte';
import Reset from '../icons/Reset.svelte';
2025-09-15 19:28:16 +00:00
import Download from '../icons/Download.svelte';
2024-10-06 19:51:29 +00:00
export let className = '';
export let svg = '';
2024-10-11 07:10:00 +00:00
export let content = '';
2024-10-06 19:51:29 +00:00
let instance: PanZoom;
2024-10-06 19:51:29 +00:00
let sceneParentElement: HTMLElement;
let sceneElement: HTMLElement;
$: if (sceneElement) {
instance = panzoom(sceneElement, {
bounds: true,
boundsPadding: 0.1,
zoomSpeed: 0.065
});
}
2025-02-18 02:50:23 +00:00
const resetPanZoomViewport = () => {
2024-12-28 23:50:23 +00:00
instance.moveTo(0, 0);
instance.zoomAbs(0, 0, 1);
console.log(instance.getTransform());
2025-02-18 02:50:23 +00:00
};
const downloadAsSVG = () => {
const svgBlob = new Blob([svg], { type: 'image/svg+xml' });
saveAs(svgBlob, `diagram.svg`);
};
2024-10-06 19:51:29 +00:00
</script>
2024-10-11 07:10:00 +00:00
<div bind:this={sceneParentElement} class="relative {className}">
2024-10-06 19:51:29 +00:00
<div bind:this={sceneElement} class="flex h-full max-h-full justify-center items-center">
2025-09-25 19:29:50 +00:00
{@html DOMPurify.sanitize(svg, {
USE_PROFILES: { svg: true, svgFilters: true }, // allow <svg>, <defs>, <filter>, etc.
WHOLE_DOCUMENT: false,
ADD_TAGS: ['style', 'foreignObject'], // include foreignObject if using HTML labels
ADD_ATTR: [
'class',
'style',
'id',
'data-*',
'viewBox',
'preserveAspectRatio',
// markers / arrows
'markerWidth',
'markerHeight',
'markerUnits',
'refX',
'refY',
'orient',
// hrefs (for gradients, markers, etc.)
'href',
'xlink:href',
// text positioning
'dominant-baseline',
'text-anchor',
// pattern / clip / mask units
'clipPathUnits',
'filterUnits',
'patternUnits',
'patternContentUnits',
'maskUnits',
// a11y niceties
'role',
'aria-label',
'aria-labelledby',
'aria-hidden',
'tabindex'
],
SANITIZE_DOM: true
})}
2024-10-06 19:51:29 +00:00
</div>
2024-10-11 07:10:00 +00:00
{#if content}
2025-09-15 19:55:05 +00:00
<div class=" absolute top-2.5 right-2.5">
2025-02-18 02:50:23 +00:00
<div class="flex gap-1">
<Tooltip content={$i18n.t('Download as SVG')}>
<button
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"
on:click={() => {
downloadAsSVG();
}}
>
2025-09-15 19:28:16 +00:00
<Download className=" size-4" />
2025-02-18 02:50:23 +00:00
</button>
</Tooltip>
<Tooltip content={$i18n.t('Reset view')}>
<button
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"
on:click={() => {
resetPanZoomViewport();
}}
>
<Reset className=" size-4" />
</button>
</Tooltip>
<Tooltip content={$i18n.t('Copy to clipboard')}>
<button
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"
on:click={() => {
copyToClipboard(content);
toast.success($i18n.t('Copied to clipboard'));
}}
>
<Clipboard className=" size-4" strokeWidth="1.5" />
</button>
</Tooltip>
</div>
</div>
2024-10-11 07:10:00 +00:00
{/if}
2024-10-06 19:51:29 +00:00
</div>