2023-07-27 07:31:52 +02:00
|
|
|
|
/*
|
2024-02-13 16:59:27 +01:00
|
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-07-27 07:31:52 +02:00
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
*/
|
|
|
|
|
|
2023-03-09 06:27:16 +01:00
|
|
|
|
import { defineAsyncComponent, reactive, ref } from 'vue';
|
2023-09-04 06:33:38 +02:00
|
|
|
|
import * as Misskey from 'misskey-js';
|
2024-10-21 12:59:20 +02:00
|
|
|
|
import { apiUrl } from '@@/js/config.js';
|
|
|
|
|
import type { MenuItem, MenuButton } from '@/types/menu.js';
|
2023-09-19 09:37:43 +02:00
|
|
|
|
import { showSuspendedDialog } from '@/scripts/show-suspended-dialog.js';
|
|
|
|
|
import { i18n } from '@/i18n.js';
|
|
|
|
|
import { miLocalStorage } from '@/local-storage.js';
|
|
|
|
|
import { del, get, set } from '@/scripts/idb-proxy.js';
|
2024-01-04 10:32:46 +01:00
|
|
|
|
import { waiting, popup, popupMenu, success, alert } from '@/os.js';
|
|
|
|
|
import { misskeyApi } from '@/scripts/misskey-api.js';
|
2023-09-19 09:37:43 +02:00
|
|
|
|
import { unisonReload, reloadChannel } from '@/scripts/unison-reload.js';
|
2020-12-19 02:55:52 +01:00
|
|
|
|
|
|
|
|
|
// TODO: 他のタブと永続化されたstateを同期
|
|
|
|
|
|
2023-12-14 04:26:02 +01:00
|
|
|
|
type Account = Misskey.entities.MeDetailed & { token: string };
|
2020-12-19 02:55:52 +01:00
|
|
|
|
|
2023-01-07 02:13:02 +01:00
|
|
|
|
const accountData = miLocalStorage.getItem('account');
|
2020-12-19 02:55:52 +01:00
|
|
|
|
|
|
|
|
|
// TODO: 外部からはreadonlyに
|
2022-05-26 15:53:09 +02:00
|
|
|
|
export const $i = accountData ? reactive(JSON.parse(accountData) as Account) : null;
|
2020-12-19 02:55:52 +01:00
|
|
|
|
|
2024-01-04 07:20:23 +01:00
|
|
|
|
export const iAmModerator = $i != null && ($i.isAdmin === true || $i.isModerator === true);
|
2022-07-02 08:12:11 +02:00
|
|
|
|
export const iAmAdmin = $i != null && $i.isAdmin;
|
2022-01-18 13:30:17 +01:00
|
|
|
|
|
2024-01-04 07:30:40 +01:00
|
|
|
|
export function signinRequired() {
|
|
|
|
|
if ($i == null) throw new Error('signin required');
|
|
|
|
|
return $i;
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-21 05:14:55 +01:00
|
|
|
|
export let notesCount = $i == null ? 0 : $i.notesCount;
|
|
|
|
|
export function incNotesCount() {
|
|
|
|
|
notesCount++;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-20 12:38:16 +02:00
|
|
|
|
export async function signout() {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
if (!$i) return;
|
|
|
|
|
|
2021-08-20 12:38:16 +02:00
|
|
|
|
waiting();
|
2023-01-07 02:13:02 +01:00
|
|
|
|
miLocalStorage.removeItem('account');
|
2022-06-20 10:38:49 +02:00
|
|
|
|
await removeAccount($i.id);
|
2022-06-24 14:16:05 +02:00
|
|
|
|
const accounts = await getAccounts();
|
|
|
|
|
|
2021-09-04 11:09:53 +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) {
|
2022-12-03 11:42:05 +01:00
|
|
|
|
await window.fetch(`${apiUrl}/sw/unregister`, {
|
2021-09-04 11:09:53 +02:00
|
|
|
|
method: 'POST',
|
|
|
|
|
body: JSON.stringify({
|
|
|
|
|
i: $i.token,
|
|
|
|
|
endpoint: push.endpoint,
|
|
|
|
|
}),
|
2022-12-03 11:42:05 +01:00
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
},
|
2021-09-04 11:09:53 +02:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (accounts.length === 0) {
|
|
|
|
|
await navigator.serviceWorker.getRegistrations()
|
|
|
|
|
.then(registrations => {
|
|
|
|
|
return Promise.all(registrations.map(registration => registration.unregister()));
|
|
|
|
|
});
|
|
|
|
|
}
|
2022-05-26 15:53:09 +02:00
|
|
|
|
} catch (err) {}
|
2021-08-20 12:38:16 +02:00
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
|
|
if (accounts.length > 0) login(accounts[0].token);
|
2021-11-04 16:09:13 +01:00
|
|
|
|
else unisonReload('/');
|
2020-12-19 02:55:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-08-20 12:38:16 +02:00
|
|
|
|
export async function getAccounts(): Promise<{ id: Account['id'], token: Account['token'] }[]> {
|
|
|
|
|
return (await get('accounts')) || [];
|
2020-12-19 02:55:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-08-20 12:38:16 +02:00
|
|
|
|
export async function addAccount(id: Account['id'], token: Account['token']) {
|
|
|
|
|
const accounts = await getAccounts();
|
2020-12-19 02:55:52 +01:00
|
|
|
|
if (!accounts.some(x => x.id === id)) {
|
2021-08-20 12:38:16 +02:00
|
|
|
|
await set('accounts', accounts.concat([{ id, token }]));
|
2020-12-19 02:55:52 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-09 06:27:16 +01:00
|
|
|
|
export async function removeAccount(idOrToken: Account['id']) {
|
2022-06-20 10:38:49 +02:00
|
|
|
|
const accounts = await getAccounts();
|
2023-03-09 06:27:16 +01:00
|
|
|
|
const i = accounts.findIndex(x => x.id === idOrToken || x.token === idOrToken);
|
|
|
|
|
if (i !== -1) accounts.splice(i, 1);
|
2022-06-20 10:38:49 +02:00
|
|
|
|
|
2023-03-09 06:27:16 +01:00
|
|
|
|
if (accounts.length > 0) {
|
|
|
|
|
await set('accounts', accounts);
|
|
|
|
|
} else {
|
|
|
|
|
await del('accounts');
|
|
|
|
|
}
|
2022-06-20 10:38:49 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-03-09 06:27:16 +01:00
|
|
|
|
function fetchAccount(token: string, id?: string, forceShowDialog?: boolean): Promise<Account> {
|
2020-12-19 02:55:52 +01:00
|
|
|
|
return new Promise((done, fail) => {
|
2022-12-03 11:42:05 +01:00
|
|
|
|
window.fetch(`${apiUrl}/i`, {
|
2020-12-19 02:55:52 +01:00
|
|
|
|
method: 'POST',
|
|
|
|
|
body: JSON.stringify({
|
2022-06-20 10:38:49 +02:00
|
|
|
|
i: token,
|
|
|
|
|
}),
|
2022-12-03 11:42:05 +01:00
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
},
|
2020-12-19 02:55:52 +01:00
|
|
|
|
})
|
2023-05-18 13:17:32 +02:00
|
|
|
|
.then(res => new Promise<Account | { error: Record<string, any> }>((done2, fail2) => {
|
|
|
|
|
if (res.status >= 500 && res.status < 600) {
|
2023-08-13 13:12:29 +02:00
|
|
|
|
// サーバーエラー(5xx)の場合をrejectとする
|
|
|
|
|
// (認証エラーなど4xxはresolve)
|
2023-05-18 13:17:32 +02:00
|
|
|
|
return fail2(res);
|
|
|
|
|
}
|
|
|
|
|
res.json().then(done2, fail2);
|
|
|
|
|
}))
|
|
|
|
|
.then(async res => {
|
2024-06-20 09:24:10 +02:00
|
|
|
|
if ('error' in res) {
|
2023-05-18 13:17:32 +02:00
|
|
|
|
if (res.error.id === 'a8c724b3-6e9c-4b46-b1a8-bc3ed6258370') {
|
2023-07-31 12:14:20 +02:00
|
|
|
|
// SUSPENDED
|
2023-05-18 13:17:32 +02:00
|
|
|
|
if (forceShowDialog || $i && (token === $i.token || id === $i.id)) {
|
|
|
|
|
await showSuspendedDialog();
|
|
|
|
|
}
|
|
|
|
|
} else if (res.error.id === 'e5b3b9f0-2b8f-4b9f-9c1f-8c5c1b2e1b1a') {
|
2023-07-31 12:14:20 +02:00
|
|
|
|
// USER_IS_DELETED
|
|
|
|
|
// アカウントが削除されている
|
2023-05-18 13:17:32 +02:00
|
|
|
|
if (forceShowDialog || $i && (token === $i.token || id === $i.id)) {
|
|
|
|
|
await alert({
|
|
|
|
|
type: 'error',
|
|
|
|
|
title: i18n.ts.accountDeleted,
|
|
|
|
|
text: i18n.ts.accountDeletedDescription,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else if (res.error.id === 'b0a7f5f8-dc2f-4171-b91f-de88ad238e14') {
|
2023-07-31 12:14:20 +02:00
|
|
|
|
// AUTHENTICATION_FAILED
|
|
|
|
|
// トークンが無効化されていたりアカウントが削除されたりしている
|
2023-05-18 13:17:32 +02:00
|
|
|
|
if (forceShowDialog || $i && (token === $i.token || id === $i.id)) {
|
|
|
|
|
await alert({
|
|
|
|
|
type: 'error',
|
|
|
|
|
title: i18n.ts.tokenRevoked,
|
|
|
|
|
text: i18n.ts.tokenRevokedDescription,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
await alert({
|
2022-12-03 11:42:05 +01:00
|
|
|
|
type: 'error',
|
2023-05-18 13:17:32 +02:00
|
|
|
|
title: i18n.ts.failedToFetchAccountInformation,
|
|
|
|
|
text: JSON.stringify(res.error),
|
2022-12-03 11:42:05 +01:00
|
|
|
|
});
|
|
|
|
|
}
|
2023-05-18 13:17:32 +02:00
|
|
|
|
|
|
|
|
|
// rejectかつ理由がtrueの場合、削除対象であることを示す
|
|
|
|
|
fail(true);
|
2021-09-18 19:23:12 +02:00
|
|
|
|
} else {
|
2023-05-18 13:17:32 +02:00
|
|
|
|
(res as Account).token = token;
|
|
|
|
|
done(res as Account);
|
2021-09-18 19:23:12 +02:00
|
|
|
|
}
|
2023-05-18 13:17:32 +02:00
|
|
|
|
})
|
|
|
|
|
.catch(fail);
|
2020-12-19 02:55:52 +01:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-21 12:59:20 +02:00
|
|
|
|
export function updateAccount(accountData: Account) {
|
|
|
|
|
if (!$i) return;
|
|
|
|
|
for (const key of Object.keys($i)) {
|
|
|
|
|
delete $i[key];
|
|
|
|
|
}
|
|
|
|
|
for (const [key, value] of Object.entries(accountData)) {
|
|
|
|
|
$i[key] = value;
|
|
|
|
|
}
|
|
|
|
|
miLocalStorage.setItem('account', JSON.stringify($i));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function updateAccountPartial(accountData: Partial<Account>) {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
if (!$i) return;
|
2022-05-26 15:53:09 +02:00
|
|
|
|
for (const [key, value] of Object.entries(accountData)) {
|
2020-12-19 02:55:52 +01:00
|
|
|
|
$i[key] = value;
|
|
|
|
|
}
|
2023-01-07 02:13:02 +01:00
|
|
|
|
miLocalStorage.setItem('account', JSON.stringify($i));
|
2020-12-19 02:55:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2023-03-09 06:27:16 +01:00
|
|
|
|
export async function refreshAccount() {
|
|
|
|
|
if (!$i) return;
|
|
|
|
|
return fetchAccount($i.token, $i.id)
|
|
|
|
|
.then(updateAccount, reason => {
|
|
|
|
|
if (reason === true) return signout();
|
|
|
|
|
return;
|
|
|
|
|
});
|
2020-12-19 02:55:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-08-20 12:38:16 +02:00
|
|
|
|
export async function login(token: Account['token'], redirect?: string) {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
const showing = ref(true);
|
2024-07-04 06:14:49 +02:00
|
|
|
|
const { dispose } = popup(defineAsyncComponent(() => import('@/components/MkWaitingDialog.vue')), {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
success: false,
|
|
|
|
|
showing: showing,
|
2024-07-04 06:14:49 +02:00
|
|
|
|
}, {
|
|
|
|
|
closed: () => dispose(),
|
|
|
|
|
});
|
2020-12-19 02:55:52 +01:00
|
|
|
|
if (_DEV_) console.log('logging as token ', token);
|
2023-03-09 06:27:16 +01:00
|
|
|
|
const me = await fetchAccount(token, undefined, true)
|
|
|
|
|
.catch(reason => {
|
|
|
|
|
if (reason === true) {
|
|
|
|
|
// 削除対象の場合
|
|
|
|
|
removeAccount(token);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
showing.value = false;
|
|
|
|
|
throw reason;
|
|
|
|
|
});
|
2023-01-07 02:13:02 +01:00
|
|
|
|
miLocalStorage.setItem('account', JSON.stringify(me));
|
2022-03-19 11:08:55 +01:00
|
|
|
|
document.cookie = `token=${token}; path=/; max-age=31536000`; // bull dashboardの認証とかで使う
|
2021-08-20 12:38:16 +02:00
|
|
|
|
await addAccount(me.id, token);
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
2022-01-21 12:17:31 +01:00
|
|
|
|
export async function openAccountMenu(opts: {
|
|
|
|
|
includeCurrentAccount?: boolean;
|
|
|
|
|
withExtraOperation: boolean;
|
2023-09-04 06:33:38 +02:00
|
|
|
|
active?: Misskey.entities.UserDetailed['id'];
|
|
|
|
|
onChoose?: (account: Misskey.entities.UserDetailed) => void;
|
2022-01-21 12:17:31 +01:00
|
|
|
|
}, ev: MouseEvent) {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
if (!$i) return;
|
|
|
|
|
|
2023-09-04 06:33:38 +02:00
|
|
|
|
async function switchAccount(account: Misskey.entities.UserDetailed) {
|
2021-10-10 08:19:16 +02:00
|
|
|
|
const storedAccounts = await getAccounts();
|
2023-03-09 06:27:16 +01:00
|
|
|
|
const found = storedAccounts.find(x => x.id === account.id);
|
|
|
|
|
if (found == null) return;
|
|
|
|
|
switchAccountWithToken(found.token);
|
2021-10-10 08:19:16 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function switchAccountWithToken(token: string) {
|
|
|
|
|
login(token);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const storedAccounts = await getAccounts().then(accounts => accounts.filter(x => x.id !== $i.id));
|
2024-01-04 10:32:46 +01:00
|
|
|
|
const accountsPromise = misskeyApi('users/show', { userIds: storedAccounts.map(x => x.id) });
|
2021-10-10 08:19:16 +02:00
|
|
|
|
|
2023-09-04 06:33:38 +02:00
|
|
|
|
function createItem(account: Misskey.entities.UserDetailed) {
|
2022-01-21 12:17:31 +01:00
|
|
|
|
return {
|
2023-03-09 06:27:16 +01:00
|
|
|
|
type: 'user' as const,
|
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-03-09 06:27:16 +01:00
|
|
|
|
const accountItemPromises = storedAccounts.map(a => new Promise<ReturnType<typeof createItem> | MenuButton>(res => {
|
2021-10-10 08:19:16 +02:00
|
|
|
|
accountsPromise.then(accounts => {
|
|
|
|
|
const account = accounts.find(x => x.id === a.id);
|
2023-03-09 06:27:16 +01:00
|
|
|
|
if (account == null) return res({
|
|
|
|
|
type: 'button' as const,
|
|
|
|
|
text: a.id,
|
|
|
|
|
action: () => {
|
|
|
|
|
switchAccountWithToken(a.token);
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
2022-01-21 12:17:31 +01:00
|
|
|
|
res(createItem(account));
|
2021-10-10 08:19:16 +02:00
|
|
|
|
});
|
|
|
|
|
}));
|
|
|
|
|
|
2024-09-23 14:50:30 +02:00
|
|
|
|
const menuItems: MenuItem[] = [];
|
|
|
|
|
|
2022-01-21 12:17:31 +01:00
|
|
|
|
if (opts.withExtraOperation) {
|
2024-09-23 14:50:30 +02:00
|
|
|
|
menuItems.push({
|
|
|
|
|
type: 'link',
|
2022-01-28 03:39:49 +01:00
|
|
|
|
text: i18n.ts.profile,
|
2024-09-23 14:50:30 +02:00
|
|
|
|
to: `/@${$i.username}`,
|
2022-01-21 12:17:31 +01:00
|
|
|
|
avatar: $i,
|
2024-09-23 14:50:30 +02:00
|
|
|
|
}, {
|
|
|
|
|
type: 'divider',
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (opts.includeCurrentAccount) {
|
|
|
|
|
menuItems.push(createItem($i));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
menuItems.push(...accountItemPromises);
|
|
|
|
|
|
|
|
|
|
menuItems.push({
|
|
|
|
|
type: 'parent',
|
2022-12-19 11:01:30 +01:00
|
|
|
|
icon: 'ti ti-plus',
|
2022-01-28 03:39:49 +01:00
|
|
|
|
text: i18n.ts.addAccount,
|
2022-07-17 16:18:05 +02:00
|
|
|
|
children: [{
|
|
|
|
|
text: i18n.ts.existingAccount,
|
2024-10-25 07:20:33 +02:00
|
|
|
|
action: () => {
|
|
|
|
|
getAccountWithSigninDialog().then(res => {
|
|
|
|
|
if (res != null) {
|
|
|
|
|
success();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
},
|
2022-07-17 16:18:05 +02:00
|
|
|
|
}, {
|
|
|
|
|
text: i18n.ts.createAccount,
|
2024-10-25 07:20:33 +02:00
|
|
|
|
action: () => {
|
|
|
|
|
getAccountWithSignupDialog().then(res => {
|
|
|
|
|
if (res != null) {
|
|
|
|
|
switchAccountWithToken(res.token);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
},
|
2022-07-17 16:18:05 +02:00
|
|
|
|
}],
|
2022-01-21 12:17:31 +01:00
|
|
|
|
}, {
|
2024-09-23 14:50:30 +02:00
|
|
|
|
type: 'link',
|
2022-12-19 11:01:30 +01:00
|
|
|
|
icon: 'ti ti-users',
|
2022-01-28 03:39:49 +01:00
|
|
|
|
text: i18n.ts.manageAccounts,
|
2022-06-20 10:38:49 +02:00
|
|
|
|
to: '/settings/accounts',
|
2022-01-21 12:17:31 +01:00
|
|
|
|
});
|
|
|
|
|
} else {
|
2024-09-23 14:50:30 +02:00
|
|
|
|
if (opts.includeCurrentAccount) {
|
|
|
|
|
menuItems.push(createItem($i));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
menuItems.push(...accountItemPromises);
|
2022-01-21 12:17:31 +01:00
|
|
|
|
}
|
2024-09-23 14:50:30 +02:00
|
|
|
|
|
|
|
|
|
popupMenu(menuItems, ev.currentTarget ?? ev.target, {
|
|
|
|
|
align: 'left',
|
|
|
|
|
});
|
2021-10-10 08:19:16 +02:00
|
|
|
|
}
|
2023-05-18 13:17:32 +02:00
|
|
|
|
|
2024-10-25 07:20:33 +02:00
|
|
|
|
export function getAccountWithSigninDialog(): Promise<{ id: string, token: string } | null> {
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
const { dispose } = popup(defineAsyncComponent(() => import('@/components/MkSigninDialog.vue')), {}, {
|
|
|
|
|
done: async (res: Misskey.entities.SigninFlowResponse & { finished: true }) => {
|
|
|
|
|
await addAccount(res.id, res.i);
|
|
|
|
|
resolve({ id: res.id, token: res.i });
|
|
|
|
|
},
|
|
|
|
|
cancelled: () => {
|
|
|
|
|
resolve(null);
|
|
|
|
|
},
|
|
|
|
|
closed: () => {
|
|
|
|
|
dispose();
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function getAccountWithSignupDialog(): Promise<{ id: string, token: string } | null> {
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
const { dispose } = popup(defineAsyncComponent(() => import('@/components/MkSignupDialog.vue')), {}, {
|
|
|
|
|
done: async (res: Misskey.entities.SignupResponse) => {
|
|
|
|
|
await addAccount(res.id, res.token);
|
|
|
|
|
resolve({ id: res.id, token: res.token });
|
|
|
|
|
},
|
|
|
|
|
cancelled: () => {
|
|
|
|
|
resolve(null);
|
|
|
|
|
},
|
|
|
|
|
closed: () => {
|
|
|
|
|
dispose();
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-18 13:17:32 +02:00
|
|
|
|
if (_DEV_) {
|
|
|
|
|
(window as any).$i = $i;
|
|
|
|
|
}
|