169 lines
4.4 KiB
Vue
169 lines
4.4 KiB
Vue
<script setup lang="ts">
|
|
import {
|
|
ContextMenu,
|
|
ContextMenuTrigger,
|
|
ContextMenuContent,
|
|
ContextMenuItem,
|
|
ContextMenuSeparator,
|
|
} from '@/components/ui/context-menu';
|
|
import {
|
|
deleteWarrenDirectory,
|
|
deleteWarrenFile,
|
|
fetchFile,
|
|
moveFile,
|
|
} from '~/lib/api/warrens';
|
|
import type { DirectoryEntry } from '#shared/types';
|
|
|
|
const warrenStore = useWarrenStore();
|
|
const renameDialog = useRenameDirectoryDialog();
|
|
|
|
const { entry, disabled } = defineProps<{
|
|
entry: DirectoryEntry;
|
|
disabled: boolean;
|
|
}>();
|
|
|
|
const deleting = ref(false);
|
|
|
|
async function submitDelete(force: boolean = false) {
|
|
if (warrenStore.current == null) {
|
|
return;
|
|
}
|
|
|
|
deleting.value = true;
|
|
|
|
if (entry.fileType === 'directory') {
|
|
await deleteWarrenDirectory(
|
|
warrenStore.current.warrenId,
|
|
warrenStore.current.path,
|
|
entry.name,
|
|
force
|
|
);
|
|
} else {
|
|
await deleteWarrenFile(
|
|
warrenStore.current.warrenId,
|
|
warrenStore.current.path,
|
|
entry.name
|
|
);
|
|
}
|
|
|
|
deleting.value = false;
|
|
}
|
|
|
|
async function openRenameDialog() {
|
|
renameDialog.openDialog(entry);
|
|
}
|
|
|
|
async function onClick() {
|
|
if (warrenStore.loading || warrenStore.current == null) {
|
|
return;
|
|
}
|
|
|
|
if (entry.fileType === 'directory') {
|
|
warrenStore.addToCurrentWarrenPath(entry.name);
|
|
return;
|
|
}
|
|
|
|
if (entry.mimeType == null) {
|
|
return;
|
|
}
|
|
|
|
if (entry.mimeType.startsWith('image/')) {
|
|
const result = await fetchFile(
|
|
warrenStore.current.warrenId,
|
|
warrenStore.current.path,
|
|
entry.name
|
|
);
|
|
if (result.success) {
|
|
const url = URL.createObjectURL(result.data);
|
|
warrenStore.imageViewer.src = url;
|
|
}
|
|
}
|
|
}
|
|
|
|
function onDragStart(e: DragEvent) {
|
|
if (e.dataTransfer == null) {
|
|
return;
|
|
}
|
|
|
|
e.dataTransfer.setData('application/warren', entry.name);
|
|
e.dataTransfer.dropEffect = 'move';
|
|
}
|
|
|
|
async function onDrop(e: DragEvent) {
|
|
if (e.dataTransfer == null || warrenStore.current == null) {
|
|
return;
|
|
}
|
|
|
|
if (entry.fileType !== 'directory') {
|
|
return;
|
|
}
|
|
|
|
const fileName = e.dataTransfer.getData('application/warren');
|
|
|
|
if (entry.name === fileName) {
|
|
return;
|
|
}
|
|
|
|
await moveFile(
|
|
warrenStore.current.warrenId,
|
|
warrenStore.current.path,
|
|
fileName,
|
|
`${warrenStore.current.path}/${entry.name}`
|
|
);
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<ContextMenu>
|
|
<ContextMenuTrigger>
|
|
<button
|
|
:disabled="warrenStore.loading || disabled"
|
|
class="bg-accent/30 border-border select-none, flex w-52 translate-0 flex-row gap-4 overflow-hidden rounded-md border-1 px-4 py-2"
|
|
draggable="true"
|
|
@dragstart="onDragStart"
|
|
@drop="onDrop"
|
|
@click="onClick"
|
|
>
|
|
<div class="flex flex-row items-center">
|
|
<Icon class="size-6" :name="getFileIcon(entry.mimeType)" />
|
|
</div>
|
|
|
|
<div
|
|
class="flex w-full flex-col items-start justify-stretch gap-0 overflow-hidden text-left leading-6"
|
|
>
|
|
<span class="w-full truncate">{{ entry.name }}</span>
|
|
<NuxtTime
|
|
v-if="entry.createdAt != null"
|
|
:datetime="entry.createdAt * 1000"
|
|
class="text-muted-foreground w-full truncate text-sm"
|
|
relative
|
|
></NuxtTime>
|
|
</div>
|
|
</button>
|
|
</ContextMenuTrigger>
|
|
<ContextMenuContent>
|
|
<ContextMenuItem @select="openRenameDialog">
|
|
<Icon name="lucide:pencil" />
|
|
Rename
|
|
</ContextMenuItem>
|
|
|
|
<ContextMenuSeparator />
|
|
|
|
<ContextMenuItem @select="() => submitDelete(false)">
|
|
<Icon name="lucide:trash-2" />
|
|
Delete
|
|
</ContextMenuItem>
|
|
<ContextMenuItem
|
|
v-if="entry.fileType === 'directory'"
|
|
@select="() => submitDelete(true)"
|
|
>
|
|
<Icon
|
|
class="text-destructive-foreground"
|
|
name="lucide:trash-2"
|
|
/>
|
|
Force delete
|
|
</ContextMenuItem>
|
|
</ContextMenuContent>
|
|
</ContextMenu>
|
|
</template>
|