Files
nocodb/packages/nc-gui/components/cell/attachment/Modal.vue
2026-01-13 14:46:10 +00:00

222 lines
6.4 KiB
Vue

<script lang="ts" setup>
import { onKeyDown, useEventListener } from '@vueuse/core'
import { useAttachmentCell } from './utils'
import { useSortable } from './sort'
const { isUIAllowed } = useRoles()
const {
isLoading,
isPublic,
isReadonly: readOnly,
visibleItems,
modalVisible,
column,
onDrop,
updateModelValue,
selectedFile,
selectedVisibleItems,
isSharedForm,
bulkDownloadAttachments,
} = useAttachmentCell()!
const { onChange: onFileDialogChange, open: _open } = useFileDialog({
reset: true,
})
const dropZoneRef = ref<HTMLDivElement>()
const canvasSelectCell = inject(CanvasSelectCellInj, null)
const sortableRef = ref<HTMLDivElement>()
const { dragging } = useSortable(sortableRef, visibleItems, updateModelValue, readOnly)
const onDropAction = function (...args: any[]) {
const draggingBool = unref(dragging)
if (!draggingBool) {
onDrop.apply(this, args)
}
}
const { isOverDropZone } = useDropZone(dropZoneRef, onDropAction)
const preventExit = ref(false)
onKeyDown('Escape', () => {
modalVisible.value = false
isOverDropZone.value = false
})
watch(modalVisible, (newVal, oldVal) => {
if (preventExit.value) return
setTimeout(() => {
preventExit.value = false
}, 500)
if (oldVal && !newVal) canvasSelectCell?.trigger()
})
function onClick(item: Record<string, any>) {
selectedFile.value = item
preventExit.value = true
modalVisible.value = false
const stopHandle = watch(selectedFile, (nextImage) => {
if (!nextImage) {
setTimeout(() => {
modalVisible.value = true
}, 50)
stopHandle?.()
}
})
}
// when user paste on modal
useEventListener(dropZoneRef, 'paste', (event: ClipboardEvent) => {
if (event.clipboardData?.files) {
onDrop(event.clipboardData.files, {} as any)
}
})
const onFileDialogOpen = (_event) => {
_open()
}
onFileDialogChange((files) => {
onDrop(files, {} as any)
})
const isNewAttachmentModalOpen = ref(false)
</script>
<template>
<NcModal
v-model:visible="modalVisible"
wrap-class-name="nc-modal-attachment-expand-cell"
class="nc-attachment-modal"
:class="{ active: modalVisible }"
width="80%"
>
<div class="flex justify-between pb-6 gap-4 items-center">
<div class="font-semibold text-xl">{{ column?.title }}</div>
<div class="flex items-center gap-2">
<NcButton
:disabled="!selectedVisibleItems.some((v) => !!v)"
type="secondary"
size="small"
@click="bulkDownloadAttachments"
>
<div class="flex gap-2 items-center">
<GeneralIcon icon="download" />
{{ $t('activity.bulkDownload') }}
</div>
</NcButton>
<NcButton
v-if="isSharedForm || (!readOnly && isUIAllowed('dataEdit') && !isPublic)"
class="nc-attach-file group"
size="small"
data-testid="attachment-expand-file-picker-button"
@click="isNewAttachmentModalOpen = true"
>
<div class="flex gap-2 items-center">
<component :is="iconMap.cellAttachment" class="w-4 h-4" />
{{ $t('activity.attachFile') }}
</div>
</NcButton>
<NcButton type="secondary" size="small" @click="modalVisible = false">
<GeneralIcon icon="close" />
</NcButton>
</div>
</div>
<div ref="dropZoneRef" tabindex="0" class="relative min-h-[96px]">
<div
v-if="isSharedForm || (!readOnly && !dragging && isOverDropZone)"
class="text-white absolute inset-0 flex flex-col items-center justify-center border-dashed border-2 border-nc-border-gray-medium rounded-lg pt-2"
:class="{
'border-nc-border-brand': !visibleItems.length,
}"
>
<component :is="iconMap.upload" class="w-8 h-8 text-nc-content-brand" />
<div class="p-4">
<h1 class="text-nc-content-brand font-bold">{{ $t('labels.dropHere') }}</h1>
</div>
</div>
<template v-if="visibleItems.length > 0">
<div
ref="sortableRef"
class="grid max-h-140 overflow-auto nc-scrollbar-md md:grid-cols-3 xl:grid-cols-5 gap-y-8 gap-x-4 relative"
>
<CellAttachmentCard
v-for="(item, i) in visibleItems"
:key="`${item?.title}-${i}`"
v-model:dragging="dragging"
v-model:selected="selectedVisibleItems[i]"
:attachment="item"
:index="i"
:allow-selection="true"
:allow-rename="!isSharedForm || (!readOnly && isUIAllowed('dataEdit') && !isPublic)"
:allow-delete="!readOnly"
:icon-height="72"
:icon-width="72"
@clicked="onClick(item)"
/>
<div v-if="isLoading" class="flex flex-col gap-1">
<a-card class="nc-attachment-item group">
<div class="nc-attachment h-full w-full flex items-center justify-center">
<a-skeleton-image class />
</div>
</a-card>
</div>
</div>
</template>
<template v-else>
<div
class="h-[30vh] min-h-[96px] border-dashed border-2 border-nc-border-medium rounded-lg justify-center cursor-pointer flex items-center flex-col"
@click="onFileDialogOpen"
>
<template v-if="!(isSharedForm || (!readOnly && !dragging && isOverDropZone))">
<component :is="iconMap.upload" class="w-8 h-8 text-nc-content-gray-muted" />
<span class="p-4">
{{ $t('labels.clickTo') }}
<span class="font-semibold text-nc-content-brand"> {{ $t('labels.browseFiles') }} </span>
{{ $t('general.or') }}
<span class="font-semibold"> {{ $t('labels.dragFilesHere') }} </span>
{{ $t('labels.toUpload') }}
</span>
</template>
</div>
</template>
<CellAttachmentAttachFile v-if="isNewAttachmentModalOpen" v-model:value="isNewAttachmentModalOpen" />
</div>
</NcModal>
</template>
<style lang="scss">
.hide-ui {
@apply md:(h-0 w-0 overflow-x-hidden whitespace-nowrap);
.group:hover & {
@apply h-auto w-auto overflow-visible whitespace-normal;
}
}
.nc-attachment-modal {
.nc-attachment-item {
@apply h-[200px] max-h-[200px] flex relative overflow-hidden;
}
.dragging {
.nc-attachment-item {
@apply !pointer-events-none;
}
}
.nc-checkbox > .ant-checkbox {
box-shadow: none !important;
}
}
</style>