forked from coracle/flotilla
87 lines
2.1 KiB
Svelte
87 lines
2.1 KiB
Svelte
<script lang="ts">
|
|
import {randomId} from '@welshman/lib'
|
|
import Icon from '@lib/components/Icon.svelte'
|
|
|
|
export let file: File | null = null
|
|
export let url: string | null = null
|
|
|
|
const id = randomId()
|
|
|
|
const onDragEnter = () => {
|
|
active = true
|
|
}
|
|
|
|
const onDragOver = () => {
|
|
active = true
|
|
}
|
|
|
|
const onDragLeave = () => {
|
|
active = false
|
|
}
|
|
|
|
const onDrop = (e: any) => {
|
|
active = false
|
|
|
|
file = e.dataTransfer.files[0]
|
|
}
|
|
|
|
const onChange = (e: any) => {
|
|
file = e.target.files[0]
|
|
}
|
|
|
|
const onClear = () => {
|
|
initialUrl = null
|
|
file = null
|
|
url = null
|
|
}
|
|
|
|
let active = false
|
|
let initialUrl = url
|
|
|
|
$: {
|
|
if (file) {
|
|
const reader = new FileReader()
|
|
|
|
reader.addEventListener("load", () => { url = reader.result as string }, false)
|
|
reader.readAsDataURL(file)
|
|
} else {
|
|
url = initialUrl
|
|
}
|
|
}
|
|
|
|
</script>
|
|
|
|
<form>
|
|
<input {id} type="file" accept="image/*" on:change={onChange} class="hidden" />
|
|
<label
|
|
for={id}
|
|
aria-label="Drag and drop files here."
|
|
style="background-image: url({url});"
|
|
class="relative flex justify-center items-center w-24 h-24 rounded-full border-2 border-dashed border-base-content transition-all bg-base-300 cursor-pointer bg-cover bg-center shrink-0"
|
|
class:border-primary={active}
|
|
on:dragenter|preventDefault|stopPropagation={onDragEnter}
|
|
on:dragover|preventDefault|stopPropagation={onDragOver}
|
|
on:dragleave|preventDefault|stopPropagation={onDragLeave}
|
|
on:drop|preventDefault|stopPropagation={onDrop}>
|
|
<div
|
|
class="bg-primary right-0 top-0 absolute rounded-full overflow-hidden"
|
|
class:bg-error={file}
|
|
class:bg-primary={!file}>
|
|
{#if file}
|
|
<span
|
|
role="button"
|
|
tabindex="-1"
|
|
on:mousedown|stopPropagation={onClear}
|
|
on:touchstart|stopPropagation={onClear}>
|
|
<Icon icon="close-circle" class="!bg-base-300 scale-150" />
|
|
</span>
|
|
{:else}
|
|
<Icon icon="add-circle" class="!bg-base-300 scale-150" />
|
|
{/if}
|
|
</div>
|
|
{#if !file}
|
|
<Icon icon="gallery-send" size={7} />
|
|
{/if}
|
|
</label>
|
|
</form>
|