<!-- SPDX-FileCopyrightText: syuilo and other misskey contributors SPDX-License-Identifier: AGPL-3.0-only --> <template> <div v-show="props.modelValue.length != 0" :class="$style.root"> <Sortable :modelValue="props.modelValue" :class="$style.files" itemKey="id" :animation="150" :delay="100" :delayOnTouchOnly="true" @update:modelValue="v => emit('update:modelValue', v)"> <template #item="{element}"> <div :class="$style.file" @click="showFileMenu(element, $event)" @contextmenu.prevent="showFileMenu(element, $event)"> <MkDriveFileThumbnail :data-id="element.id" :class="$style.thumbnail" :file="element" fit="cover"/> <div v-if="element.isSensitive" :class="$style.sensitive"> <i class="ti ti-eye-exclamation" style="margin: auto;"></i> </div> </div> </template> </Sortable> <p :class="$style.remain">{{ 16 - props.modelValue.length }}/16</p> </div> </template> <script lang="ts" setup> import { defineAsyncComponent } from 'vue'; import * as Misskey from 'misskey-js'; import MkDriveFileThumbnail from '@/components/MkDriveFileThumbnail.vue'; import * as os from '@/os.js'; import { i18n } from '@/i18n.js'; const Sortable = defineAsyncComponent(() => import('vuedraggable').then(x => x.default)); const props = defineProps<{ modelValue: any[]; detachMediaFn?: (id: string) => void; }>(); const emit = defineEmits<{ (ev: 'update:modelValue', value: any[]): void; (ev: 'detach', id: string): void; (ev: 'changeSensitive', file: Misskey.entities.DriveFile, isSensitive: boolean): void; (ev: 'changeName', file: Misskey.entities.DriveFile, newName: string): void; (ev: 'replaceFile', file: Misskey.entities.DriveFile, newFile: Misskey.entities.DriveFile): void; }>(); let menuShowing = false; function detachMedia(id: string) { if (props.detachMediaFn) { props.detachMediaFn(id); } else { emit('detach', id); } } function toggleSensitive(file) { os.api('drive/files/update', { fileId: file.id, isSensitive: !file.isSensitive, }).then(() => { emit('changeSensitive', file, !file.isSensitive); }); } async function rename(file) { const { canceled, result } = await os.inputText({ title: i18n.ts.enterFileName, default: file.name, allowEmpty: false, }); if (canceled) return; os.api('drive/files/update', { fileId: file.id, name: result, }).then(() => { emit('changeName', file, result); file.name = result; }); } async function describe(file) { os.popup(defineAsyncComponent(() => import('@/components/MkFileCaptionEditWindow.vue')), { default: file.comment !== null ? file.comment : '', file: file, }, { done: caption => { let comment = caption.length === 0 ? null : caption; os.api('drive/files/update', { fileId: file.id, comment: comment, }).then(() => { file.comment = comment; }); }, }, 'closed'); } async function crop(file: Misskey.entities.DriveFile): Promise<void> { const newFile = await os.cropImage(file, { aspectRatio: NaN }); emit('replaceFile', file, newFile); } function showFileMenu(file: Misskey.entities.DriveFile, ev: MouseEvent): void { if (menuShowing) return; const isImage = file.type.startsWith('image/'); os.popupMenu([{ text: i18n.ts.renameFile, icon: 'ti ti-forms', action: () => { rename(file); }, }, { text: file.isSensitive ? i18n.ts.unmarkAsSensitive : i18n.ts.markAsSensitive, icon: file.isSensitive ? 'ti ti-eye-exclamation' : 'ti ti-eye', action: () => { toggleSensitive(file); }, }, { text: i18n.ts.describeFile, icon: 'ti ti-text-caption', action: () => { describe(file); }, }, ...isImage ? [{ text: i18n.ts.cropImage, icon: 'ti ti-crop', action: () : void => { crop(file); }, }] : [], { text: i18n.ts.attachCancel, icon: 'ti ti-circle-x', action: () => { detachMedia(file.id); }, }], ev.currentTarget ?? ev.target).then(() => menuShowing = false); menuShowing = true; } </script> <style lang="scss" module> .root { padding: 8px 16px; position: relative; } .files { display: flex; flex-wrap: wrap; } .file { position: relative; width: 64px; height: 64px; margin-right: 4px; border-radius: 4px; overflow: hidden; cursor: move; } .thumbnail { width: 100%; height: 100%; z-index: 1; color: var(--fg); } .sensitive { display: flex; position: absolute; width: 64px; height: 64px; top: 0; left: 0; z-index: 2; background: rgba(17, 17, 17, .7); color: #fff; } .remain { display: block; position: absolute; top: 8px; right: 8px; margin: 0; padding: 0; font-size: 90%; } </style>