import { defineAsyncComponent, Ref, inject } from "vue"; import * as misskey from "firefish-js"; import { $i } from "@/account"; import { i18n } from "@/i18n"; import { instance } from "@/instance"; import * as os from "@/os"; import copyToClipboard from "@/scripts/copy-to-clipboard"; import { url } from "@/config"; import { noteActions } from "@/store"; import { shareAvailable } from "@/scripts/share-available"; import { getUserMenu } from "@/scripts/get-user-menu"; export function getNoteMenu(props: { note: misskey.entities.Note; menuButton: Ref; translation: Ref; translating: Ref; isDeleted: Ref; currentClipPage?: Ref; }) { const isRenote = props.note.renote != null && props.note.text == null && props.note.fileIds.length === 0 && props.note.poll == null; const appearNote = isRenote ? (props.note.renote as misskey.entities.Note) : props.note; function del(): void { os.confirm({ type: "warning", text: i18n.ts.noteDeleteConfirm, }).then(({ canceled }) => { if (canceled) return; os.api("notes/delete", { noteId: appearNote.id, }); }); } function delEdit(): void { os.confirm({ type: "warning", text: i18n.ts.deleteAndEditConfirm, }).then(({ canceled }) => { if (canceled) return; os.api("notes/delete", { noteId: appearNote.id, }); os.post({ initialNote: appearNote, renote: appearNote.renote, reply: appearNote.reply, channel: appearNote.channel, }); }); } function edit(): void { os.post({ initialNote: appearNote, renote: appearNote.renote, reply: appearNote.reply, channel: appearNote.channel, editId: appearNote.id, }); } function toggleFavorite(favorite: boolean): void { os.apiWithDialog( favorite ? "notes/favorites/create" : "notes/favorites/delete", { noteId: appearNote.id, }, ); } function toggleWatch(watch: boolean): void { os.apiWithDialog( watch ? "notes/watching/create" : "notes/watching/delete", { noteId: appearNote.id, }, ); } function toggleThreadMute(mute: boolean): void { os.apiWithDialog( mute ? "notes/thread-muting/create" : "notes/thread-muting/delete", { noteId: appearNote.id, }, ); } function copyContent(): void { copyToClipboard(appearNote.text); os.success(); } function copyLink(): void { copyToClipboard(`${url}/notes/${appearNote.id}`); os.success(); } function copyOriginal(): void { copyToClipboard(appearNote.url ?? appearNote.uri); os.success(); } function togglePin(pin: boolean): void { os.apiWithDialog( pin ? "i/pin" : "i/unpin", { noteId: appearNote.id, }, undefined, ).catch((res) => { if (res.id === "72dab508-c64d-498f-8740-a8eec1ba385a") { os.alert({ type: "error", text: i18n.ts.pinLimitExceeded, }); } }); } async function clip(): Promise { const clips = await os.api("clips/list"); os.popupMenu( [ { icon: "ph-plus ph-bold ph-lg", text: i18n.ts.createNew, action: async () => { const { canceled, result } = await os.form(i18n.ts.createNewClip, { name: { type: "string", label: i18n.ts.name, }, description: { type: "string", required: false, multiline: true, label: i18n.ts.description, }, isPublic: { type: "boolean", label: i18n.ts.public, default: false, }, }); if (canceled) return; const clip = await os.apiWithDialog("clips/create", result); os.apiWithDialog("clips/add-note", { clipId: clip.id, noteId: appearNote.id, }); }, }, null, ...clips.map((clip) => ({ text: clip.name, action: () => { os.promiseDialog( os.api("clips/add-note", { clipId: clip.id, noteId: appearNote.id, }), null, async (err) => { if (err.id === "734806c4-542c-463a-9311-15c512803965") { const confirm = await os.confirm({ type: "warning", text: i18n.t("confirmToUnclipAlreadyClippedNote", { name: clip.name, }), }); if (!confirm.canceled) { os.apiWithDialog("clips/remove-note", { clipId: clip.id, noteId: appearNote.id, }); if (props.currentClipPage?.value.id === clip.id) props.isDeleted.value = true; } } else { os.alert({ type: "error", text: err.message + "\n" + err.id, }); } }, ); }, })), ], props.menuButton.value, {}, ).then(focus); } async function unclip(): Promise { os.apiWithDialog("clips/remove-note", { clipId: props.currentClipPage.value.id, noteId: appearNote.id, }); props.isDeleted.value = true; } async function promote(): Promise { const { canceled, result: days } = await os.inputNumber({ title: i18n.ts.numberOfDays, }); if (canceled) return; os.apiWithDialog("admin/promo/create", { noteId: appearNote.id, expiresAt: Date.now() + 86400000 * days, }); } function share(): void { navigator.share({ title: i18n.t("noteOf", { user: appearNote.user.name }), text: appearNote.text, url: `${url}/notes/${appearNote.id}`, }); } const localTranslateLang = localStorage.getItem("translateLang"); const localLang = localStorage.getItem("lang"); const translate_ = async (noteId: number, targetLang: string) => { return await os.api("notes/translate", { noteId: noteId, targetLang: targetLang, }); }; const translate = async () => { if (props.translation.value != null) return; props.translating.value = true; props.translation.value = await translate_( appearNote.id, localTranslateLang || localLang || navigator.language, ); // use UI language as the second translation target if ( localTranslateLang != null && localLang != null && localTranslateLang !== localLang && (!props.translation.value || props.translation.value.sourceLang.toLowerCase() === localTranslateLang.slice(0, 2)) ) props.translation.value = await translate_(appearNote.id, localLang); props.translating.value = false; }; let menu; if ($i) { const statePromise = os.api("notes/state", { noteId: appearNote.id, }); const isAppearAuthor = appearNote.userId === $i.id; const isModerator = $i.isAdmin || $i.isModerator; menu = [ ...(props.currentClipPage?.value.userId === $i.id ? [ { icon: "ph-minus-circle ph-bold ph-lg", text: i18n.ts.unclip, danger: true, action: unclip, }, null, ] : []), statePromise.then((state) => state?.isFavorited ? { icon: "ph-bookmark-simple ph-bold ph-lg", text: i18n.ts.unfavorite, action: () => toggleFavorite(false), } : { icon: "ph-bookmark-simple ph-bold ph-lg", text: i18n.ts.favorite, action: () => toggleFavorite(true), }, ), { icon: "ph-paperclip ph-bold ph-lg", text: i18n.ts.clip, action: () => clip(), }, !isAppearAuthor ? statePromise.then((state) => state.isWatching ? { icon: "ph-eye-slash ph-bold ph-lg", text: i18n.ts.unwatch, action: () => toggleWatch(false), } : { icon: "ph-eye ph-bold ph-lg", text: i18n.ts.watch, action: () => toggleWatch(true), }, ) : undefined, statePromise.then((state) => state.isMutedThread ? { icon: "ph-speaker-x ph-bold ph-lg", text: i18n.ts.unmuteThread, action: () => toggleThreadMute(false), } : { icon: "ph-speaker-x ph-bold ph-lg", text: i18n.ts.muteThread, action: () => toggleThreadMute(true), }, ), isAppearAuthor ? ($i.pinnedNoteIds || []).includes(appearNote.id) ? { icon: "ph-push-pin ph-bold ph-lg", text: i18n.ts.unpin, action: () => togglePin(false), } : { icon: "ph-push-pin ph-bold ph-lg", text: i18n.ts.pin, action: () => togglePin(true), } : undefined, instance.translatorAvailable ? { icon: "ph-translate ph-bold ph-lg", text: i18n.ts.translate, action: translate, } : undefined, appearNote.url || appearNote.uri ? { icon: "ph-arrow-square-out ph-bold ph-lg", text: i18n.ts.showOnRemote, action: () => { window.open(appearNote.url || appearNote.uri, "_blank"); }, } : undefined, { type: "parent", icon: "ph-share-network ph-bold ph-lg", text: i18n.ts.share, children: [ { icon: "ph-clipboard-text ph-bold ph-lg", text: i18n.ts.copyContent, action: copyContent, }, { icon: "ph-link-simple ph-bold ph-lg", text: i18n.ts.copyLink, action: copyLink, }, appearNote.url || appearNote.uri ? { icon: "ph-link-simple ph-bold ph-lg", text: `${i18n.ts.copyLink} (${i18n.ts.origin})`, action: copyOriginal, } : undefined, shareAvailable() ? { icon: "ph-share-network ph-bold ph-lg", text: i18n.ts.share, action: share, } : undefined, ], }, /* ...($i.isModerator || $i.isAdmin ? [ null, { icon: 'ph-megaphone-simple ph-bold ph-lg', text: i18n.ts.promote, action: promote }] : [] ),*/ null, !isAppearAuthor ? { icon: "ph-warning-circle ph-bold ph-lg", text: i18n.ts.reportAbuse, action: () => { const u = appearNote.url || appearNote.uri || `${url}/notes/${appearNote.id}`; os.popup( defineAsyncComponent( () => import("@/components/MkAbuseReportWindow.vue"), ), { user: appearNote.user, initialComment: `Note: ${u}\n-----\n`, }, {}, "closed", ); }, } : undefined, isAppearAuthor ? { icon: "ph-pencil-line ph-bold ph-lg", text: i18n.ts.edit, accent: true, action: edit, } : undefined, isAppearAuthor ? { icon: "ph-eraser ph-bold ph-lg", text: i18n.ts.deleteAndEdit, danger: true, action: delEdit, } : undefined, isAppearAuthor || isModerator ? { icon: "ph-trash ph-bold ph-lg", text: i18n.ts.delete, danger: true, action: del, } : undefined, !isAppearAuthor ? null : undefined, !isAppearAuthor ? { type: "parent", icon: "ph-user ph-bold ph-lg", text: i18n.ts.user, children: getUserMenu(appearNote.user), } : undefined, ].filter((x) => x !== undefined); } else { menu = [ appearNote.url || appearNote.uri ? { icon: "ph-arrow-square-out ph-bold ph-lg", text: i18n.ts.showOnRemote, action: () => { window.open(appearNote.url || appearNote.uri, "_blank"); }, } : undefined, { icon: "ph-clipboard-text ph-bold ph-lg", text: i18n.ts.copyContent, action: copyContent, }, { icon: "ph-link-simple ph-bold ph-lg", text: i18n.ts.copyLink, action: copyLink, }, appearNote.url || appearNote.uri ? { icon: "ph-link-simple ph-bold ph-lg", text: `${i18n.ts.copyLink} (${i18n.ts.origin})`, action: copyOriginal, } : undefined, shareAvailable() ? { icon: "ph-share-network ph-bold ph-lg", text: i18n.ts.share, action: share, } : undefined, ].filter((x) => x !== undefined); } if (noteActions.length > 0) { menu = menu.concat([ null, ...noteActions.map((action) => ({ icon: "ph-plug ph-bold ph-lg", text: action.title, action: () => { action.handler(appearNote); }, })), ]); } return menu; }