to keep things manageable i merged a lot of one off values into just a handful of common sizes, so some parts of the ui will look different than upstream even with the "Misskey" rounding mode
179 lines
4.5 KiB
Vue
179 lines
4.5 KiB
Vue
<!--
|
|
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="ph-eye-closed ph-bold ph-lg" 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: 'ph-textbox ph-bold ph-lg',
|
|
action: () => { rename(file); },
|
|
}, {
|
|
text: file.isSensitive ? i18n.ts.unmarkAsSensitive : i18n.ts.markAsSensitive,
|
|
icon: file.isSensitive ? 'ph-eye-closed ph-bold ph-lg' : 'ph-eye ph-bold ph-lg',
|
|
action: () => { toggleSensitive(file); },
|
|
}, {
|
|
text: i18n.ts.describeFile,
|
|
icon: 'ph-text-indent ph-bold ph-lg',
|
|
action: () => { describe(file); },
|
|
}, ...isImage ? [{
|
|
text: i18n.ts.cropImage,
|
|
icon: 'ph-crop ph-bold ph-lg',
|
|
action: () : void => { crop(file); },
|
|
}] : [], {
|
|
text: i18n.ts.attachCancel,
|
|
icon: 'ph-x-circle ph-bold ph-lg',
|
|
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: var(--radius-xs);
|
|
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>
|