/** * Client entry point */ import '@client/style.scss'; import { set } from '@client/scripts/idb-proxy'; // TODO: そのうち消す if (localStorage.getItem('vuex') != null) { const vuex = JSON.parse(localStorage.getItem('vuex')); localStorage.setItem('account', JSON.stringify({ ...vuex.i, token: localStorage.getItem('i') })); set('accounts', JSON.parse(JSON.stringify(vuex.device.accounts))); localStorage.setItem('miux:themes', JSON.stringify(vuex.device.themes)); if (vuex.device.userData) { for (const [k, v] of Object.entries(vuex.device.userData)) { localStorage.setItem('pizzax::base::' + k, JSON.stringify({ widgets: v.widgets })); if (v.deck) { localStorage.setItem('pizzax::deck::' + k, JSON.stringify({ columns: v.deck.columns, layout: v.deck.layout, })); } } } localStorage.setItem('vuex-old', JSON.stringify(vuex)); localStorage.removeItem('vuex'); localStorage.removeItem('i'); localStorage.removeItem('locale'); location.reload(); } if (localStorage.getItem('accounts') != null) { set('accounts', JSON.parse(localStorage.getItem('accounts'))); localStorage.removeItem('accounts'); } import * as Sentry from '@sentry/browser'; import { Integrations } from '@sentry/tracing'; import { computed, createApp, watch } from 'vue'; import widgets from '@client/widgets'; import directives from '@client/directives'; import components from '@client/components'; import { version, ui, lang, host } from '@client/config'; import { router } from '@client/router'; import { applyTheme } from '@client/scripts/theme'; import { isDeviceDarkmode } from '@client/scripts/is-device-darkmode'; import { i18n } from '@client/i18n'; import { stream, dialog, post } from '@client/os'; import * as sound from '@client/scripts/sound'; import { $i, refreshAccount, login, updateAccount, signout } from '@client/account'; import { defaultStore, ColdDeviceStorage } from '@client/store'; import { fetchInstance, instance } from '@client/instance'; import { makeHotkey } from '@client/scripts/hotkey'; import { search } from '@client/scripts/search'; import { isMobile } from '@client/scripts/is-mobile'; import { initializeSw } from '@client/scripts/initialize-sw'; import { reload, reloadChannel } from '@client/scripts/unison-reload'; import { reactionPicker } from '@client/scripts/reaction-picker'; import { deleteLoginId } from '@client/scripts/login-id'; import { getAccountFromId } from '@client/scripts/get-account-from-id'; console.info(`Misskey v${version}`); // boot.jsのやつを解除 window.onerror = null; window.onunhandledrejection = null; // 後方互換性のため。 // TODO: そのうち消す if ((typeof ColdDeviceStorage.get('lightTheme') === 'string') || (typeof ColdDeviceStorage.get('darkTheme') === 'string')) { ColdDeviceStorage.set('lightTheme', require('@client/themes/l-light.json5')); ColdDeviceStorage.set('darkTheme', require('@client/themes/d-dark.json5')); } const link = document.createElement('link'); link.rel = 'stylesheet'; link.href = 'https://use.fontawesome.com/releases/v5.15.3/css/all.css'; document.head.appendChild(link); // TODOここまで if (_DEV_) { console.warn('Development mode!!!'); (window as any).$i = $i; (window as any).$store = defaultStore; window.addEventListener('error', event => { console.error(event); /* dialog({ type: 'error', title: 'DEV: Unhandled error', text: event.message }); */ }); window.addEventListener('unhandledrejection', event => { console.error(event); /* dialog({ type: 'error', title: 'DEV: Unhandled promise rejection', text: event.reason }); */ }); } if (defaultStore.state.reportError && !_DEV_) { Sentry.init({ dsn: 'https://fd273254a07a4b61857607a9ea05d629@o501808.ingest.sentry.io/5583438', tracesSampleRate: 1.0, }); Sentry.setTag('misskey_version', version); Sentry.setTag('ui', ui); Sentry.setTag('lang', lang); Sentry.setTag('host', host); } // タッチデバイスでCSSの:hoverを機能させる document.addEventListener('touchend', () => {}, { passive: true }); // 一斉リロード reloadChannel.addEventListener('message', shouldRedirectToRoot => reload(shouldRedirectToRoot)); //#region SEE: https://css-tricks.com/the-trick-to-viewport-units-on-mobile/ // TODO: いつの日にか消したい const vh = window.innerHeight * 0.01; document.documentElement.style.setProperty('--vh', `${vh}px`); window.addEventListener('resize', () => { const vh = window.innerHeight * 0.01; document.documentElement.style.setProperty('--vh', `${vh}px`); }); //#endregion // Get the
element const head = document.getElementsByTagName('head')[0]; // If mobile, insert the viewport meta tag if (isMobile || window.innerWidth <= 1024) { const viewport = document.getElementsByName('viewport').item(0); viewport.setAttribute('content', `${viewport.getAttribute('content')},minimum-scale=1,maximum-scale=1,user-scalable=no`); head.appendChild(viewport); } //#region Set lang attr const html = document.documentElement; html.setAttribute('lang', lang); //#endregion //#region loginId const params = new URLSearchParams(location.href); const loginId = params.get('loginId'); if (loginId) { const target = deleteLoginId(location.toString()); if (!$i || $i.id !== loginId) { const account = await getAccountFromId(loginId); if (account) { login(account.token, target); } } history.replaceState({ misskey: 'loginId' }, '', target); } //#endregion //#region Fetch user if ($i && $i.token) { if (_DEV_) { console.log('account cache found. refreshing...'); } refreshAccount(); } else { if (_DEV_) { console.log('no account cache found.'); } // 連携ログインの場合用にCookieを参照する const i = (document.cookie.match(/igi=(\w+)/) || [null, null])[1]; if (i != null && i !== 'null') { if (_DEV_) { console.log('signing...'); } try { document.body.innerHTML = '