2023-07-27 14:31:52 +09:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: syuilo and other misskey contributors
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2023-05-15 19:08:46 +09:00
|
|
|
import { computed, createApp, watch, markRaw, version as vueVersion, defineAsyncComponent, App } from 'vue';
|
|
|
|
import { compareVersions } from 'compare-versions';
|
2023-09-19 16:37:43 +09:00
|
|
|
import widgets from '@/widgets/index.js';
|
|
|
|
import directives from '@/directives/index.js';
|
|
|
|
import components from '@/components/index.js';
|
2023-12-14 20:21:57 +09:00
|
|
|
import { version, ui, lang, updateLocale, locale } from '@/config.js';
|
2023-09-19 16:37:43 +09:00
|
|
|
import { applyTheme } from '@/scripts/theme.js';
|
|
|
|
import { isDeviceDarkmode } from '@/scripts/is-device-darkmode.js';
|
|
|
|
import { i18n, updateI18n } from '@/i18n.js';
|
|
|
|
import { $i, refreshAccount, login, updateAccount, signout } from '@/account.js';
|
|
|
|
import { defaultStore, ColdDeviceStorage } from '@/store.js';
|
|
|
|
import { fetchInstance, instance } from '@/instance.js';
|
|
|
|
import { deviceKind } from '@/scripts/device-kind.js';
|
|
|
|
import { reloadChannel } from '@/scripts/unison-reload.js';
|
|
|
|
import { getUrlWithoutLoginId } from '@/scripts/login-id.js';
|
|
|
|
import { getAccountFromId } from '@/scripts/get-account-from-id.js';
|
|
|
|
import { deckStore } from '@/ui/deck/deck-store.js';
|
|
|
|
import { miLocalStorage } from '@/local-storage.js';
|
|
|
|
import { fetchCustomEmojis } from '@/custom-emojis.js';
|
2023-05-15 19:08:46 +09:00
|
|
|
|
|
|
|
export async function common(createVue: () => App<Element>) {
|
|
|
|
console.info(`Misskey v${version}`);
|
|
|
|
|
|
|
|
if (_DEV_) {
|
|
|
|
console.warn('Development mode!!!');
|
|
|
|
|
|
|
|
console.info(`vue ${vueVersion}`);
|
|
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
(window as any).$i = $i;
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
(window as any).$store = defaultStore;
|
|
|
|
|
|
|
|
window.addEventListener('error', event => {
|
|
|
|
console.error(event);
|
|
|
|
/*
|
|
|
|
alert({
|
|
|
|
type: 'error',
|
|
|
|
title: 'DEV: Unhandled error',
|
|
|
|
text: event.message
|
|
|
|
});
|
|
|
|
*/
|
|
|
|
});
|
|
|
|
|
|
|
|
window.addEventListener('unhandledrejection', event => {
|
|
|
|
console.error(event);
|
|
|
|
/*
|
|
|
|
alert({
|
|
|
|
type: 'error',
|
|
|
|
title: 'DEV: Unhandled promise rejection',
|
|
|
|
text: event.reason
|
|
|
|
});
|
|
|
|
*/
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const splash = document.getElementById('splash');
|
|
|
|
// 念のためnullチェック(HTMLが古い場合があるため(そのうち消す))
|
|
|
|
if (splash) splash.addEventListener('transitionend', () => {
|
|
|
|
splash.remove();
|
|
|
|
});
|
|
|
|
|
|
|
|
let isClientUpdated = false;
|
|
|
|
|
|
|
|
//#region クライアントが更新されたかチェック
|
|
|
|
const lastVersion = miLocalStorage.getItem('lastVersion');
|
|
|
|
if (lastVersion !== version) {
|
|
|
|
miLocalStorage.setItem('lastVersion', version);
|
|
|
|
|
|
|
|
// テーマリビルドするため
|
|
|
|
miLocalStorage.removeItem('theme');
|
|
|
|
|
|
|
|
try { // 変なバージョン文字列来るとcompareVersionsでエラーになるため
|
|
|
|
if (lastVersion != null && compareVersions(version, lastVersion) === 1) {
|
|
|
|
isClientUpdated = true;
|
|
|
|
}
|
|
|
|
} catch (err) { /* empty */ }
|
|
|
|
}
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
//#region Detect language & fetch translations
|
|
|
|
const localeVersion = miLocalStorage.getItem('localeVersion');
|
2023-12-14 20:21:57 +09:00
|
|
|
const localeOutdated = (localeVersion == null || localeVersion !== version || locale == null);
|
2023-05-15 19:08:46 +09:00
|
|
|
if (localeOutdated) {
|
|
|
|
const res = await window.fetch(`/assets/locales/${lang}.${version}.json`);
|
|
|
|
if (res.status === 200) {
|
|
|
|
const newLocale = await res.text();
|
|
|
|
const parsedNewLocale = JSON.parse(newLocale);
|
|
|
|
miLocalStorage.setItem('locale', newLocale);
|
|
|
|
miLocalStorage.setItem('localeVersion', version);
|
|
|
|
updateLocale(parsedNewLocale);
|
|
|
|
updateI18n(parsedNewLocale);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
// タッチデバイスでCSSの:hoverを機能させる
|
|
|
|
document.addEventListener('touchend', () => {}, { passive: true });
|
|
|
|
|
|
|
|
// 一斉リロード
|
|
|
|
reloadChannel.addEventListener('message', path => {
|
|
|
|
if (path !== null) location.href = path;
|
|
|
|
else location.reload();
|
|
|
|
});
|
|
|
|
|
|
|
|
// If mobile, insert the viewport meta tag
|
|
|
|
if (['smartphone', 'tablet'].includes(deviceKind)) {
|
|
|
|
const viewport = document.getElementsByName('viewport').item(0);
|
|
|
|
viewport.setAttribute('content',
|
|
|
|
`${viewport.getAttribute('content')}, minimum-scale=1, maximum-scale=1, user-scalable=no, viewport-fit=cover`);
|
|
|
|
}
|
|
|
|
|
|
|
|
//#region Set lang attr
|
|
|
|
const html = document.documentElement;
|
|
|
|
html.setAttribute('lang', lang);
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
await defaultStore.ready;
|
|
|
|
await deckStore.ready;
|
|
|
|
|
|
|
|
const fetchInstanceMetaPromise = fetchInstance();
|
|
|
|
|
|
|
|
fetchInstanceMetaPromise.then(() => {
|
|
|
|
miLocalStorage.setItem('v', instance.version);
|
|
|
|
});
|
|
|
|
|
|
|
|
//#region loginId
|
|
|
|
const params = new URLSearchParams(location.search);
|
|
|
|
const loginId = params.get('loginId');
|
|
|
|
|
|
|
|
if (loginId) {
|
|
|
|
const target = getUrlWithoutLoginId(location.href);
|
|
|
|
|
|
|
|
if (!$i || $i.id !== loginId) {
|
|
|
|
const account = await getAccountFromId(loginId);
|
|
|
|
if (account) {
|
|
|
|
await login(account.token, target);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
history.replaceState({ misskey: 'loginId' }, '', target);
|
|
|
|
}
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
// NOTE: この処理は必ずクライアント更新チェック処理より後に来ること(テーマ再構築のため)
|
|
|
|
watch(defaultStore.reactiveState.darkMode, (darkMode) => {
|
|
|
|
applyTheme(darkMode ? ColdDeviceStorage.get('darkTheme') : ColdDeviceStorage.get('lightTheme'));
|
|
|
|
}, { immediate: miLocalStorage.getItem('theme') == null });
|
|
|
|
|
|
|
|
const darkTheme = computed(ColdDeviceStorage.makeGetterSetter('darkTheme'));
|
|
|
|
const lightTheme = computed(ColdDeviceStorage.makeGetterSetter('lightTheme'));
|
|
|
|
|
|
|
|
watch(darkTheme, (theme) => {
|
|
|
|
if (defaultStore.state.darkMode) {
|
|
|
|
applyTheme(theme);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
watch(lightTheme, (theme) => {
|
|
|
|
if (!defaultStore.state.darkMode) {
|
|
|
|
applyTheme(theme);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
//#region Sync dark mode
|
|
|
|
if (ColdDeviceStorage.get('syncDeviceDarkMode')) {
|
|
|
|
defaultStore.set('darkMode', isDeviceDarkmode());
|
|
|
|
}
|
|
|
|
|
2023-10-23 15:32:10 +09:00
|
|
|
window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', (mql) => {
|
2023-05-15 19:08:46 +09:00
|
|
|
if (ColdDeviceStorage.get('syncDeviceDarkMode')) {
|
|
|
|
defaultStore.set('darkMode', mql.matches);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
fetchInstanceMetaPromise.then(() => {
|
|
|
|
if (defaultStore.state.themeInitial) {
|
2023-05-30 11:51:29 +09:00
|
|
|
if (instance.defaultLightTheme != null) ColdDeviceStorage.set('lightTheme', JSON.parse(instance.defaultLightTheme));
|
|
|
|
if (instance.defaultDarkTheme != null) ColdDeviceStorage.set('darkTheme', JSON.parse(instance.defaultDarkTheme));
|
2023-05-15 19:08:46 +09:00
|
|
|
defaultStore.set('themeInitial', false);
|
2023-12-08 08:22:08 +09:00
|
|
|
} else {
|
|
|
|
if (defaultStore.state.darkMode) {
|
|
|
|
applyTheme(darkTheme.value);
|
|
|
|
} else {
|
|
|
|
applyTheme(lightTheme.value);
|
|
|
|
}
|
2023-05-15 19:08:46 +09:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
watch(defaultStore.reactiveState.useBlurEffectForModal, v => {
|
|
|
|
document.documentElement.style.setProperty('--modalBgFilter', v ? 'blur(4px)' : 'none');
|
|
|
|
}, { immediate: true });
|
|
|
|
|
|
|
|
watch(defaultStore.reactiveState.useBlurEffect, v => {
|
|
|
|
if (v) {
|
|
|
|
document.documentElement.style.removeProperty('--blur');
|
|
|
|
} else {
|
|
|
|
document.documentElement.style.setProperty('--blur', 'none');
|
|
|
|
}
|
|
|
|
}, { immediate: true });
|
|
|
|
|
2023-11-29 10:29:24 +09:00
|
|
|
// Keep screen on
|
|
|
|
const onVisibilityChange = () => document.addEventListener('visibilitychange', () => {
|
|
|
|
if (document.visibilityState === 'visible') {
|
|
|
|
navigator.wakeLock.request('screen');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (defaultStore.state.keepScreenOn && 'wakeLock' in navigator) {
|
|
|
|
navigator.wakeLock.request('screen')
|
|
|
|
.then(onVisibilityChange)
|
2023-11-26 23:35:53 +09:00
|
|
|
.catch(() => {
|
2023-11-29 10:29:24 +09:00
|
|
|
// On WebKit-based browsers, user activation is required to send wake lock request
|
|
|
|
// https://webkit.org/blog/13862/the-user-activation-api/
|
|
|
|
document.addEventListener(
|
|
|
|
'click',
|
|
|
|
() => navigator.wakeLock.request('screen').then(onVisibilityChange),
|
|
|
|
{ once: true },
|
|
|
|
);
|
2023-09-20 10:02:39 +09:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-05-15 19:08:46 +09:00
|
|
|
//#region Fetch user
|
|
|
|
if ($i && $i.token) {
|
|
|
|
if (_DEV_) {
|
|
|
|
console.log('account cache found. refreshing...');
|
|
|
|
}
|
|
|
|
|
|
|
|
refreshAccount();
|
|
|
|
}
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
try {
|
|
|
|
await fetchCustomEmojis();
|
|
|
|
} catch (err) { /* empty */ }
|
|
|
|
|
|
|
|
const app = createVue();
|
|
|
|
|
|
|
|
if (_DEV_) {
|
|
|
|
app.config.performance = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
widgets(app);
|
|
|
|
directives(app);
|
|
|
|
components(app);
|
|
|
|
|
|
|
|
// https://github.com/misskey-dev/misskey/pull/8575#issuecomment-1114239210
|
|
|
|
// なぜか2回実行されることがあるため、mountするdivを1つに制限する
|
|
|
|
const rootEl = ((): HTMLElement => {
|
|
|
|
const MISSKEY_MOUNT_DIV_ID = 'misskey_app';
|
|
|
|
|
|
|
|
const currentRoot = document.getElementById(MISSKEY_MOUNT_DIV_ID);
|
|
|
|
|
|
|
|
if (currentRoot) {
|
|
|
|
console.warn('multiple import detected');
|
|
|
|
return currentRoot;
|
|
|
|
}
|
|
|
|
|
|
|
|
const root = document.createElement('div');
|
|
|
|
root.id = MISSKEY_MOUNT_DIV_ID;
|
|
|
|
document.body.appendChild(root);
|
|
|
|
return root;
|
|
|
|
})();
|
|
|
|
|
|
|
|
app.mount(rootEl);
|
|
|
|
|
|
|
|
// boot.jsのやつを解除
|
|
|
|
window.onerror = null;
|
|
|
|
window.onunhandledrejection = null;
|
|
|
|
|
|
|
|
removeSplash();
|
|
|
|
|
|
|
|
return {
|
|
|
|
isClientUpdated,
|
|
|
|
app,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function removeSplash() {
|
|
|
|
const splash = document.getElementById('splash');
|
|
|
|
if (splash) {
|
|
|
|
splash.style.opacity = '0';
|
|
|
|
splash.style.pointerEvents = 'none';
|
|
|
|
}
|
|
|
|
}
|