mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-10-25 10:44:51 +00:00 
			
		
		
		
	* wip * wip * wip * test * wip rollup pluginでsearchIndexの情報生成 * wip * SPDX * wip: markerIdを自動付与 * rollupでビルド時・devモード時に毎回uuidを生成するように * 開発サーバーでだけ必要な挙動は開発サーバーのみで * 条件が逆 * wip: childrenの生成 * update comment * update comment * rename auto generated file * hashをパスと行数から決定 * Update privacy.vue * Update privacy.vue * wip * Update general.vue * Update general.vue * wip * wip * Update SearchMarker.vue * wip * Update profile.vue * Update mute-block.vue * Update mute-block.vue * Update general.vue * Update general.vue * childrenがduplicate key errorを吐く問題をいったん解決 * マーカーの形を成形 * loggerを置きかえ * とりあえず省略記法に対応 * Refactor and Format codes * wip * Update settings-search-index.ts * wip * wip * とりあえず不確定要因の仮置きidを削除 * hashの生成を正規化(絶対パスになっていたのを緩和) * pathの入力を省略可能に * adminでもパス生成できるように * Update settings-search-index.ts * Update privacy.vue * wip * build searchIndex * wip * build * Update general.vue * build * Update sounds.vue * build * build * Update sounds.vue * 🎨 * 🎨 * Update privacy.vue * Update privacy.vue * Update security.vue * create-search-indexを多少改善 * build * Update 2fa.vue * wip * 必ずtransformCodeCacheを利用するように, キャッシュの明確な受け渡しを定義 * キャッシュはdevServerでなくても更新 * Revert "wip" This reverts commit 41bffd3a13f55618bf939dc1c9acb2a77ead4054. * inlining * wip * Update theme.vue * 🎨 * wip normalize * Update theme.vue * キャッシュのパス変換 * build * wip * wip * Update SearchMarker.vue * i18n.ts['key'] の形式が取り出せない問題のFix * build * 仮でpath入れ * 必ず絶対パスが使われるように * wip * 🎨 * storybookビルド時はcreateSearchIndexをしない * inliningの構造化 * format code * Update index.vue * wip * wip * 🎨 * wip * wip * wip * wip * wip * wip * wip * wip * clean up * wip * wip * wip * Update rollup-plugin-unwind-css-module-class-name.test.ts * Update navbar.vue * clean up * wip * wip * wip * wip * wip * Update preferences-backups.vue * Update common.ts * Update preferences.ts * wip * wip * wip * wip * Update MkPreferenceContainer.vue * Update MkPreferenceContainer.vue * Update MkPreferenceContainer.vue * enhance: 検索で上下矢印を使用することで検索結果を移動できるように * Update main-boot.ts * refactor * wip * Update sounds.vue * fix(frontend): PageWindowでSearchMarkerが動作するように * enhance(frontend): SearchMarkerの点滅を一定時間で止める * wip * lint fix * fix: 子要素監視が抜けていたのを修正 * アニメーションの回数はCSSで制御するように * refactor * enhance(frontend): 検索インデックス作成時のログを削減 * revert * fix * fix * Update preferences.ts * Update preferences.ts * wip * Update preferences.ts * wip * 🎨 * wip * Update MkPreferenceContainer.vue * wip * Update preferences.ts * wip * Update preferences.ts * Update preferences.ts * wip * wip * Update preferences.ts * wip * wip * Update preferences.ts * Update CHANGELOG.md * Update preferences.ts * Update deck-store.ts * deckStoreをdefaultStoreに統合 * wip * defaultStore -> store * Update profile.ts * wip * refactor * wip: plugin * plugin * plugin * plugin * Update plugin.ts * wip * Update plugin.vue * Update preferences.ts * Update main-boot.ts * wip * fix test * Update plugin.vue * Update plugin.vue * Update utility.ts * wip * wip * Update utility.ts * wip * wip * clean up * Update utility.ts --------- Co-authored-by: tai-cha <dev@taichan.site> Co-authored-by: taichan <40626578+tai-cha@users.noreply.github.com> Co-authored-by: kakkokari-gtyih <67428053+kakkokari-gtyih@users.noreply.github.com>
		
			
				
	
	
		
			441 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			441 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /*
 | |
|  * SPDX-FileCopyrightText: syuilo and misskey-project
 | |
|  * SPDX-License-Identifier: AGPL-3.0-only
 | |
|  */
 | |
