| <script> |
| 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'; |
| |
| export let open = true; |
| |
| export let id = ''; |
| export let name = ''; |
| export let collapsible = 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(() => { |
| folderElement.addEventListener('dragover', onDragOver); |
| folderElement.addEventListener('drop', onDrop); |
| folderElement.addEventListener('dragleave', onDragLeave); |
| }); |
| |
| onDestroy(() => { |
| 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-sm bg-[hsla(260,85%,65%,0.1)] 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"> |
| <button |
| class="w-full py-1.5 px-2 rounded-md flex items-center gap-1.5 text-xs text-gray-500 dark:text-gray-500 font-medium hover:bg-gray-100 dark:hover:bg-gray-900 transition" |
| > |
| <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> |
| </div> |
|
|
| <div slot="content" class="w-full"> |
| <slot></slot> |
| </div> |
| </Collapsible> |
| {:else} |
| <slot></slot> |
| {/if} |
| </div> |
|
|