2024-02-12 16:40:46 +01:00
|
|
|
import type { entities } from "firefish-js";
|
2023-12-05 08:12:33 +01:00
|
|
|
import { defineAsyncComponent } from "vue";
|
|
|
|
import { i18n } from "./i18n";
|
2023-01-13 05:40:33 +01:00
|
|
|
import { apiUrl } from "@/config";
|
2024-03-07 03:06:45 +01:00
|
|
|
import { me } from "@/me";
|
2024-03-16 16:49:12 +01:00
|
|
|
import { alert, api, popup, popupMenu, waiting } from "@/os";
|
2023-11-16 21:18:19 +01:00
|
|
|
import { del, get, set } from "@/scripts/idb-proxy";
|
|
|
|
import { reloadChannel, unisonReload } from "@/scripts/unison-reload";
|
2020-12-19 02:55:52 +01:00
|
|
|
|
|
|
|
// TODO: 他のタブと永続化されたstateを同期
|
|
|
|
|
2024-02-12 16:40:46 +01:00
|
|
|
export type Account = entities.MeDetailed;
|
2020-12-19 02:55:52 +01:00
|
|
|
|
2024-03-07 03:13:21 +01:00
|
|
|
export async function signOut() {
|
2021-08-20 12:38:16 +02:00
|
|
|
waiting();
|
2023-01-13 05:40:33 +01:00
|
|
|
localStorage.removeItem("account");
|
2021-08-20 12:38:16 +02:00
|
|
|
|
2024-03-07 03:06:45 +01:00
|
|
|
await removeAccount(me.id);
|
2021-08-20 12:38:16 +02:00
|
|
|
|
2022-06-24 14:16:05 +02:00
|
|
|
const accounts = await getAccounts();
|
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
// #region Remove service worker registration
|
2021-08-21 04:51:46 +02:00
|
|
|
try {
|
2021-09-04 11:09:53 +02:00
|
|
|
if (navigator.serviceWorker.controller) {
|
|
|
|
const registration = await navigator.serviceWorker.ready;
|
|
|
|
const push = await registration.pushManager.getSubscription();
|
|
|
|
if (push) {
|
|
|
|
await fetch(`${apiUrl}/sw/unregister`, {
|
2023-01-13 05:40:33 +01:00
|
|
|
method: "POST",
|
2021-09-04 11:09:53 +02:00
|
|
|
body: JSON.stringify({
|
2024-03-07 03:06:45 +01:00
|
|
|
i: me.token,
|
2021-09-04 11:09:53 +02:00
|
|
|
endpoint: push.endpoint,
|
|
|
|
}),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (accounts.length === 0) {
|
2023-01-13 05:40:33 +01:00
|
|
|
await navigator.serviceWorker.getRegistrations().then((registrations) => {
|
|
|
|
return Promise.all(
|
|
|
|
registrations.map((registration) => registration.unregister()),
|
|
|
|
);
|
|
|
|
});
|
2021-09-04 11:09:53 +02:00
|
|
|
}
|
2022-05-26 15:53:09 +02:00
|
|
|
} catch (err) {}
|
2023-09-02 01:27:33 +02:00
|
|
|
// #endregion
|
2021-08-20 12:38:16 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
document.cookie = "igi=; path=/";
|
2021-08-20 12:38:16 +02:00
|
|
|
|
2024-03-07 03:13:21 +01:00
|
|
|
if (accounts.length > 0) signIn(accounts[0].token);
|
2023-01-13 05:40:33 +01:00
|
|
|
else unisonReload("/");
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export async function getAccounts(): Promise<
|
|
|
|
{ id: Account["id"]; token: Account["token"] }[]
|
|
|
|
> {
|
|
|
|
return (await get("accounts")) || [];
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export async function addAccount(id: Account["id"], token: Account["token"]) {
|
2021-08-20 12:38:16 +02:00
|
|
|
const accounts = await getAccounts();
|
2023-01-13 05:40:33 +01:00
|
|
|
if (!accounts.some((x) => x.id === id)) {
|
|
|
|
await set("accounts", accounts.concat([{ id, token }]));
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export async function removeAccount(id: Account["id"]) {
|
2022-06-20 10:38:49 +02:00
|
|
|
const accounts = await getAccounts();
|
2023-01-13 05:40:33 +01:00
|
|
|
accounts.splice(
|
|
|
|
accounts.findIndex((x) => x.id === id),
|
|
|
|
1,
|
|
|
|
);
|
2022-06-20 10:38:49 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (accounts.length > 0) await set("accounts", accounts);
|
|
|
|
else await del("accounts");
|
2022-06-20 10:38:49 +02:00
|
|
|
}
|
|
|
|
|
2022-04-11 15:50:53 +02:00
|
|
|
function fetchAccount(token: string): Promise<Account> {
|
2020-12-19 02:55:52 +01:00
|
|
|
return new Promise((done, fail) => {
|
|
|
|
// Fetch user
|
|
|
|
fetch(`${apiUrl}/i`, {
|
2023-01-13 05:40:33 +01:00
|
|
|
method: "POST",
|
2020-12-19 02:55:52 +01:00
|
|
|
body: JSON.stringify({
|
2022-06-20 10:38:49 +02:00
|
|
|
i: token,
|
|
|
|
}),
|
2020-12-19 02:55:52 +01:00
|
|
|
})
|
2023-01-13 05:40:33 +01:00
|
|
|
.then((res) => res.json())
|
|
|
|
.then((res) => {
|
|
|
|
if (res.error) {
|
|
|
|
if (res.error.id === "a8c724b3-6e9c-4b46-b1a8-bc3ed6258370") {
|
2023-09-27 20:16:58 +02:00
|
|
|
showSuspendedDialog();
|
2024-03-07 03:13:21 +01:00
|
|
|
signOut();
|
2023-01-13 05:40:33 +01:00
|
|
|
} else {
|
|
|
|
alert({
|
|
|
|
type: "error",
|
|
|
|
title: i18n.ts.failedToFetchAccountInformation,
|
|
|
|
text: JSON.stringify(res.error),
|
|
|
|
});
|
|
|
|
}
|
2021-09-18 19:23:12 +02:00
|
|
|
} else {
|
2023-01-13 05:40:33 +01:00
|
|
|
res.token = token;
|
|
|
|
done(res);
|
2021-09-18 19:23:12 +02:00
|
|
|
}
|
2023-01-13 05:40:33 +01:00
|
|
|
})
|
|
|
|
.catch(fail);
|
2020-12-19 02:55:52 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-09-27 20:16:58 +02:00
|
|
|
function showSuspendedDialog() {
|
|
|
|
alert({
|
|
|
|
type: "error",
|
|
|
|
title: i18n.ts.yourAccountSuspendedTitle,
|
|
|
|
text: i18n.ts.yourAccountSuspendedDescription,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-05-26 15:53:09 +02:00
|
|
|
export function updateAccount(accountData) {
|
|
|
|
for (const [key, value] of Object.entries(accountData)) {
|
2024-03-07 03:06:45 +01:00
|
|
|
me[key] = value;
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
2024-03-07 03:06:45 +01:00
|
|
|
localStorage.setItem("account", JSON.stringify(me));
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
|
|
|
|
2024-03-07 03:06:45 +01:00
|
|
|
export async function refreshAccount() {
|
|
|
|
const accountData = await fetchAccount(me.token);
|
|
|
|
return updateAccount(accountData);
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
|
|
|
|
2024-03-07 03:13:21 +01:00
|
|
|
export async function signIn(token: Account["token"], redirect?: string) {
|
2020-12-19 02:55:52 +01:00
|
|
|
waiting();
|
2023-01-13 05:40:33 +01:00
|
|
|
if (_DEV_) console.log("logging as token ", token);
|
2024-03-07 03:06:45 +01:00
|
|
|
const newAccount = await fetchAccount(token);
|
|
|
|
localStorage.setItem("account", JSON.stringify(newAccount));
|
2022-03-19 11:08:55 +01:00
|
|
|
document.cookie = `token=${token}; path=/; max-age=31536000`; // bull dashboardの認証とかで使う
|
2024-03-07 03:06:45 +01:00
|
|
|
await addAccount(newAccount.id, token);
|
2021-08-20 12:38:16 +02:00
|
|
|
|
|
|
|
if (redirect) {
|
2021-11-04 16:09:13 +01:00
|
|
|
// 他のタブは再読み込みするだけ
|
|
|
|
reloadChannel.postMessage(null);
|
|
|
|
// このページはredirectで指定された先に移動
|
2021-08-20 12:38:16 +02:00
|
|
|
location.href = redirect;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-02-17 13:36:56 +01:00
|
|
|
unisonReload();
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export async function openAccountMenu(
|
|
|
|
opts: {
|
|
|
|
includeCurrentAccount?: boolean;
|
|
|
|
withExtraOperation: boolean;
|
2024-02-12 16:40:46 +01:00
|
|
|
active?: entities.UserDetailed["id"];
|
|
|
|
onChoose?: (account: entities.UserDetailed) => void;
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
ev: MouseEvent,
|
2024-02-15 05:35:26 +01:00
|
|
|
isMobile?: boolean,
|
2023-01-13 05:40:33 +01:00
|
|
|
) {
|
2021-10-10 08:19:16 +02:00
|
|
|
function showSigninDialog() {
|
2023-01-13 05:40:33 +01:00
|
|
|
popup(
|
|
|
|
defineAsyncComponent(() => import("@/components/MkSigninDialog.vue")),
|
|
|
|
{},
|
|
|
|
{
|
|
|
|
done: (res) => {
|
|
|
|
addAccount(res.id, res.i);
|
2023-07-07 02:44:32 +02:00
|
|
|
switchAccountWithToken(res.i);
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
2021-10-10 08:19:16 +02:00
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
"closed",
|
|
|
|
);
|
2021-10-10 08:19:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function createAccount() {
|
2023-01-13 05:40:33 +01:00
|
|
|
popup(
|
|
|
|
defineAsyncComponent(() => import("@/components/MkSignupDialog.vue")),
|
|
|
|
{},
|
|
|
|
{
|
|
|
|
done: (res) => {
|
|
|
|
addAccount(res.id, res.i);
|
|
|
|
switchAccountWithToken(res.i);
|
|
|
|
},
|
2021-10-10 08:19:16 +02:00
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
"closed",
|
|
|
|
);
|
2021-10-10 08:19:16 +02:00
|
|
|
}
|
|
|
|
|
2024-02-12 16:40:46 +01:00
|
|
|
async function switchAccount(account: entities.UserDetailed) {
|
2021-10-10 08:19:16 +02:00
|
|
|
const storedAccounts = await getAccounts();
|
2023-01-13 05:40:33 +01:00
|
|
|
const token = storedAccounts.find((x) => x.id === account.id).token;
|
2021-10-10 08:19:16 +02:00
|
|
|
switchAccountWithToken(token);
|
|
|
|
}
|
|
|
|
|
|
|
|
function switchAccountWithToken(token: string) {
|
2024-03-07 03:13:21 +01:00
|
|
|
signIn(token);
|
2021-10-10 08:19:16 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const storedAccounts = await getAccounts().then((accounts) =>
|
2024-03-07 03:06:45 +01:00
|
|
|
accounts.filter((x) => x.id !== me.id),
|
2023-01-13 05:40:33 +01:00
|
|
|
);
|
|
|
|
const accountsPromise = api("users/show", {
|
|
|
|
userIds: storedAccounts.map((x) => x.id),
|
|
|
|
});
|
2021-10-10 08:19:16 +02:00
|
|
|
|
2024-02-12 16:40:46 +01:00
|
|
|
function createItem(account: entities.UserDetailed) {
|
2022-01-21 12:17:31 +01:00
|
|
|
return {
|
2023-01-13 05:40:33 +01:00
|
|
|
type: "user",
|
2022-01-21 12:17:31 +01:00
|
|
|
user: account,
|
|
|
|
active: opts.active != null ? opts.active === account.id : false,
|
|
|
|
action: () => {
|
|
|
|
if (opts.onChoose) {
|
|
|
|
opts.onChoose(account);
|
|
|
|
} else {
|
|
|
|
switchAccount(account);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const accountItemPromises = storedAccounts.map(
|
|
|
|
(a) =>
|
|
|
|
new Promise((res) => {
|
|
|
|
accountsPromise.then((accounts) => {
|
|
|
|
const account = accounts.find((x) => x.id === a.id);
|
|
|
|
if (account == null) return res(null);
|
|
|
|
res(createItem(account));
|
|
|
|
});
|
|
|
|
}),
|
|
|
|
);
|
2021-10-10 08:19:16 +02:00
|
|
|
|
2022-01-21 12:17:31 +01:00
|
|
|
if (opts.withExtraOperation) {
|
2023-01-13 05:40:33 +01:00
|
|
|
popupMenu(
|
|
|
|
[
|
|
|
|
...[
|
2024-02-15 05:35:26 +01:00
|
|
|
...(isMobile ?? false
|
|
|
|
? [
|
|
|
|
{
|
|
|
|
type: "parent",
|
|
|
|
icon: "ph-plus ph-bold ph-lg",
|
|
|
|
text: i18n.ts.addAccount,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
text: i18n.ts.existingAccount,
|
|
|
|
action: () => {
|
|
|
|
showSigninDialog();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
text: i18n.ts.createAccount,
|
|
|
|
action: () => {
|
|
|
|
createAccount();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
]
|
|
|
|
: [
|
|
|
|
{
|
|
|
|
type: "link",
|
|
|
|
text: i18n.ts.profile,
|
2024-03-07 03:06:45 +01:00
|
|
|
to: `/@${me.username}`,
|
|
|
|
avatar: me,
|
2024-02-15 05:35:26 +01:00
|
|
|
},
|
|
|
|
null,
|
|
|
|
]),
|
2024-03-07 03:06:45 +01:00
|
|
|
...(opts.includeCurrentAccount ? [createItem(me)] : []),
|
2023-01-13 05:40:33 +01:00
|
|
|
...accountItemPromises,
|
2024-02-15 05:35:26 +01:00
|
|
|
...(isMobile ?? false
|
|
|
|
? [
|
|
|
|
null,
|
|
|
|
{
|
|
|
|
type: "link",
|
|
|
|
text: i18n.ts.profile,
|
2024-03-07 03:06:45 +01:00
|
|
|
to: `/@${me.username}`,
|
|
|
|
avatar: me,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
2024-02-15 05:35:26 +01:00
|
|
|
]
|
|
|
|
: [
|
|
|
|
{
|
|
|
|
type: "parent",
|
|
|
|
icon: "ph-plus ph-bold ph-lg",
|
|
|
|
text: i18n.ts.addAccount,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
text: i18n.ts.existingAccount,
|
|
|
|
action: () => {
|
|
|
|
showSigninDialog();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
text: i18n.ts.createAccount,
|
|
|
|
action: () => {
|
|
|
|
createAccount();
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
2024-02-15 05:35:26 +01:00
|
|
|
]),
|
2023-01-13 05:40:33 +01:00
|
|
|
],
|
|
|
|
],
|
|
|
|
ev.currentTarget ?? ev.target,
|
|
|
|
{
|
|
|
|
align: "left",
|
|
|
|
},
|
|
|
|
);
|
2022-01-21 12:17:31 +01:00
|
|
|
} else {
|
2023-01-13 05:40:33 +01:00
|
|
|
popupMenu(
|
|
|
|
[
|
2024-03-07 03:06:45 +01:00
|
|
|
...(opts.includeCurrentAccount ? [createItem(me)] : []),
|
2023-01-13 05:40:33 +01:00
|
|
|
...accountItemPromises,
|
|
|
|
],
|
|
|
|
ev.currentTarget ?? ev.target,
|
|
|
|
{
|
|
|
|
align: "left",
|
|
|
|
},
|
|
|
|
);
|
2022-01-21 12:17:31 +01:00
|
|
|
}
|
2021-10-10 08:19:16 +02:00
|
|
|
}
|