2023-11-27 18:17:53 +09:00
|
|
|
import { iAmModerator } from "@/account";
|
2023-07-20 04:17:05 +09:00
|
|
|
import { host } from "@/config";
|
2023-11-27 18:17:53 +09:00
|
|
|
import { i18n } from "@/i18n";
|
|
|
|
import type { Router } from "@/nirax";
|
2023-07-20 04:17:05 +09:00
|
|
|
import * as os from "@/os";
|
2023-10-29 20:09:26 +09:00
|
|
|
import { $i } from "@/reactiveAccount";
|
2023-07-20 04:17:05 +09:00
|
|
|
import { mainRouter } from "@/router";
|
2023-11-27 18:17:53 +09:00
|
|
|
import copyToClipboard from "@/scripts/copy-to-clipboard";
|
2023-10-17 02:12:35 +09:00
|
|
|
import icon from "@/scripts/icon";
|
2023-11-27 18:17:53 +09:00
|
|
|
import { userActions } from "@/store";
|
|
|
|
import * as Acct from "firefish-js/built/acct";
|
|
|
|
import { defineAsyncComponent } from "vue";
|
2023-07-20 04:17:05 +09:00
|
|
|
|
|
|
|
export function getUserMenu(user, router: Router = mainRouter) {
|
|
|
|
const meId = $i ? $i.id : null;
|
|
|
|
|
|
|
|
async function pushList() {
|
|
|
|
const t = i18n.ts.selectList; // なぜか後で参照すると null になるので最初にメモリに確保しておく
|
|
|
|
const lists = await os.api("users/lists/list");
|
|
|
|
if (lists.length === 0) {
|
|
|
|
os.alert({
|
|
|
|
type: "error",
|
|
|
|
text: i18n.ts.youHaveNoLists,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const { canceled, result: listId } = await os.select({
|
|
|
|
title: t,
|
|
|
|
items: lists.map((list) => ({
|
|
|
|
value: list.id,
|
|
|
|
text: list.name,
|
|
|
|
})),
|
|
|
|
});
|
|
|
|
if (canceled) return;
|
|
|
|
os.apiWithDialog("users/lists/push", {
|
2023-09-04 17:47:24 +09:00
|
|
|
listId,
|
2023-07-20 04:17:05 +09:00
|
|
|
userId: user.id,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function inviteGroup() {
|
|
|
|
const groups = await os.api("users/groups/owned");
|
|
|
|
if (groups.length === 0) {
|
|
|
|
os.alert({
|
|
|
|
type: "error",
|
|
|
|
text: i18n.ts.youHaveNoGroups,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const { canceled, result: groupId } = await os.select({
|
|
|
|
title: i18n.ts.group,
|
|
|
|
items: groups.map((group) => ({
|
|
|
|
value: group.id,
|
|
|
|
text: group.name,
|
|
|
|
})),
|
|
|
|
});
|
|
|
|
if (canceled) return;
|
|
|
|
os.apiWithDialog("users/groups/invite", {
|
2023-09-04 17:47:24 +09:00
|
|
|
groupId,
|
2023-07-20 04:17:05 +09:00
|
|
|
userId: user.id,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
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(): Promise<void> {
|
|
|
|
os.apiWithDialog(
|
|
|
|
user.isRenoteMuted ? "renote-mute/delete" : "renote-mute/create",
|
|
|
|
{
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
).then(() => {
|
|
|
|
user.isRenoteMuted = !user.isRenoteMuted;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function toggleBlock(): Promise<void> {
|
|
|
|
if (
|
|
|
|
!(await getConfirmed(
|
|
|
|
user.isBlocking ? i18n.ts.unblockConfirm : i18n.ts.blockConfirm,
|
|
|
|
))
|
|
|
|
)
|
|
|
|
return;
|
|
|
|
|
|
|
|
await os.apiWithDialog(
|
|
|
|
user.isBlocking ? "blocking/delete" : "blocking/create",
|
|
|
|
{
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
user.isBlocking = !user.isBlocking;
|
|
|
|
await os.api(user.isBlocking ? "mute/create" : "mute/delete", {
|
|
|
|
userId: user.id,
|
|
|
|
});
|
|
|
|
user.isMuted = user.isBlocking;
|
|
|
|
if (user.isBlocking) {
|
|
|
|
await os.api("following/delete", {
|
|
|
|
userId: user.id,
|
|
|
|
});
|
|
|
|
user.isFollowing = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function toggleSilence() {
|
|
|
|
if (
|
|
|
|
!(await getConfirmed(
|
|
|
|
i18n.t(user.isSilenced ? "unsilenceConfirm" : "silenceConfirm"),
|
|
|
|
))
|
|
|
|
)
|
|
|
|
return;
|
|
|
|
|
|
|
|
os.apiWithDialog(
|
|
|
|
user.isSilenced ? "admin/unsilence-user" : "admin/silence-user",
|
|
|
|
{
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
).then(() => {
|
|
|
|
user.isSilenced = !user.isSilenced;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function toggleSuspend() {
|
|
|
|
if (
|
|
|
|
!(await getConfirmed(
|
|
|
|
i18n.t(user.isSuspended ? "unsuspendConfirm" : "suspendConfirm"),
|
|
|
|
))
|
|
|
|
)
|
|
|
|
return;
|
|
|
|
|
|
|
|
os.apiWithDialog(
|
|
|
|
user.isSuspended ? "admin/unsuspend-user" : "admin/suspend-user",
|
|
|
|
{
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
).then(() => {
|
|
|
|
user.isSuspended = !user.isSuspended;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function reportAbuse() {
|
|
|
|
os.popup(
|
|
|
|
defineAsyncComponent(
|
|
|
|
() => import("@/components/MkAbuseReportWindow.vue"),
|
|
|
|
),
|
|
|
|
{
|
2023-09-04 17:47:24 +09:00
|
|
|
user,
|
2023-07-20 04:17:05 +09:00
|
|
|
},
|
|
|
|
{},
|
|
|
|
"closed",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getConfirmed(text: string): Promise<boolean> {
|
|
|
|
const confirm = await os.confirm({
|
|
|
|
type: "warning",
|
2023-08-10 04:45:41 +09:00
|
|
|
title: i18n.ts.confirm,
|
2023-07-20 04:17:05 +09:00
|
|
|
text,
|
|
|
|
});
|
|
|
|
|
|
|
|
return !confirm.canceled;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function invalidateFollow() {
|
|
|
|
if (!(await getConfirmed(i18n.ts.breakFollowConfirm))) return;
|
|
|
|
|
|
|
|
os.apiWithDialog("following/invalidate", {
|
|
|
|
userId: user.id,
|
|
|
|
}).then(() => {
|
|
|
|
user.isFollowed = !user.isFollowed;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let menu = [
|
|
|
|
{
|
|
|
|
type: "label",
|
|
|
|
text: user.host
|
|
|
|
? `@${user.username}@${user.host || host}`
|
|
|
|
: `@${user.username}`,
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-at")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.copyUsername,
|
|
|
|
action: () => {
|
|
|
|
copyToClipboard(`@${user.username}@${user.host || host}`);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-info")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.info,
|
|
|
|
action: () => {
|
|
|
|
router.push(`/user-info/${user.id}`);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-newspaper")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts._feeds.copyFeed,
|
|
|
|
type: "parent",
|
|
|
|
children: [
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-rss")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts._feeds.rss,
|
|
|
|
action: () => {
|
|
|
|
copyToClipboard(`https://${host}/@${user.username}.rss`);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-atom")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts._feeds.atom,
|
|
|
|
action: () => {
|
|
|
|
copyToClipboard(`https://${host}/@${user.username}.atom`);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-brackets-curly")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts._feeds.jsonFeed,
|
|
|
|
action: () => {
|
|
|
|
copyToClipboard(`https://${host}/@${user.username}.json`);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-envelope-simple-open")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.sendMessage,
|
|
|
|
action: () => {
|
|
|
|
os.post({ specified: user });
|
|
|
|
},
|
|
|
|
},
|
|
|
|
meId !== user.id
|
|
|
|
? {
|
|
|
|
type: "link",
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-chats-teardrop")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.startMessaging,
|
|
|
|
to: `/my/messaging/${Acct.toString(user)}`,
|
|
|
|
}
|
|
|
|
: undefined,
|
|
|
|
user.host != null && user.url
|
|
|
|
? {
|
|
|
|
type: "a",
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-arrow-square-out")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.showOnRemote,
|
|
|
|
href: user.url,
|
|
|
|
target: "_blank",
|
|
|
|
}
|
|
|
|
: undefined,
|
|
|
|
null,
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-list-bullets")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.addToList,
|
|
|
|
action: pushList,
|
|
|
|
},
|
|
|
|
meId !== user.id
|
|
|
|
? {
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-users-three")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.inviteToGroup,
|
|
|
|
action: inviteGroup,
|
|
|
|
}
|
|
|
|
: undefined,
|
|
|
|
null,
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: user.isRenoteMuted ? "ph-eye ph-lg" : "ph-eye-slash ph-lg",
|
2023-07-20 04:17:05 +09:00
|
|
|
text: user.isRenoteMuted ? i18n.ts.renoteUnmute : i18n.ts.renoteMute,
|
|
|
|
action: toggleRenoteMute,
|
|
|
|
},
|
|
|
|
] as any;
|
|
|
|
|
|
|
|
if ($i && meId !== user.id) {
|
|
|
|
menu = menu.concat([
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: user.isMuted ? "ph-eye ph-lg" : "ph-eye-slash ph-lg",
|
2023-07-20 04:17:05 +09:00
|
|
|
text: user.isMuted ? i18n.ts.unmute : i18n.ts.mute,
|
|
|
|
hidden: user.isBlocking === true,
|
|
|
|
action: toggleMute,
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-prohibit")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: user.isBlocking ? i18n.ts.unblock : i18n.ts.block,
|
|
|
|
action: toggleBlock,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
|
|
|
|
if (user.isFollowed) {
|
|
|
|
menu = menu.concat([
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-link-break")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.breakFollow,
|
|
|
|
action: invalidateFollow,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
menu = menu.concat([
|
|
|
|
null,
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-warning-circle")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.reportAbuse,
|
|
|
|
action: reportAbuse,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
|
|
|
|
if (iAmModerator) {
|
|
|
|
menu = menu.concat([
|
|
|
|
null,
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-microphone-slash")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: user.isSilenced ? i18n.ts.unsilence : i18n.ts.silence,
|
|
|
|
action: toggleSilence,
|
|
|
|
},
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-snowflake")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: user.isSuspended ? i18n.ts.unsuspend : i18n.ts.suspend,
|
|
|
|
action: toggleSuspend,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($i && meId === user.id) {
|
|
|
|
menu = menu.concat([
|
|
|
|
null,
|
|
|
|
{
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-pencil")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: i18n.ts.editProfile,
|
|
|
|
action: () => {
|
|
|
|
router.push("/settings/profile");
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (userActions.length > 0) {
|
|
|
|
menu = menu.concat([
|
|
|
|
null,
|
|
|
|
...userActions.map((action) => ({
|
2023-10-17 02:12:35 +09:00
|
|
|
icon: `${icon("ph-plug")}`,
|
2023-07-20 04:17:05 +09:00
|
|
|
text: action.title,
|
|
|
|
action: () => {
|
|
|
|
action.handler(user);
|
|
|
|
},
|
|
|
|
})),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|