| 
 | |
| import { toUnicode } from 'punycode.js';
 | |
| import { defineAsyncComponent, ref, watch } from 'vue';
 | |
| import * as Misskey from 'misskey-js';
 | |
| import { host, url } from '@@/js/config.js';
 | |
| import type { IRouter } from '@/nirax.js';
 | |
| import type { MenuItem } from '@/types/menu.js';
 | |
| import { i18n } from '@/i18n.js';
 | |
| import { copyToClipboard } from '@/scripts/copy-to-clipboard.js';
 | |
| import * as os from '@/os.js';
 | |
| import { misskeyApi } from '@/scripts/misskey-api.js';
 | |
| import { userActions } from '@/store.js';
 | |
| import { $i, iAmModerator } from '@/account.js';
 | |
| import { notesSearchAvailable, canSearchNonLocalNotes } from '@/scripts/check-permissions.js';
 | |
| import { antennasCache, rolesCache, userListsCache } from '@/cache.js';
 | |
| import { mainRouter } from '@/router/main.js';
 | |
| import { genEmbedCode } from '@/scripts/get-embed-code.js';
 | |
| import { prefer } from '@/preferences.js';
 | |
| 
 | |
| export function getUserMenu(user: Misskey.entities.UserDetailed, router: IRouter = mainRouter) {
 | |
| 	const meId = $i ? $i.id : null;
 | |
| 
 | |
| 	const cleanups = [] as (() => void)[];
 | |
| 
 | |
| 	async function toggleMute() {
 | |
| 		if (user.isMuted) {
 | |
| 			os.apiWithDialog('mute/delete', {
 | |
| 				userId: user.id,
 | |
| 			}).then(() => {
 | |
| 				user.isMuted = false;
 | |
| 			});
 | |
| 		} else {
 | |
| 			const { canceled, result: period } = await os.select({
 | |
| 				title: i18n.ts.mutePeriod,
 | |
| 				items: [{
 | |
| 					value: 'indefinitely', text: i18n.ts.indefinitely,
 | |
| 				}, {
 | |
| 					value: 'tenMinutes', text: i18n.ts.tenMinutes,
 | |
| 				}, {
 | |
| 					value: 'oneHour', text: i18n.ts.oneHour,
 | |
| 				}, {
 | |
| 					value: 'oneDay', text: i18n.ts.oneDay,
 | |
| 				}, {
 | |
| 					value: 'oneWeek', text: i18n.ts.oneWeek,
 | |
| 				}],
 | |
| 				default: 'indefinitely',
 | |
| 			});
 | |
| 			if (canceled) return;
 | |
| 
 | |
| 			const expiresAt = period === 'indefinitely' ? null
 | |
| 				: period === 'tenMinutes' ? Date.now() + (1000 * 60 * 10)
 | |
| 				: period === 'oneHour' ? Date.now() + (1000 * 60 * 60)
 | |
| 				: period === 'oneDay' ? Date.now() + (1000 * 60 * 60 * 24)
 | |
| 				: period === 'oneWeek' ? Date.now() + (1000 * 60 * 60 * 24 * 7)
 | |
| 				: null;
 | |
| 
 | |
| 			os.apiWithDialog('mute/create', {
 | |
| 				userId: user.id,
 | |
| 				expiresAt,
 | |
| 			}).then(() => {
 | |
| 				user.isMuted = true;
 | |
| 			});
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	async function toggleRenoteMute() {
 | |
| 		os.apiWithDialog(user.isRenoteMuted ? 'renote-mute/delete' : 'renote-mute/create', {
 | |
| 			userId: user.id,
 | |
| 		}).then(() => {
 | |
| 			user.isRenoteMuted = !user.isRenoteMuted;
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	async function toggleBlock() {
 | |
| 		if (!await getConfirmed(user.isBlocking ? i18n.ts.unblockConfirm : i18n.ts.blockConfirm)) return;
 | |
| 
 | |
| 		os.apiWithDialog(user.isBlocking ? 'blocking/delete' : 'blocking/create', {
 | |
| 			userId: user.id,
 | |
| 		}).then(() => {
 | |
| 			user.isBlocking = !user.isBlocking;
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	async function toggleNotify() {
 | |
| 		os.apiWithDialog('following/update', {
 | |
| 			userId: user.id,
 | |
| 			notify: user.notify === 'normal' ? 'none' : 'normal',
 | |
| 		}).then(() => {
 | |
| 			user.notify = user.notify === 'normal' ? 'none' : 'normal';
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	function reportAbuse() {
 | |
| 		const { dispose } = os.popup(defineAsyncComponent(() => import('@/components/MkAbuseReportWindow.vue')), {
 | |
| 			user: user,
 | |
| 		}, {
 | |
| 			closed: () => dispose(),
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	async function getConfirmed(text: string): Promise<boolean> {
 | |
| 		const confirm = await os.confirm({
 | |
| 			type: 'warning',
 | |
| 			title: 'confirm',
 | |
| 			text,
 | |
| 		});
 | |
| 
 | |
| 		return !confirm.canceled;
 | |
| 	}
 | |
| 
 | |
| 	async function userInfoUpdate() {
 | |
| 		os.apiWithDialog('federation/update-remote-user', {
 | |
| 			userId: user.id,
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	async function invalidateFollow() {
 | |
| 		if (!await getConfirmed(i18n.ts.breakFollowConfirm)) return;
 | |
| 
 | |
| 		os.apiWithDialog('following/invalidate', {
 | |
| 			userId: user.id,
 | |
| 		}).then(() => {
 | |
| 			user.isFollowed = !user.isFollowed;
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	async function editMemo(): Promise<void> {
 | |
| 		const userDetailed = await misskeyApi('users/show', {
 | |
| 			userId: user.id,
 | |
| 		});
 | |
| 		const { canceled, result } = await os.form(i18n.ts.editMemo, {
 | |
| 			memo: {
 | |
| 				type: 'string',
 | |
| 				required: true,
 | |
| 				multiline: true,
 | |
| 				label: i18n.ts.memo,
 | |
| 				default: userDetailed.memo,
 | |
| 			},
 | |
| 		});
 | |
| 		if (canceled) return;
 | |
| 
 | |
| 		os.apiWithDialog('users/update-memo', {
 | |
| 			memo: result.memo,
 | |
| 			userId: user.id,
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	const menuItems: MenuItem[] = [];
 | |
| 
 | |
| 	menuItems.push({
 | |
| 		icon: 'ti ti-at',
 | |
| 		text: i18n.ts.copyUsername,
 | |
| 		action: () => {
 | |
| 			copyToClipboard(`@${user.username}@${user.host ?? host}`);
 | |
| 		},
 | |
| 	});
 | |
| 
 | |
| 	if (notesSearchAvailable && (user.host == null || canSearchNonLocalNotes)) {
 | |
| 		menuItems.push({
 | |
| 			icon: 'ti ti-search',
 | |
| 			text: i18n.ts.searchThisUsersNotes,
 | |
| 			action: () => {
 | |
| 				router.push(`/search?username=${encodeURIComponent(user.username)}${user.host != null ? '&host=' + encodeURIComponent(user.host) : ''}`);
 | |
| 			},
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	if (iAmModerator) {
 | |
| 		menuItems.push({
 | |
| 			icon: 'ti ti-user-exclamation',
 | |
| 			text: i18n.ts.moderation,
 | |
| 			action: () => {
 | |
| 				router.push(`/admin/user/${user.id}`);
 | |
| 			},
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	menuItems.push({
 | |
| 		icon: 'ti ti-rss',
 | |
| 		text: i18n.ts.copyRSS,
 | |
| 		action: () => {
 | |
| 			copyToClipboard(`${user.host ?? host}/@${user.username}.atom`);
 | |
| 		},
 | |
| 	});
 | |
| 
 | |
| 	if (user.host != null && user.url != null) {
 | |
| 		menuItems.push({
 | |
| 			icon: 'ti ti-external-link',
 | |
| 			text: i18n.ts.showOnRemote,
 | |
| 			action: () => {
 | |
| 				if (user.url == null) return;
 | |
| 				window.open(user.url, '_blank', 'noopener');
 | |
| 			},
 | |
| 		});
 | |
| 	} else {
 | |
| 		menuItems.push({
 | |
| 			icon: 'ti ti-code',
 | |
| 			text: i18n.ts.genEmbedCode,
 | |
| 			type: 'parent',
 | |
| 			children: [{
 | |
| 				text: i18n.ts.noteOfThisUser,
 | |
| 				action: () => {
 | |
| 					genEmbedCode('user-timeline', user.id);
 | |
| 				},
 | |
| 			}], // TODO: ユーザーカードの埋め込みなど
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	menuItems.push({
 | |
| 		icon: 'ti ti-share',
 | |
| 		text: i18n.ts.copyProfileUrl,
 | |
| 		action: () => {
 | |
| 			const canonical = user.host === null ? `@${user.username}` : `@${user.username}@${toUnicode(user.host)}`;
 | |
| 			copyToClipboard(`${url}/${canonical}`);
 | |
| 		},
 | |
| 	});
 | |
| 
 | |
| 	if ($i) {
 | |
| 		menuItems.push({
 | |
| 			icon: 'ti ti-mail',
 | |
| 			text: i18n.ts.sendMessage,
 | |
| 			action: () => {
 | |
| 				const canonical = user.host === null ? `@${user.username}` : `@${user.username}@${user.host}`;
 | |
| 				os.post({ specified: user, initialText: `${canonical} ` });
 | |
| 			},
 | |
| 		}, { type: 'divider' }, {
 | |
| 			icon: 'ti ti-pencil',
 | |
| 			text: i18n.ts.editMemo,
 | |
| 			action: editMemo,
 | |
| 		}, {
 | |
| 			type: 'parent',
 | |
| 			icon: 'ti ti-list',
 | |
| 			text: i18n.ts.addToList,
 | |
| 			children: async () => {
 | |
| 				const lists = await userListsCache.fetch();
 | |
| 				return lists.map(list => {
 | |
| 					const isListed = ref(list.userIds?.includes(user.id) ?? false);
 | |
| 					cleanups.push(watch(isListed, () => {
 | |
| 						if (isListed.value) {
 | |
| 							os.apiWithDialog('users/lists/push', {
 | |
| 								listId: list.id,
 | |
| 								userId: user.id,
 | |
| 							}).then(() => {
 | |
| 								list.userIds?.push(user.id);
 | |
| 							});
 | |
| 						} else {
 | |
| 							os.apiWithDialog('users/lists/pull', {
 | |
| 								listId: list.id,
 | |
| 								userId: user.id,
 | |
| 							}).then(() => {
 | |
| 								list.userIds?.splice(list.userIds.indexOf(user.id), 1);
 | |
| 							});
 | |
| 						}
 | |
| 					}));
 | |
| 
 | |
| 					return {
 | |
| 						type: 'switch',
 | |
| 						text: list.name,
 | |
| 						ref: isListed,
 | |
| 					};
 | |
| 				});
 | |
| 			},
 | |
| 		}, {
 | |
| 			type: 'parent',
 | |
| 			icon: 'ti ti-antenna',
 | |
| 			text: i18n.ts.addToAntenna,
 | |
| 			children: async () => {
 | |
| 				const antennas = await antennasCache.fetch();
 | |
| 				const canonical = user.host === null ? `@${user.username}` : `@${user.username}@${toUnicode(user.host)}`;
 | |
| 				return antennas.filter((a) => a.src === 'users').map(antenna => ({
 | |
| 					text: antenna.name,
 | |
| 					action: async () => {
 | |
| 						await os.apiWithDialog('antennas/update', {
 | |
| 							antennaId: antenna.id,
 | |
| 							name: antenna.name,
 | |
| 							keywords: antenna.keywords,
 | |
| 							excludeKeywords: antenna.excludeKeywords,
 | |
| 							src: antenna.src,
 | |
| 							userListId: antenna.userListId,
 | |
| 							users: [...antenna.users, canonical],
 | |
| 							caseSensitive: antenna.caseSensitive,
 | |
| 							withReplies: antenna.withReplies,
 | |
| 							withFile: antenna.withFile,
 | |
| 							notify: antenna.notify,
 | |
| 						});
 | |
| 						antennasCache.delete();
 | |
| 					},
 | |
| 				}));
 | |
| 			},
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	if ($i && meId !== user.id) {
 | |
| 		if (iAmModerator) {
 | |
| 			menuItems.push({
 | |
| 				type: 'parent',
 | |
| 				icon: 'ti ti-badges',
 | |
| 				text: i18n.ts.roles,
 | |
| 				children: async () => {
 | |
| 					const roles = await rolesCache.fetch();
 | |
| 
 | |
| 					return roles.filter(r => r.target === 'manual').map(r => ({
 | |
| 						text: r.name,
 | |
| 						action: async () => {
 | |
| 							const { canceled, result: period } = await os.select({
 | |
| 								title: i18n.ts.period + ': ' + r.name,
 | |
| 								items: [{
 | |
| 									value: 'indefinitely', text: i18n.ts.indefinitely,
 | |
| 								}, {
 | |
| 									value: 'oneHour', text: i18n.ts.oneHour,
 | |
| 								}, {
 | |
| 									value: 'oneDay', text: i18n.ts.oneDay,
 | |
| 								}, {
 | |
| 									value: 'oneWeek', text: i18n.ts.oneWeek,
 | |
| 								}, {
 | |
| 									value: 'oneMonth', text: i18n.ts.oneMonth,
 | |
| 								}],
 | |
| 								default: 'indefinitely',
 | |
| 							});
 | |
| 							if (canceled) return;
 | |
| 
 | |
| 							const expiresAt = period === 'indefinitely' ? null
 | |
| 								: period === 'oneHour' ? Date.now() + (1000 * 60 * 60)
 | |
| 								: period === 'oneDay' ? Date.now() + (1000 * 60 * 60 * 24)
 | |
| 								: period === 'oneWeek' ? Date.now() + (1000 * 60 * 60 * 24 * 7)
 | |
| 								: period === 'oneMonth' ? Date.now() + (1000 * 60 * 60 * 24 * 30)
 | |
| 								: null;
 | |
| 
 | |
| 							os.apiWithDialog('admin/roles/assign', { roleId: r.id, userId: user.id, expiresAt });
 | |
| 						},
 | |
| 					}));
 | |
| 				},
 | |
| 			});
 | |
| 		}
 | |
| 
 | |
| 		// フォローしたとしても user.isFollowing はリアルタイム更新されないので不便なため
 | |
| 		//if (user.isFollowing) {
 | |
| 		const withRepliesRef = ref(user.withReplies ?? false);
 | |
| 
 | |
| 		menuItems.push({
 | |
| 			type: 'switch',
 | |
| 			icon: 'ti ti-messages',
 | |
| 			text: i18n.ts.showRepliesToOthersInTimeline,
 | |
| 			ref: withRepliesRef,
 | |
| 		}, {
 | |
| 			icon: user.notify === 'none' ? 'ti ti-bell' : 'ti ti-bell-off',
 | |
| 			text: user.notify === 'none' ? i18n.ts.notifyNotes : i18n.ts.unnotifyNotes,
 | |
| 			action: toggleNotify,
 | |
| 		});
 | |
| 
 | |
| 		watch(withRepliesRef, (withReplies) => {
 | |
| 			misskeyApi('following/update', {
 | |
| 				userId: user.id,
 | |
| 				withReplies,
 | |
| 			}).then(() => {
 | |
| 				user.withReplies = withReplies;
 | |
| 			});
 | |
| 		});
 | |
| 		//}
 | |
| 
 | |
| 		menuItems.push({ type: 'divider' }, {
 | |
| 			icon: user.isMuted ? 'ti ti-eye' : 'ti ti-eye-off',
 | |
| 			text: user.isMuted ? i18n.ts.unmute : i18n.ts.mute,
 | |
| 			action: toggleMute,
 | |
| 		}, {
 | |
| 			icon: user.isRenoteMuted ? 'ti ti-repeat' : 'ti ti-repeat-off',
 | |
| 			text: user.isRenoteMuted ? i18n.ts.renoteUnmute : i18n.ts.renoteMute,
 | |
| 			action: toggleRenoteMute,
 | |
| 		}, {
 | |
| 			icon: 'ti ti-ban',
 | |
| 			text: user.isBlocking ? i18n.ts.unblock : i18n.ts.block,
 | |
| 			action: toggleBlock,
 | |
| 		});
 | |
| 
 | |
| 		if (user.isFollowed) {
 | |
| 			menuItems.push({
 | |
| 				icon: 'ti ti-link-off',
 | |
| 				text: i18n.ts.breakFollow,
 | |
| 				action: invalidateFollow,
 | |
| 			});
 | |
| 		}
 | |
| 
 | |
| 		menuItems.push({ type: 'divider' }, {
 | |
| 			icon: 'ti ti-exclamation-circle',
 | |
| 			text: i18n.ts.reportAbuse,
 | |
| 			action: reportAbuse,
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	if (user.host !== null) {
 | |
| 		menuItems.push({ type: 'divider' }, {
 | |
| 			icon: 'ti ti-refresh',
 | |
| 			text: i18n.ts.updateRemoteUser,
 | |
| 			action: userInfoUpdate,
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	if (prefer.s.devMode) {
 | |
| 		menuItems.push({ type: 'divider' }, {
 | |
| 			icon: 'ti ti-id',
 | |
| 			text: i18n.ts.copyUserId,
 | |
| 			action: () => {
 | |
| 				copyToClipboard(user.id);
 | |
| 			},
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	if ($i && meId === user.id) {
 | |
| 		menuItems.push({ type: 'divider' }, {
 | |
| 			icon: 'ti ti-pencil',
 | |
| 			text: i18n.ts.editProfile,
 | |
| 			action: () => {
 | |
| 				router.push('/settings/profile');
 | |
| 			},
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	if (userActions.length > 0) {
 | |
| 		menuItems.push({ type: 'divider' }, ...userActions.map(action => ({
 | |
| 			icon: 'ti ti-plug',
 | |
| 			text: action.title,
 | |
| 			action: () => {
 | |
| 				action.handler(user);
 | |
| 			},
 | |
| 		})));
 | |
| 	}
 | |
| 
 | |
| 	return {
 | |
| 		menu: menuItems,
 | |
| 		cleanup: () => {
 | |
| 			if (_DEV_) console.log('user menu cleanup', cleanups);
 | |
| 			for (const cl of cleanups) {
 | |
| 				cl();
 | |
| 			}
 | |
| 		},
 | |
| 	};
 | |
| }
 |