/* * 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 . */ import {observer} from 'mobx-react-lite'; import type {ReactNode} from 'react'; import {vi} from 'vitest'; if (typeof globalThis.URLPattern === 'undefined') { await import('urlpattern-polyfill'); } vi.mock('~/lib/Platform', () => ({ Platform: { OS: 'web', isWeb: true, isNative: false, isIOS: false, isAndroid: false, isElectron: false, select: (specifics: Record) => specifics.web ?? specifics.ios ?? specifics.android ?? Object.values(specifics)[0], Version: '1.0.0', }, isWebPlatform: () => true, isNativePlatform: () => false, isElectronPlatform: () => false, getNativeLocaleIdentifier: () => null, })); vi.mock('mobx-persist-store', () => ({ makePersistable: vi.fn(() => Promise.resolve()), stopPersisting: vi.fn(), configurePersistable: vi.fn(), isHydrated: vi.fn(() => true), isPersisting: vi.fn(() => false), })); vi.mock('~/lib/Logger', () => { const noop = () => {}; class MockLogger { child() { return new MockLogger(); } trace = noop; debug = noop; info = noop; warn = noop; error = noop; fatal = noop; } return { Logger: MockLogger, LogLevel: {Trace: 0, Debug: 1, Info: 2, Warn: 3, Error: 4, Fatal: 5, Silent: 6}, }; }); vi.mock('~/stores/UpdaterStore', () => ({ default: { hasUpdate: false, state: 'idle', isChecking: false, updateType: null, updateInfo: { native: {available: false, version: null}, web: {available: false, sha: null, buildNumber: null}, }, displayVersion: null, currentVersion: null, channel: null, lastCheckedAt: null, checkForUpdates: vi.fn(() => Promise.resolve()), dismissUpdate: vi.fn(), applyUpdate: vi.fn(), }, })); vi.mock('katex', () => ({ default: { renderToString: vi.fn(() => 'mocked'), }, })); vi.mock('~/lib/HttpClient', () => { const defaultInstance = { api_code_version: 1, endpoints: { api: 'https://localhost/api', api_client: 'https://localhost/api', api_public: 'https://localhost/api', gateway: 'wss://localhost/gateway', media: 'https://localhost/media', cdn: 'https://localhost/cdn', marketing: 'https://localhost/marketing', admin: 'https://localhost/admin', invite: 'https://localhost/invite', gift: 'https://localhost/gift', webapp: 'https://localhost', }, captcha: {provider: 'none', hcaptcha_site_key: null, turnstile_site_key: null}, features: {sms_mfa_enabled: false, voice_enabled: false, stripe_enabled: false, self_hosted: false}, }; const mockResponse = Promise.resolve({ok: true, status: 200, headers: {}, body: defaultInstance}); const client = { get: vi.fn(() => mockResponse), post: vi.fn(() => mockResponse), put: vi.fn(() => mockResponse), patch: vi.fn(() => mockResponse), delete: vi.fn(() => mockResponse), request: vi.fn(() => mockResponse), setBaseUrl: vi.fn(), setSudoHandler: vi.fn(), setSudoFailureHandler: vi.fn(), setSudoTokenProvider: vi.fn(), setSudoTokenListener: vi.fn(), setSudoTokenInvalidator: vi.fn(), }; return {__esModule: true, default: client}; }); vi.mock('~/components/channel/emoji-picker/EmojiPickerConstants', () => ({ EMOJI_CLAP: '👏', EMOJI_SPRITE_SIZE: 32, EMOJI_ROW_HEIGHT: 48, CATEGORY_HEADER_HEIGHT: 32, EMOJIS_PER_ROW: 9, OVERSCAN_ROWS: 5, getSpriteSheetPath: () => 'sprite.png', getSpriteSheetBackground: () => 'url(sprite.png)', })); vi.mock('~/components/modals/ImageCropModal', () => ({ default: () => null, })); vi.mock('@lingui/core/macro', () => { const formatMessage = ( str: TemplateStringsArray | string | {message?: string; defaultMessage?: string} | undefined, ...expr: Array ) => { if (typeof str === 'string') return str; if (Array.isArray(str)) { return String(str.reduce((acc, chunk, i) => acc + chunk + (expr[i] ?? ''), '')); } if (str && typeof str === 'object') { return (str as any).message ?? (str as any).defaultMessage ?? ''; } return ''; }; return { t: formatMessage, msg: formatMessage, }; }); vi.mock('@lingui/react/macro', async () => { const React = await import('react'); interface TransProps { children?: ReactNode; id?: string; message?: ReactNode; } const Trans = observer(({children, id, message}: TransProps) => { if (children != null) return React.createElement(React.Fragment, null, children); return React.createElement('span', null, message ?? id ?? null); }); interface PluralProps { value: number; one?: ReactNode; other?: ReactNode; zero?: ReactNode; few?: ReactNode; many?: ReactNode; } const Plural = observer(({value, one, other, zero, few, many}: PluralProps) => { if (value === 0 && zero != null) return zero; if (value === 1 && one != null) return one; if (typeof few !== 'undefined' && value >= 2 && value <= 4) return few; if (typeof many !== 'undefined' && value >= 5) return many; return other ?? null; }); interface SelectProps extends Record { value: string | number; other?: ReactNode; } const Select = observer(({value, other, ...cases}: SelectProps) => { const key = String(value); return Object.hasOwn(cases, key) ? (cases[key] as ReactNode) : (other ?? null); }); const SelectOrdinal = Plural; return { Trans, Plural, Select, SelectOrdinal, }; }); vi.mock('~/utils/NotificationUtils', () => ({ ensureDesktopNotificationClickHandler: vi.fn(), hasNotification: () => false, isGranted: async () => false, playNotificationSoundIfEnabled: vi.fn(), requestPermission: async () => {}, showNotification: async () => ({browserNotification: null, nativeNotificationId: null}), closeNativeNotification: () => undefined, closeNativeNotifications: () => undefined, }));