Files
gpx.studio/website/src/lib/components/FileList.svelte

130 lines
3.2 KiB
Svelte
Raw Normal View History

2024-04-18 10:52:45 +02:00
<script lang="ts">
2024-04-24 20:13:42 +02:00
import { fileOrder, files, selectedFiles, selectFiles } from '$lib/stores';
2024-04-18 10:52:45 +02:00
2024-04-20 18:47:16 +02:00
import { ScrollArea } from '$lib/components/ui/scroll-area/index';
import Sortable from 'sortablejs/Sortable';
import type { GPXFile } from 'gpx';
2024-04-19 17:06:36 +02:00
2024-04-22 11:45:57 +02:00
import { onMount } from 'svelte';
2024-04-24 20:13:42 +02:00
import { get, type Writable } from 'svelte/store';
2024-04-22 11:45:57 +02:00
2024-04-19 17:06:36 +02:00
let tabs: HTMLDivElement;
2024-04-20 18:47:16 +02:00
let buttons: HTMLButtonElement[] = [];
let sortable: Sortable;
2024-04-24 20:13:42 +02:00
function selectFile(file: Writable<GPXFile>) {
2024-04-20 18:47:16 +02:00
selectedFiles.update((selectedFiles) => {
selectedFiles.clear();
selectedFiles.add(file);
return selectedFiles;
});
}
2024-04-19 17:06:36 +02:00
2024-04-24 20:13:42 +02:00
function addSelectFile(file: Writable<GPXFile>) {
2024-04-20 18:47:16 +02:00
selectedFiles.update((selectedFiles) => {
selectedFiles.add(file);
return selectedFiles;
});
}
2024-04-20 15:38:17 +02:00
2024-04-22 11:45:57 +02:00
function selectAllFiles() {
selectedFiles.update((selectedFiles) => {
2024-04-24 20:13:42 +02:00
get(files).forEach((file) => {
2024-04-22 11:45:57 +02:00
selectedFiles.add(file);
});
return selectedFiles;
});
}
2024-04-24 20:13:42 +02:00
function deselectFile(file: Writable<GPXFile>) {
2024-04-20 18:47:16 +02:00
selectedFiles.update((selectedFiles) => {
selectedFiles.delete(file);
return selectedFiles;
});
}
2024-04-20 15:38:17 +02:00
2024-04-20 18:47:16 +02:00
onMount(() => {
sortable = Sortable.create(tabs, {
2024-04-20 15:38:17 +02:00
forceAutoScrollFallback: true,
multiDrag: true,
2024-04-20 18:47:16 +02:00
multiDragKey: 'shift',
selectedClass: 'sortable-selected',
avoidImplicitDeselect: true,
onSelect: (e) => {
const index = parseInt(e.item.getAttribute('data-id'));
2024-04-24 20:13:42 +02:00
addSelectFile($files[index]);
if (!e.originalEvent.shiftKey && $selectedFiles.size > 1) {
$selectedFiles.forEach((file) => {
2024-04-24 20:13:42 +02:00
if (file !== $files[index]) {
deselectFile(file);
2024-04-24 20:13:42 +02:00
const index = $files.indexOf(file);
Sortable.utils.deselect(buttons[index]);
}
});
}
2024-04-20 18:47:16 +02:00
},
onDeselect: (e) => {
const index = parseInt(e.item.getAttribute('data-id'));
2024-04-24 20:13:42 +02:00
deselectFile($files[index]);
},
onSort: () => {
2024-04-24 20:13:42 +02:00
$fileOrder = sortable.toArray().map((index: string) => $files[parseInt(index)]);
2024-04-20 18:47:16 +02:00
}
2024-04-19 17:06:36 +02:00
});
});
2024-04-20 18:47:16 +02:00
selectFiles.update(() => {
return {
2024-04-24 20:13:42 +02:00
select: (file: Writable<GPXFile>) => {
2024-04-20 18:47:16 +02:00
buttons.forEach((button) => {
if (button) {
Sortable.utils.deselect(button);
}
});
2024-04-24 20:13:42 +02:00
const index = $files.indexOf(file);
2024-04-20 18:47:16 +02:00
Sortable.utils.select(buttons[index]);
selectFile(file);
},
2024-04-24 20:13:42 +02:00
addSelect: (file: Writable<GPXFile>) => {
const index = $files.indexOf(file);
2024-04-20 18:47:16 +02:00
Sortable.utils.select(buttons[index]);
addSelectFile(file);
},
2024-04-22 11:45:57 +02:00
selectAllFiles: () => {
2024-04-24 20:13:42 +02:00
$files.forEach((file, index) => {
2024-04-22 11:45:57 +02:00
Sortable.utils.select(buttons[index]);
});
selectAllFiles();
},
2024-04-24 20:13:42 +02:00
removeSelect: (file: Writable<GPXFile>) => {
const index = $files.indexOf(file);
2024-04-20 18:47:16 +02:00
Sortable.utils.deselect(buttons[index]);
deselectFile(file);
}
};
});
2024-04-18 10:52:45 +02:00
</script>
2024-04-24 12:55:53 +02:00
<div class="h-10 -translate-y-10 w-full">
2024-04-19 16:13:08 +02:00
<ScrollArea orientation="horizontal" class="w-full h-full" scrollbarXClasses="h-2">
2024-04-19 17:06:36 +02:00
<div bind:this={tabs} class="flex flex-row gap-1">
2024-04-24 20:13:42 +02:00
{#each $files as file, index}
2024-04-19 16:13:08 +02:00
<button
2024-04-20 18:47:16 +02:00
bind:this={buttons[index]}
data-id={index}
class="my-1 px-1.5 py-1 rounded bg-secondary hover:bg-gray-200 shadow-none first:ml-1 last:mr-1"
2024-04-18 15:30:19 +02:00
>
2024-04-24 20:13:42 +02:00
{get(file).metadata.name}
2024-04-19 16:13:08 +02:00
</button>
2024-04-18 15:30:19 +02:00
{/each}
</div>
</ScrollArea>
</div>
2024-04-20 18:47:16 +02:00
<style lang="postcss">
div :global(.sortable-selected) {
@apply bg-background shadow;
}
</style>