2026-02-17 12:22:36 +00:00

468 lines
16 KiB
TypeScript

/*
* Copyright (C) 2026 Fluxer Contributors
*
* This file is part of Fluxer.
*
* Fluxer is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Fluxer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
*/
import '@app/components/modals/SudoVerificationModal';
import 'highlight.js/styles/github-dark.css';
import 'katex/dist/katex.min.css';
import styles from '@app/App.module.css';
import * as ModalActionCreators from '@app/actions/ModalActionCreators';
import * as WindowActionCreators from '@app/actions/WindowActionCreators';
import Config from '@app/Config';
import {DndContext} from '@app/components/layout/DndContext';
import GlobalOverlays from '@app/components/layout/GlobalOverlays';
import {NativeTitlebar} from '@app/components/layout/NativeTitlebar';
import {NativeTrafficLightsBackdrop} from '@app/components/layout/NativeTrafficLightsBackdrop';
import {UserSettingsModal} from '@app/components/modals/UserSettingsModal';
import {QUICK_SWITCHER_PORTAL_ID} from '@app/components/quick_switcher/QuickSwitcherConstants';
import FocusRingScope from '@app/components/uikit/focus_ring/FocusRingScope';
import {SVGMasks} from '@app/components/uikit/SVGMasks';
import {IncomingCallManager} from '@app/components/voice/IncomingCallManager';
import {type LayoutVariant, LayoutVariantProvider} from '@app/contexts/LayoutVariantContext';
import {showMyselfTypingHelper} from '@app/devtools/ShowMyselfTypingHelper';
import {useActivityRecorder} from '@app/hooks/useActivityRecorder';
import {useElectronScreenSharePicker} from '@app/hooks/useElectronScreenSharePicker';
import {useNativePlatform} from '@app/hooks/useNativePlatform';
import {useTextInputContextMenu} from '@app/hooks/useTextInputContextMenu';
import CaptchaInterceptorStore from '@app/lib/CaptchaInterceptor';
import FocusManager from '@app/lib/FocusManager';
import KeybindManager from '@app/lib/KeybindManager';
import {Logger} from '@app/lib/Logger';
import {startReadStateCleanup} from '@app/lib/ReadStateCleanup';
import {Outlet, RouterProvider} from '@app/lib/router/React';
import {router} from '@app/Router';
import AccessibilityStore, {HdrDisplayMode} from '@app/stores/AccessibilityStore';
import ModalStore from '@app/stores/ModalStore';
import PopoutStore from '@app/stores/PopoutStore';
import ReadStateStore from '@app/stores/ReadStateStore';
import RuntimeCrashStore from '@app/stores/RuntimeCrashStore';
import ThemeStore from '@app/stores/ThemeStore';
import UserStore from '@app/stores/UserStore';
import MediaEngineStore from '@app/stores/voice/MediaEngineFacade';
import {ensureAutostartDefaultEnabled} from '@app/utils/AutostartUtils';
import {startDeepLinkHandling} from '@app/utils/DeepLinkUtils';
import {attachExternalLinkInterceptor, getElectronAPI, getNativePlatform} from '@app/utils/NativeUtils';
import {i18n} from '@lingui/core';
import {I18nProvider} from '@lingui/react';
import {RoomAudioRenderer, RoomContext} from '@livekit/components-react';
import {IconContext} from '@phosphor-icons/react';
import * as Sentry from '@sentry/react';
import {observer} from 'mobx-react-lite';
import React, {type ReactNode, useCallback, useEffect, useMemo, useRef, useState} from 'react';
const logger = new Logger('App');
interface AppWrapperProps {
children: ReactNode;
}
export const AppWrapper = observer(({children}: AppWrapperProps) => {
const saturationFactor = AccessibilityStore.saturationFactor;
const alwaysUnderlineLinks = AccessibilityStore.alwaysUnderlineLinks;
const enableTextSelection = AccessibilityStore.textSelectionEnabled;
const fontSize = AccessibilityStore.fontSize;
const messageGutter = AccessibilityStore.messageGutter;
const messageGroupSpacing = AccessibilityStore.messageGroupSpacingValue;
const reducedMotion = AccessibilityStore.useReducedMotion;
const hdrDisplayMode = AccessibilityStore.hdrDisplayMode;
const {platform, isNative, isMacOS} = useNativePlatform();
useElectronScreenSharePicker();
const customThemeCss = AccessibilityStore.customThemeCss;
const effectiveTheme = ThemeStore.effectiveTheme;
const [layoutVariant, setLayoutVariant] = useState<LayoutVariant>('app');
const layoutVariantContextValue = useMemo(
() => ({variant: layoutVariant, setVariant: setLayoutVariant}),
[layoutVariant],
);
const popouts = PopoutStore.getPopouts();
const topPopout = popouts.length ? popouts[popouts.length - 1] : null;
const topPopoutRequiresBackdrop = Boolean(topPopout && !topPopout.disableBackdrop);
const room = MediaEngineStore.room;
const ringsContainerRef = useRef<HTMLDivElement>(null);
const overlayScopeRef = useRef<HTMLDivElement>(null);
const recordActivity = useActivityRecorder();
const handleUserActivity = useCallback(() => recordActivity(), [recordActivity]);
const handleImmediateActivity = useCallback(() => recordActivity(true), [recordActivity]);
const handleResize = useCallback(() => WindowActionCreators.resized(), []);
useTextInputContextMenu();
const hasBlockingModal = ModalStore.hasModalOpen();
useEffect(() => {
const node = ringsContainerRef.current;
if (!node) return;
const shouldBlockBackground = hasBlockingModal || topPopoutRequiresBackdrop;
node.toggleAttribute('inert', shouldBlockBackground);
return () => {
node.removeAttribute('inert');
};
}, [hasBlockingModal, topPopoutRequiresBackdrop]);
useEffect(() => {
showMyselfTypingHelper.start();
return () => showMyselfTypingHelper.stop();
}, []);
useEffect(() => {
startReadStateCleanup();
}, []);
useEffect(() => {
if (!('serviceWorker' in navigator)) {
return;
}
const postBadgeUpdate = (count: number) => {
const controller = navigator.serviceWorker.controller;
if (!controller) {
return;
}
try {
controller.postMessage({type: 'APP_UPDATE_BADGE', count});
} catch (error) {
logger.warn('Failed to post badge update to service worker', error);
}
};
const updateBadgeFromReadState = () => {
const channelIds = ReadStateStore.getChannelIds();
const totalMentions = channelIds.reduce((sum, channelId) => sum + ReadStateStore.getMentionCount(channelId), 0);
postBadgeUpdate(totalMentions);
};
const unsubscribe = ReadStateStore.subscribe(() => {
updateBadgeFromReadState();
});
return () => {
unsubscribe();
};
}, []);
useEffect(() => {
void KeybindManager.init(i18n);
void CaptchaInterceptorStore;
return () => {
KeybindManager.destroy();
};
}, []);
useEffect(() => {
void AccessibilityStore.applyStoredZoom();
const electronApi = getElectronAPI();
if (!electronApi) return;
const unsubZoomIn = electronApi.onZoomIn?.(() => void AccessibilityStore.adjustZoom(0.1));
const unsubZoomOut = electronApi.onZoomOut?.(() => void AccessibilityStore.adjustZoom(-0.1));
const unsubZoomReset = electronApi.onZoomReset?.(() => AccessibilityStore.updateSettings({zoomLevel: 1.0}));
const unsubOpenSettings = electronApi.onOpenSettings?.(() => {
ModalActionCreators.push(ModalActionCreators.modal(() => <UserSettingsModal />));
});
return () => {
unsubZoomIn?.();
unsubZoomOut?.();
unsubZoomReset?.();
unsubOpenSettings?.();
};
}, []);
useEffect(() => {
const root = document.documentElement;
root.classList.toggle('reduced-motion', reducedMotion);
return () => {
root.classList.remove('reduced-motion');
};
}, [reducedMotion]);
useEffect(() => {
if (Config.PUBLIC_BUILD_SHA && Config.PUBLIC_BUILD_TIMESTAMP) {
const buildInfo = Config.PUBLIC_BUILD_NUMBER
? `build ${Config.PUBLIC_BUILD_NUMBER} (${Config.PUBLIC_BUILD_SHA})`
: Config.PUBLIC_BUILD_SHA;
logger.info(`[BUILD INFO] ${Config.PUBLIC_RELEASE_CHANNEL} - ${buildInfo} - ${Config.PUBLIC_BUILD_TIMESTAMP}`);
}
FocusManager.init();
const shouldRegisterWindowListeners = !isNative;
if (shouldRegisterWindowListeners && document.hasFocus()) {
document.documentElement.classList.add('window-focused');
}
const preventScroll = (event: Event) => event.preventDefault();
const handleBlur = () => {
WindowActionCreators.focused(false);
if (shouldRegisterWindowListeners) {
document.documentElement.classList.remove('window-focused');
}
};
const handleFocus = () => {
WindowActionCreators.focused(true);
if (shouldRegisterWindowListeners) {
document.documentElement.classList.add('window-focused');
}
handleImmediateActivity();
};
const handleVisibilityChange = () => {
WindowActionCreators.visibilityChanged(!document.hidden);
};
const preventPinchZoom = (event: TouchEvent) => {
if (event.touches.length > 1) {
event.preventDefault();
}
};
if (shouldRegisterWindowListeners) {
document.addEventListener('scroll', preventScroll);
window.addEventListener('blur', handleBlur);
window.addEventListener('focus', handleFocus);
document.addEventListener('visibilitychange', handleVisibilityChange);
window.addEventListener('mousedown', handleImmediateActivity);
window.addEventListener('keydown', handleUserActivity);
window.addEventListener('resize', handleResize);
window.addEventListener('touchstart', handleImmediateActivity);
document.addEventListener('touchstart', preventPinchZoom, {passive: false});
document.addEventListener('touchmove', preventPinchZoom, {passive: false});
}
return () => {
FocusManager.destroy();
if (shouldRegisterWindowListeners) {
document.removeEventListener('scroll', preventScroll);
window.removeEventListener('blur', handleBlur);
window.removeEventListener('focus', handleFocus);
document.removeEventListener('visibilitychange', handleVisibilityChange);
window.removeEventListener('mousedown', handleImmediateActivity);
window.removeEventListener('keydown', handleUserActivity);
window.removeEventListener('resize', handleResize);
window.removeEventListener('touchstart', handleImmediateActivity);
document.removeEventListener('touchstart', preventPinchZoom);
document.removeEventListener('touchmove', preventPinchZoom);
}
};
}, [handleImmediateActivity, handleResize, isNative]);
useEffect(() => {
if (!isNative) {
return;
}
const htmlNode = document.documentElement;
const updateClass = (focused: boolean) => {
htmlNode.classList.toggle('window-focused', focused);
};
const handleFocus = () => {
updateClass(true);
WindowActionCreators.focused(true);
handleImmediateActivity();
};
const handleBlur = () => {
updateClass(false);
WindowActionCreators.focused(false);
};
const handleVisibilityChange = () => {
WindowActionCreators.visibilityChanged(!document.hidden);
};
const preventPinchZoom = (event: TouchEvent) => {
if (event.touches.length > 1) {
event.preventDefault();
}
};
updateClass(document.hasFocus());
window.addEventListener('focus', handleFocus);
window.addEventListener('blur', handleBlur);
document.addEventListener('visibilitychange', handleVisibilityChange);
window.addEventListener('mousedown', handleImmediateActivity);
window.addEventListener('keydown', handleUserActivity);
window.addEventListener('resize', handleResize);
window.addEventListener('touchstart', handleImmediateActivity);
document.addEventListener('touchstart', preventPinchZoom, {passive: false});
document.addEventListener('touchmove', preventPinchZoom, {passive: false});
return () => {
window.removeEventListener('focus', handleFocus);
window.removeEventListener('blur', handleBlur);
document.removeEventListener('visibilitychange', handleVisibilityChange);
window.removeEventListener('mousedown', handleImmediateActivity);
window.removeEventListener('keydown', handleUserActivity);
window.removeEventListener('resize', handleResize);
window.removeEventListener('touchstart', handleImmediateActivity);
document.removeEventListener('touchstart', preventPinchZoom);
document.removeEventListener('touchmove', preventPinchZoom);
};
}, [handleImmediateActivity, handleResize, isNative]);
useEffect(() => {
const htmlNode = document.documentElement;
const platformClasses = [isNative ? 'platform-native' : 'platform-web', `platform-${platform}`];
htmlNode.classList.add(...platformClasses);
return () => {
htmlNode.classList.remove(...platformClasses);
};
}, [isNative, platform]);
useEffect(() => {
const htmlNode = document.documentElement;
htmlNode.classList.add(`theme-${effectiveTheme}`);
htmlNode.style.setProperty('--saturation-factor', saturationFactor.toString());
htmlNode.style.setProperty('--user-select', enableTextSelection ? 'auto' : 'none');
htmlNode.style.setProperty('--font-size', `${fontSize}px`);
htmlNode.style.setProperty('--chat-horizontal-padding', `${messageGutter}px`);
htmlNode.style.setProperty('--message-group-spacing', `${messageGroupSpacing}px`);
htmlNode.style.setProperty('dynamic-range-limit', hdrDisplayMode === HdrDisplayMode.FULL ? 'high' : 'standard');
if (alwaysUnderlineLinks) {
htmlNode.style.setProperty('--link-decoration', 'underline');
} else {
htmlNode.style.removeProperty('--link-decoration');
}
return () => {
htmlNode.classList.remove(`theme-${effectiveTheme}`);
htmlNode.style.removeProperty('--saturation-factor');
htmlNode.style.removeProperty('--link-decoration');
htmlNode.style.removeProperty('--user-select');
htmlNode.style.removeProperty('--font-size');
htmlNode.style.removeProperty('--chat-horizontal-padding');
htmlNode.style.removeProperty('--message-group-spacing');
htmlNode.style.removeProperty('dynamic-range-limit');
};
}, [
effectiveTheme,
saturationFactor,
alwaysUnderlineLinks,
enableTextSelection,
fontSize,
messageGutter,
messageGroupSpacing,
hdrDisplayMode,
]);
useEffect(() => {
const styleElementId = 'fluxer-custom-theme-style';
const existing = document.getElementById(styleElementId) as HTMLStyleElement | null;
const css = customThemeCss?.trim() ?? '';
if (!css) {
if (existing?.parentNode) {
existing.parentNode.removeChild(existing);
}
return;
}
const styleElement = existing ?? document.createElement('style');
styleElement.id = styleElementId;
styleElement.textContent = css;
if (!existing) {
document.head.appendChild(styleElement);
}
}, [customThemeCss]);
return (
<LayoutVariantProvider value={layoutVariantContextValue}>
<SVGMasks />
<RoomContext.Provider value={room ?? undefined}>
{room && <RoomAudioRenderer />}
<div ref={ringsContainerRef} className={styles.appContainer}>
<FocusRingScope containerRef={ringsContainerRef}>
<NativeTrafficLightsBackdrop variant={layoutVariant} />
{isNative && !isMacOS && <NativeTitlebar platform={platform} />}
{children}
</FocusRingScope>
</div>
<div ref={overlayScopeRef} className={styles.overlayScope}>
<div
id={QUICK_SWITCHER_PORTAL_ID}
className={styles.quickSwitcherPortal}
data-overlay-pass-through="true"
aria-hidden="true"
/>
<GlobalOverlays />
<IncomingCallManager />
</div>
</RoomContext.Provider>
</LayoutVariantProvider>
);
});
export const App = observer((): React.ReactElement => {
const currentUser = UserStore.currentUser;
const fatalError = RuntimeCrashStore.fatalError;
if (fatalError) {
throw fatalError;
}
useEffect(() => {
const initAutostart = async () => {
const platform = await getNativePlatform();
if (platform === 'macos') {
void ensureAutostartDefaultEnabled();
}
};
void initAutostart();
}, []);
useEffect(() => {
void startDeepLinkHandling();
}, []);
useEffect(() => {
const detach = attachExternalLinkInterceptor();
return () => detach?.();
}, []);
useEffect(() => {
if (currentUser) {
Sentry.setUser({
id: currentUser.id,
username: currentUser.username,
email: currentUser.email ?? undefined,
});
} else {
Sentry.setUser(null);
}
}, [currentUser]);
return (
<I18nProvider i18n={i18n}>
<IconContext.Provider value={{color: 'currentColor', weight: 'fill'}}>
<DndContext>
<RouterProvider router={router}>
<AppWrapper>
<Outlet />
</AppWrapper>
</RouterProvider>
</DndContext>
</IconContext.Provider>
</I18nProvider>
);
});