|
<script lang="ts"> |
|
import { onDestroy, onMount } from 'svelte'; |
|
import panzoom, { type PanZoom } from 'panzoom'; |
|
|
|
import fileSaver from 'file-saver'; |
|
const { saveAs } = fileSaver; |
|
|
|
export let show = false; |
|
export let src = ''; |
|
export let alt = ''; |
|
|
|
let mounted = false; |
|
|
|
let previewElement = null; |
|
|
|
let instance: PanZoom; |
|
|
|
let sceneParentElement: HTMLElement; |
|
let sceneElement: HTMLElement; |
|
|
|
$: if (sceneElement) { |
|
instance = panzoom(sceneElement, { |
|
bounds: true, |
|
boundsPadding: 0.1, |
|
|
|
zoomSpeed: 0.065 |
|
}); |
|
} |
|
const resetPanZoomViewport = () => { |
|
instance.moveTo(0, 0); |
|
instance.zoomAbs(0, 0, 1); |
|
console.log(instance.getTransform()); |
|
}; |
|
|
|
const handleKeyDown = (event: KeyboardEvent) => { |
|
if (event.key === 'Escape') { |
|
console.log('Escape'); |
|
show = false; |
|
} |
|
}; |
|
|
|
onMount(() => { |
|
mounted = true; |
|
}); |
|
|
|
$: if (show && previewElement) { |
|
document.body.appendChild(previewElement); |
|
window.addEventListener('keydown', handleKeyDown); |
|
document.body.style.overflow = 'hidden'; |
|
} else if (previewElement) { |
|
window.removeEventListener('keydown', handleKeyDown); |
|
document.body.removeChild(previewElement); |
|
document.body.style.overflow = 'unset'; |
|
} |
|
|
|
onDestroy(() => { |
|
show = false; |
|
|
|
if (previewElement) { |
|
document.body.removeChild(previewElement); |
|
} |
|
}); |
|
</script> |
|
|
|
{#if show} |
|
|
|
|
|
<div |
|
bind:this={previewElement} |
|
class="modal fixed top-0 right-0 left-0 bottom-0 bg-black text-white w-full min-h-screen h-screen flex justify-center z-9999 overflow-hidden overscroll-contain" |
|
> |
|
<div class=" absolute left-0 w-full flex justify-between select-none z-20"> |
|
<div> |
|
<button |
|
class=" p-5" |
|
on:pointerdown={(e) => { |
|
e.stopImmediatePropagation(); |
|
e.preventDefault(); |
|
show = false; |
|
}} |
|
on:click={(e) => { |
|
show = false; |
|
}} |
|
> |
|
<svg |
|
xmlns="http://www.w3.org/2000/svg" |
|
fill="none" |
|
viewBox="0 0 24 24" |
|
stroke-width="2" |
|
stroke="currentColor" |
|
class="w-6 h-6" |
|
> |
|
<path stroke-linecap="round" stroke-linejoin="round" d="M6 18 18 6M6 6l12 12" /> |
|
</svg> |
|
</button> |
|
</div> |
|
|
|
<div> |
|
<button |
|
class=" p-5 z-999" |
|
on:click={() => { |
|
if (src.startsWith('data:image/')) { |
|
const base64Data = src.split(',')[1]; |
|
const blob = new Blob([Uint8Array.from(atob(base64Data), (c) => c.charCodeAt(0))], { |
|
type: 'image/png' |
|
}); |
|
|
|
const mimeType = blob.type || 'image/png'; |
|
// create file name based on the MIME type, alt should be a valid file name with extension |
|
const fileName = alt |
|
? `${alt.replaceAll('.', '')}.${mimeType.split('/')[1]}` |
|
: 'download.png'; |
|
|
|
// Use FileSaver to save the blob |
|
saveAs(blob, fileName); |
|
return; |
|
} else if (src.startsWith('blob:')) { |
|
// Handle blob URLs |
|
fetch(src) |
|
.then((response) => response.blob()) |
|
.then((blob) => { |
|
// detect the MIME type from the blob |
|
const mimeType = blob.type || 'image/png'; |
|
|
|
// Create a new Blob with the correct MIME type |
|
const blobWithType = new Blob([blob], { type: mimeType }); |
|
|
|
// create file name based on the MIME type, alt should be a valid file name with extension |
|
const fileName = alt |
|
? `${alt.replaceAll('.', '')}.${mimeType.split('/')[1]}` |
|
: 'download.png'; |
|
|
|
// Use FileSaver to save the blob |
|
saveAs(blobWithType, fileName); |
|
}) |
|
.catch((error) => { |
|
console.error('Error downloading blob:', error); |
|
}); |
|
return; |
|
} else if ( |
|
src.startsWith('/') || |
|
src.startsWith('http://') || |
|
src.startsWith('https://') |
|
) { |
|
// Handle remote URLs |
|
fetch(src) |
|
.then((response) => response.blob()) |
|
.then((blob) => { |
|
// detect the MIME type from the blob |
|
const mimeType = blob.type || 'image/png'; |
|
|
|
// Create a new Blob with the correct MIME type |
|
const blobWithType = new Blob([blob], { type: mimeType }); |
|
|
|
// create file name based on the MIME type, alt should be a valid file name with extension |
|
const fileName = alt |
|
? `${alt.replaceAll('.', '')}.${mimeType.split('/')[1]}` |
|
: 'download.png'; |
|
|
|
// Use FileSaver to save the blob |
|
saveAs(blobWithType, fileName); |
|
}) |
|
.catch((error) => { |
|
console.error('Error downloading remote image:', error); |
|
}); |
|
return; |
|
} |
|
}} |
|
> |
|
<svg |
|
xmlns="http://www.w3.org/2000/svg" |
|
viewBox="0 0 20 20" |
|
fill="currentColor" |
|
class="w-6 h-6" |
|
> |
|
<path |
|
d="M10.75 2.75a.75.75 0 0 0-1.5 0v8.614L6.295 8.235a.75.75 0 1 0-1.09 1.03l4.25 4.5a.75.75 0 0 0 1.09 0l4.25-4.5a.75.75 0 0 0-1.09-1.03l-2.955 3.129V2.75Z" |
|
/> |
|
<path |
|
d="M3.5 12.75a.75.75 0 0 0-1.5 0v2.5A2.75 2.75 0 0 0 4.75 18h10.5A2.75 2.75 0 0 0 18 15.25v-2.5a.75.75 0 0 0-1.5 0v2.5c0 .69-.56 1.25-1.25 1.25H4.75c-.69 0-1.25-.56-1.25-1.25v-2.5Z" |
|
/> |
|
</svg> |
|
</button> |
|
</div> |
|
</div> |
|
<div class="flex h-full max-h-full justify-center items-center z-0"> |
|
<img |
|
bind:this={sceneElement} |
|
{src} |
|
{alt} |
|
class=" mx-auto h-full object-scale-down select-none" |
|
draggable="false" |
|
/> |
|
</div> |
|
</div> |
|
{/if} |
|
|