'use strict'; import { QueryClientProvider } from '@tanstack/react-query'; import clsx from 'clsx'; import React, { useState, useEffect, useRef } from 'react'; import { Toaster } from 'react-hot-toast'; import { HotKeys } from 'react-hotkeys'; import { IntlProvider } from 'react-intl'; import { Provider } from 'react-redux'; import { BrowserRouter, Switch, Redirect, Route, useHistory } from 'react-router-dom'; import { CompatRouter } from 'react-router-dom-v5-compat'; // @ts-ignore: it doesn't have types import { ScrollContext } from 'react-router-scroll-4'; import { resetCompose } from 'soapbox/actions/compose'; import { loadInstance } from 'soapbox/actions/instance'; import { fetchMe } from 'soapbox/actions/me'; import { openModal } from 'soapbox/actions/modals'; import { loadSoapboxConfig, getSoapboxConfig } from 'soapbox/actions/soapbox'; import { fetchVerificationConfig } from 'soapbox/actions/verification'; import * as BuildConfig from 'soapbox/build-config'; import GdprBanner from 'soapbox/components/gdpr-banner'; import Helmet from 'soapbox/components/helmet'; import LoadingScreen from 'soapbox/components/loading-screen'; import { StatProvider } from 'soapbox/contexts/stat-context'; import AuthLayout from 'soapbox/features/auth-layout'; import EmbeddedStatus from 'soapbox/features/embedded-status'; import PublicLayout from 'soapbox/features/public-layout'; import BundleContainer from 'soapbox/features/ui/containers/bundle-container'; import { ModalContainer, OnboardingWizard, WaitlistPage, } from 'soapbox/features/ui/util/async-components'; import { createGlobals } from 'soapbox/globals'; import { useAppSelector, useAppDispatch, useOwnAccount, useFeatures, useSoapboxConfig, useSettings, useTheme, useLocale, useInstance, useRegistrationStatus, } from 'soapbox/hooks'; import MESSAGES from 'soapbox/locales/messages'; import { normalizeSoapboxConfig } from 'soapbox/normalizers'; import { queryClient } from 'soapbox/queries/client'; import { useCachedLocationHandler } from 'soapbox/utils/redirect'; import { generateThemeCss } from 'soapbox/utils/theme'; import { checkOnboardingStatus } from '../actions/onboarding'; import { preload } from '../actions/preload'; import ErrorBoundary from '../components/error-boundary'; import UI from '../features/ui'; import { store } from '../store'; // Configure global functions for developers createGlobals(store); // Preload happens synchronously store.dispatch(preload() as any); // This happens synchronously store.dispatch(checkOnboardingStatus() as any); const keyMap = { help: '?', new: 'n', search: ['s', '/'], forceNew: 'option+n', reply: 'r', favourite: 'f', react: 'e', boost: 'b', mention: 'm', open: ['enter', 'o'], openProfile: 'p', moveDown: ['down', 'j'], moveUp: ['up', 'k'], back: 'backspace', goToHome: 'g h', goToNotifications: 'g n', goToFavourites: 'g f', goToPinned: 'g p', goToProfile: 'g u', goToBlocked: 'g b', goToMuted: 'g m', goToRequests: 'g r', toggleHidden: 'x', toggleSensitive: 'h', openMedia: 'a', }; /** Load initial data from the backend */ const loadInitial = () => { // @ts-ignore return async(dispatch, getState) => { // Await for authenticated fetch await dispatch(fetchMe()); // Await for feature detection await dispatch(loadInstance()); // Await for configuration await dispatch(loadSoapboxConfig()); const state = getState(); const soapboxConfig = getSoapboxConfig(state); const pepeEnabled = soapboxConfig.getIn(['extensions', 'pepe', 'enabled']) === true; if (pepeEnabled && !state.me) { await dispatch(fetchVerificationConfig()); } }; }; /** Highest level node with the Redux store. */ const SoapboxMount = () => { useCachedLocationHandler(); const hotkeys = useRef(null); const history = useHistory(); const dispatch = useAppDispatch(); const me = useAppSelector(state => state.me); const instance = useInstance(); const { account } = useOwnAccount(); const soapboxConfig = useSoapboxConfig(); const features = useFeatures(); const { pepeEnabled } = useRegistrationStatus(); const waitlisted = account && account.source?.approved === false; const needsOnboarding = useAppSelector(state => state.onboarding.needsOnboarding); const showOnboarding = account && !waitlisted && needsOnboarding; const { redirectRootNoLogin } = soapboxConfig; // @ts-ignore: I don't actually know what these should be, lol const shouldUpdateScroll = (prevRouterProps, { location }) => { return !(location.state?.soapboxModalKey && location.state?.soapboxModalKey !== prevRouterProps?.location?.state?.soapboxModalKey); }; const handleHotkeyNew = (e?: KeyboardEvent) => { e?.preventDefault(); if (!hotkeys.current) return; const element = hotkeys.current.querySelector('textarea#compose-textarea') as HTMLTextAreaElement; if (element) { element.focus(); } }; const handleHotkeySearch = (e?: KeyboardEvent) => { e?.preventDefault(); if (!hotkeys.current) return; const element = hotkeys.current.querySelector('input#search') as HTMLInputElement; if (element) { element.focus(); } }; const handleHotkeyForceNew = (e?: KeyboardEvent) => { handleHotkeyNew(e); dispatch(resetCompose()); }; const handleHotkeyBack = () => { if (window.history && window.history.length === 1) { history.push('/'); } else { history.goBack(); } }; const setHotkeysRef: React.LegacyRef = (c: any) => { hotkeys.current = c; if (!me || !hotkeys.current) return; // @ts-ignore hotkeys.current.__mousetrap__.stopCallback = (_e, element) => { return ['TEXTAREA', 'SELECT', 'INPUT', 'EM-EMOJI-PICKER'].includes(element.tagName); }; }; const handleHotkeyToggleHelp = () => { dispatch(openModal('HOTKEYS')); }; const handleHotkeyGoToHome = () => { history.push('/'); }; const handleHotkeyGoToNotifications = () => { history.push('/notifications'); }; const handleHotkeyGoToFavourites = () => { if (!account) return; history.push(`/@${account.username}/favorites`); }; const handleHotkeyGoToPinned = () => { if (!account) return; history.push(`/@${account.username}/pins`); }; const handleHotkeyGoToProfile = () => { if (!account) return; history.push(`/@${account.username}`); }; const handleHotkeyGoToBlocked = () => { history.push('/blocks'); }; const handleHotkeyGoToMuted = () => { history.push('/mutes'); }; const handleHotkeyGoToRequests = () => { history.push('/follow_requests'); }; type HotkeyHandlers = { [key: string]: (keyEvent?: KeyboardEvent) => void }; const handlers: HotkeyHandlers = { help: handleHotkeyToggleHelp, new: handleHotkeyNew, search: handleHotkeySearch, forceNew: handleHotkeyForceNew, back: handleHotkeyBack, goToHome: handleHotkeyGoToHome, goToNotifications: handleHotkeyGoToNotifications, goToFavourites: handleHotkeyGoToFavourites, goToPinned: handleHotkeyGoToPinned, goToProfile: handleHotkeyGoToProfile, goToBlocked: handleHotkeyGoToBlocked, goToMuted: handleHotkeyGoToMuted, goToRequests: handleHotkeyGoToRequests, }; /** Render the onboarding flow. */ const renderOnboarding = () => ( {(Component) => } ); /** Render the auth layout or UI. */ const renderSwitch = () => ( {/* Redirect signup route depending on Pepe enablement. */} {/* We should prefer using /signup in components. */} {pepeEnabled ? ( ) : ( )} {waitlisted && ( ( {(Component) => } )} /> )} {!me && (redirectRootNoLogin ? : )} {!me && ( )} {(features.accountCreation && instance.registrations) && ( )} {pepeEnabled && ( )} ); /** Render the onboarding flow or UI. */ const renderBody = () => { if (showOnboarding) { return renderOnboarding(); } else { return renderSwitch(); } }; return ( } /> {renderBody()} {Component => }
); }; interface ISoapboxLoad { children: React.ReactNode } /** Initial data loader. */ const SoapboxLoad: React.FC = ({ children }) => { const dispatch = useAppDispatch(); const me = useAppSelector(state => state.me); const { account } = useOwnAccount(); const swUpdating = useAppSelector(state => state.meta.swUpdating); const { locale } = useLocale(); const [messages, setMessages] = useState>({}); const [localeLoading, setLocaleLoading] = useState(true); const [isLoaded, setIsLoaded] = useState(false); /** Whether to display a loading indicator. */ const showLoading = [ me === null, me && !account, !isLoaded, localeLoading, swUpdating, ].some(Boolean); // Load the user's locale useEffect(() => { MESSAGES[locale]().then(messages => { setMessages(messages); setLocaleLoading(false); }).catch(() => { }); }, [locale]); // Load initial data from the API useEffect(() => { dispatch(loadInitial()).then(() => { setIsLoaded(true); }).catch(() => { setIsLoaded(true); }); }, []); // intl is part of loading. // It's important nothing in here depends on intl. if (showLoading) { return ; } return ( {children} ); }; interface ISoapboxHead { children: React.ReactNode } /** Injects metadata into site head with Helmet. */ const SoapboxHead: React.FC = ({ children }) => { const { locale, direction } = useLocale(); const settings = useSettings(); const soapboxConfig = useSoapboxConfig(); const demo = !!settings.get('demo'); const darkMode = useTheme() === 'dark'; const themeCss = generateThemeCss(demo ? normalizeSoapboxConfig({ brandColor: '#0482d8' }) : soapboxConfig); const bodyClass = clsx('h-full bg-white text-base dark:bg-gray-800', { 'no-reduce-motion': !settings.get('reduceMotion'), 'underline-links': settings.get('underlineLinks'), 'demetricator': settings.get('demetricator'), }); return ( <> {themeCss && } {darkMode && } {children} ); }; /** The root React node of the application. */ const Soapbox: React.FC = () => { return ( ); }; export default Soapbox;