| | <script lang="ts"> |
| | import { getContext, createEventDispatcher, onMount, onDestroy } from 'svelte'; |
| | |
| | const i18n = getContext('i18n'); |
| | const dispatch = createEventDispatcher(); |
| | |
| | import ChevronDown from '../icons/ChevronDown.svelte'; |
| | import ChevronRight from '../icons/ChevronRight.svelte'; |
| | import Collapsible from './Collapsible.svelte'; |
| | import Tooltip from './Tooltip.svelte'; |
| | import Plus from '../icons/Plus.svelte'; |
| | |
| | export let open = true; |
| | |
| | export let id = ''; |
| | export let name = ''; |
| | export let collapsible = true; |
| | |
| | export let onAddLabel: string = ''; |
| | export let onAdd: null | Function = null; |
| | |
| | export let dragAndDrop = true; |
| | |
| | export let className = ''; |
| | |
| | let folderElement; |
| | |
| | let draggedOver = false; |
| | |
| | const onDragOver = (e) => { |
| | e.preventDefault(); |
| | e.stopPropagation(); |
| | draggedOver = true; |
| | }; |
| | |
| | const onDrop = (e) => { |
| | e.preventDefault(); |
| | e.stopPropagation(); |
| | |
| | if (folderElement.contains(e.target)) { |
| | console.log('Dropped on the Button'); |
| | |
| | if (e.dataTransfer.items && e.dataTransfer.items.length > 0) { |
| | |
| | for (const item of Array.from(e.dataTransfer.items)) { |
| | |
| | if (item.kind === 'file') { |
| | const file = item.getAsFile(); |
| | if (file && file.type === 'application/json') { |
| | console.log('Dropped file is a JSON file!'); |
| | |
| | |
| | const reader = new FileReader(); |
| | reader.onload = async function (event) { |
| | try { |
| | const fileContent = JSON.parse(event.target.result); |
| | console.log('Parsed JSON Content: ', fileContent); |
| | open = true; |
| | dispatch('import', fileContent); |
| | } catch (error) { |
| | console.error('Error parsing JSON file:', error); |
| | } |
| | }; |
| | |
| | |
| | reader.readAsText(file); |
| | } else { |
| | console.error('Only JSON file types are supported.'); |
| | } |
| | } else { |
| | open = true; |
| | |
| | const dataTransfer = e.dataTransfer.getData('text/plain'); |
| | const data = JSON.parse(dataTransfer); |
| | |
| | console.log(data); |
| | dispatch('drop', data); |
| | } |
| | } |
| | } |
| | |
| | draggedOver = false; |
| | } |
| | }; |
| | |
| | const onDragLeave = (e) => { |
| | e.preventDefault(); |
| | e.stopPropagation(); |
| | |
| | draggedOver = false; |
| | }; |
| | |
| | onMount(() => { |
| | if (!dragAndDrop) { |
| | return; |
| | } |
| | folderElement.addEventListener('dragover', onDragOver); |
| | folderElement.addEventListener('drop', onDrop); |
| | folderElement.addEventListener('dragleave', onDragLeave); |
| | }); |
| | |
| | onDestroy(() => { |
| | if (!dragAndDrop) { |
| | return; |
| | } |
| | folderElement.addEventListener('dragover', onDragOver); |
| | folderElement.removeEventListener('drop', onDrop); |
| | folderElement.removeEventListener('dragleave', onDragLeave); |
| | }); |
| | </script> |
| |
|
| | <div bind:this={folderElement} class="relative {className}"> |
| | {#if draggedOver} |
| | <div |
| | class="absolute top-0 left-0 w-full h-full rounded-xs bg-gray-100/50 dark:bg-gray-700/20 bg-opacity-50 dark:bg-opacity-10 z-50 pointer-events-none touch-none" |
| | ></div> |
| | {/if} |
| |
|
| | {#if collapsible} |
| | <Collapsible |
| | bind:open |
| | className="w-full " |
| | buttonClassName="w-full" |
| | on:change={(e) => { |
| | dispatch('change', e.detail); |
| | }} |
| | > |
| | |
| | <div |
| | class="w-full group rounded-md relative flex items-center justify-between hover:bg-gray-100 dark:hover:bg-gray-900 text-gray-500 dark:text-gray-500 transition" |
| | > |
| | <button class="w-full py-1.5 pl-2 flex items-center gap-1.5 text-xs font-medium"> |
| | <div class="text-gray-300 dark:text-gray-600"> |
| | {#if open} |
| | <ChevronDown className=" size-3" strokeWidth="2.5" /> |
| | {:else} |
| | <ChevronRight className=" size-3" strokeWidth="2.5" /> |
| | {/if} |
| | </div> |
| |
|
| | <div class="translate-y-[0.5px]"> |
| | {name} |
| | </div> |
| | </button> |
| |
|
| | {#if onAdd} |
| | <button |
| | class="absolute z-10 right-2 invisible group-hover:visible self-center flex items-center dark:text-gray-300" |
| | on:pointerup={(e) => { |
| | e.stopPropagation(); |
| | }} |
| | on:click={(e) => { |
| | e.stopPropagation(); |
| | onAdd(); |
| | }} |
| | > |
| | <Tooltip content={onAddLabel}> |
| | <button |
| | class="p-0.5 dark:hover:bg-gray-850 rounded-lg touch-auto" |
| | on:click={(e) => {}} |
| | > |
| | <Plus className=" size-3" strokeWidth="2.5" /> |
| | </button> |
| | </Tooltip> |
| | </button> |
| | {/if} |
| | </div> |
| |
|
| | <div slot="content" class="w-full"> |
| | <slot></slot> |
| | </div> |
| | </Collapsible> |
| | {:else} |
| | <slot></slot> |
| | {/if} |
| | </div> |
| |
|