phanpy/src/app.jsx

266 lines
8.3 KiB
React
Raw Normal View History

2023-02-11 00:37:42 +00:00
import './app.css';
import {
useEffect,
useLayoutEffect,
useMemo,
useRef,
useState,
} from 'preact/hooks';
import { matchPath, Route, Routes, useLocation } from 'react-router-dom';
2023-04-25 12:41:08 +00:00
import 'swiped-events';
2022-12-10 09:14:48 +00:00
import { useSnapshot } from 'valtio';
2023-09-02 10:19:09 +00:00
import BackgroundService from './components/background-service';
2023-09-05 10:49:16 +00:00
import ComposeButton from './components/compose-button';
import { ICONS } from './components/icon';
2023-09-06 14:54:05 +00:00
import KeyboardShortcutsHelp from './components/keyboard-shortcuts-help';
2022-12-10 09:14:48 +00:00
import Loader from './components/loader';
import Modals from './components/modals';
2023-09-02 10:19:09 +00:00
import NotificationService from './components/notification-service';
2023-09-04 06:49:39 +00:00
import SearchCommand from './components/search-command';
2023-02-16 09:51:54 +00:00
import Shortcuts from './components/shortcuts';
import NotFound from './pages/404';
import AccountStatuses from './pages/account-statuses';
import Bookmarks from './pages/bookmarks';
import Favourites from './pages/favourites';
2023-02-11 08:48:47 +00:00
import FollowedHashtags from './pages/followed-hashtags';
2023-02-03 13:08:08 +00:00
import Following from './pages/following';
2023-02-18 12:48:24 +00:00
import Hashtag from './pages/hashtag';
2022-12-10 09:14:48 +00:00
import Home from './pages/home';
2023-09-02 10:19:09 +00:00
import HttpRoute from './pages/http-route';
2023-02-10 16:05:18 +00:00
import List from './pages/list';
import Lists from './pages/lists';
2022-12-10 09:14:48 +00:00
import Login from './pages/login';
2023-04-06 11:32:26 +00:00
import Mentions from './pages/mentions';
2022-12-10 09:14:48 +00:00
import Notifications from './pages/notifications';
import Public from './pages/public';
import Search from './pages/search';
2023-09-02 10:19:09 +00:00
import StatusRoute from './pages/status-route';
2023-04-05 17:14:38 +00:00
import Trending from './pages/trending';
2022-12-10 09:14:48 +00:00
import Welcome from './pages/welcome';
2023-02-09 15:59:57 +00:00
import {
api,
initAccount,
initClient,
initInstance,
initPreferences,
} from './utils/api';
2022-12-10 09:14:48 +00:00
import { getAccessToken } from './utils/auth';
import focusDeck from './utils/focus-deck';
2023-09-02 10:19:09 +00:00
import states, { initStates } from './utils/states';
2022-12-10 09:14:48 +00:00
import store from './utils/store';
2023-09-02 10:19:09 +00:00
import { getCurrentAccount } from './utils/store-utils';
import './utils/toast-alert';
2022-12-10 09:14:48 +00:00
window.__STATES__ = states;
2022-12-10 09:14:48 +00:00
2023-08-13 09:15:49 +00:00
// Preload icons
// There's probably a better way to do this
// Related: https://github.com/vitejs/vite/issues/10600
setTimeout(() => {
for (const icon in ICONS) {
if (Array.isArray(ICONS[icon])) {
ICONS[icon][0]?.();
} else {
ICONS[icon]?.();
}
}
}, 5000);
2022-12-31 17:46:08 +00:00
function App() {
2022-12-10 09:14:48 +00:00
const snapStates = useSnapshot(states);
const [isLoggedIn, setIsLoggedIn] = useState(false);
const [uiState, setUIState] = useState('loading');
2022-12-10 09:14:48 +00:00
useLayoutEffect(() => {
const theme = store.local.get('theme');
if (theme) {
document.documentElement.classList.add(`is-${theme}`);
document
.querySelector('meta[name="color-scheme"]')
.setAttribute('content', theme === 'auto' ? 'dark light' : theme);
2022-12-10 09:14:48 +00:00
}
2023-03-08 09:17:23 +00:00
const textSize = store.local.get('textSize');
if (textSize) {
document.documentElement.style.setProperty(
'--text-size',
`${textSize}px`,
);
}
2022-12-10 09:14:48 +00:00
}, []);
useEffect(() => {
const instanceURL = store.local.get('instanceURL');
2023-05-20 02:08:41 +00:00
const code = decodeURIComponent(
(window.location.search.match(/code=([^&]+)/) || [, ''])[1],
);
2022-12-10 09:14:48 +00:00
if (code) {
console.log({ code });
// Clear the code from the URL
2023-08-30 09:46:22 +00:00
window.history.replaceState({}, document.title, location.pathname || '/');
2022-12-10 09:14:48 +00:00
const clientID = store.session.get('clientID');
const clientSecret = store.session.get('clientSecret');
const vapidKey = store.session.get('vapidKey');
2022-12-10 09:14:48 +00:00
(async () => {
setUIState('loading');
const { access_token: accessToken } = await getAccessToken({
2022-12-10 09:14:48 +00:00
instanceURL,
client_id: clientID,
client_secret: clientSecret,
code,
});
const client = initClient({ instance: instanceURL, accessToken });
await Promise.allSettled([
initInstance(client, instanceURL),
initAccount(client, instanceURL, accessToken, vapidKey),
]);
initStates();
initPreferences(client);
2022-12-10 09:14:48 +00:00
setIsLoggedIn(true);
setUIState('default');
})();
} else {
window.__IGNORE_GET_ACCOUNT_ERROR__ = true;
const account = getCurrentAccount();
if (account) {
store.session.set('currentAccount', account.info.id);
const { client } = api({ account });
const { instance } = client;
// console.log('masto', masto);
initStates();
initPreferences(client);
setUIState('loading');
2023-02-07 16:31:46 +00:00
(async () => {
try {
await initInstance(client, instance);
} catch (e) {
} finally {
setIsLoggedIn(true);
setUIState('default');
}
2023-02-07 16:31:46 +00:00
})();
2023-02-10 04:29:07 +00:00
} else {
setUIState('default');
}
2022-12-10 09:14:48 +00:00
}
}, []);
let location = useLocation();
states.currentLocation = location.pathname;
useEffect(focusDeck, [location, isLoggedIn]);
2022-12-10 09:14:48 +00:00
const prevLocation = snapStates.prevLocation;
const backgroundLocation = useRef(prevLocation || null);
2023-09-10 07:30:04 +00:00
const isModalPage = useMemo(() => {
return (
matchPath('/:instance/s/:id', location.pathname) ||
matchPath('/s/:id', location.pathname)
);
}, [location.pathname, matchPath]);
if (isModalPage) {
if (!backgroundLocation.current) backgroundLocation.current = prevLocation;
} else {
backgroundLocation.current = null;
}
console.debug({
backgroundLocation: backgroundLocation.current,
location,
});
2023-04-17 10:56:09 +00:00
if (/\/https?:/.test(location.pathname)) {
return <HttpRoute />;
}
const nonRootLocation = useMemo(() => {
const { pathname } = location;
return !/^\/(login|welcome)/.test(pathname);
}, [location]);
2023-04-18 09:46:40 +00:00
// Change #app dataset based on snapStates.settings.shortcutsViewMode
useEffect(() => {
const $app = document.getElementById('app');
if ($app) {
$app.dataset.shortcutsViewMode = snapStates.shortcuts?.length
? snapStates.settings.shortcutsViewMode
: '';
}
}, [snapStates.shortcuts, snapStates.settings.shortcutsViewMode]);
2023-04-23 04:08:41 +00:00
// Add/Remove cloak class to body
useEffect(() => {
const $body = document.body;
$body.classList.toggle('cloak', snapStates.settings.cloakMode);
}, [snapStates.settings.cloakMode]);
2022-12-10 09:14:48 +00:00
return (
<>
<Routes location={nonRootLocation || location}>
<Route
path="/"
element={
isLoggedIn ? (
<Home />
) : uiState === 'loading' ? (
2023-08-19 11:21:51 +00:00
<Loader id="loader-root" />
) : (
<Welcome />
)
2022-12-10 09:14:48 +00:00
}
/>
<Route path="/login" element={<Login />} />
<Route path="/welcome" element={<Welcome />} />
</Routes>
<Routes location={backgroundLocation.current || location}>
{isLoggedIn && (
<Route path="/notifications" element={<Notifications />} />
)}
2023-04-06 11:32:26 +00:00
{isLoggedIn && <Route path="/mentions" element={<Mentions />} />}
{isLoggedIn && <Route path="/following" element={<Following />} />}
{isLoggedIn && <Route path="/b" element={<Bookmarks />} />}
{isLoggedIn && <Route path="/f" element={<Favourites />} />}
2023-02-10 16:05:18 +00:00
{isLoggedIn && (
<Route path="/l">
<Route index element={<Lists />} />
<Route path=":id" element={<List />} />
</Route>
)}
2023-02-11 08:48:47 +00:00
{isLoggedIn && <Route path="/ft" element={<FollowedHashtags />} />}
2023-02-18 12:48:24 +00:00
<Route path="/:instance?/t/:hashtag" element={<Hashtag />} />
2023-02-06 11:54:18 +00:00
<Route path="/:instance?/a/:id" element={<AccountStatuses />} />
2023-02-06 12:17:07 +00:00
<Route path="/:instance?/p">
<Route index element={<Public />} />
<Route path="l" element={<Public local />} />
</Route>
2023-04-05 17:14:38 +00:00
<Route path="/:instance?/trending" element={<Trending />} />
<Route path="/:instance?/search" element={<Search />} />
{/* <Route path="/:anything" element={<NotFound />} /> */}
</Routes>
{uiState === 'default' && (
<Routes>
<Route path="/:instance?/s/:id" element={<StatusRoute />} />
</Routes>
)}
2023-09-05 10:49:16 +00:00
{isLoggedIn && <ComposeButton />}
2023-05-08 04:53:27 +00:00
{isLoggedIn &&
!snapStates.settings.shortcutsColumnsMode &&
snapStates.settings.shortcutsViewMode !== 'multi-column' && (
<Shortcuts />
)}
<Modals />
{isLoggedIn && <NotificationService />}
<BackgroundService isLoggedIn={isLoggedIn} />
{uiState !== 'loading' && <SearchCommand onClose={focusDeck} />}
2023-09-06 14:54:05 +00:00
<KeyboardShortcutsHelp />
2022-12-10 09:14:48 +00:00
</>
);
}
2022-12-31 17:46:08 +00:00
export { App };