import Backbone from 'backbone'; import _ from 'lodash'; import { createRoot } from 'react-dom/client'; import nativeEmojiData from '@emoji-mart/data'; import { ipcRenderer } from 'electron'; // eslint-disable-next-line import/no-named-default import { isMacOS } from '../OS'; import { SessionInboxView } from '../components/SessionInboxView'; import { SessionRegistrationView } from '../components/registration/SessionRegistrationView'; import { Data } from '../data/data'; import { OpenGroupData } from '../data/opengroups'; import { SettingsKey } from '../data/settings-key'; import { MessageModel } from '../models/message'; import { queueAllCached } from '../receiver/receiver'; import { loadKnownBlindedKeys } from '../session/apis/open_group_api/sogsv3/knownBlindedkeys'; import { getConversationController } from '../session/conversations'; import { DisappearingMessages } from '../session/disappearing_messages'; import { AttachmentDownloads, ToastUtils } from '../session/utils'; import { getOurPubKeyStrFromCache } from '../session/utils/User'; import { runners } from '../session/utils/job_runners/JobRunner'; import { LibSessionUtil } from '../session/utils/libsession/libsession_utils'; import { switchPrimaryColorTo } from '../themes/switchPrimaryColor'; import { switchThemeTo } from '../themes/switchTheme'; import { BlockedNumberController } from '../util'; import { initialiseEmojiData } from '../util/emoji'; import { Notifications } from '../util/notifications'; import { Registration } from '../util/registration'; import { Storage, isSignInByLinking } from '../util/storage'; import { getOppositeTheme, isThemeMismatched } from '../util/theme'; import { getCrowdinLocale } from '../util/i18n/shared'; import { rtlLocales } from '../localization/constants'; // Globally disable drag and drop document.body.addEventListener( 'dragover', e => { e.preventDefault(); e.stopPropagation(); }, false ); document.body.addEventListener( 'drop', e => { e.preventDefault(); e.stopPropagation(); }, false ); // Load these images now to ensure that they don't flicker on first use const images = []; function preload(list: Array) { for (let index = 0, max = list.length; index < max; index += 1) { const image = new Image(); image.src = `./images/${list[index]}`; images.push(image); } } preload([ 'alert-outline.svg', 'check.svg', 'error.svg', 'file-gradient.svg', 'file.svg', 'image.svg', 'microphone.svg', 'movie.svg', 'open_link.svg', 'play.svg', 'save.svg', 'shield.svg', 'timer.svg', 'video.svg', 'warning.svg', 'x.svg', ]); // We add this to window here because the default Node context is erased at the end // of preload.js processing window.setImmediate = window.nodeSetImmediate; window.globalOnlineStatus = true; // default to true as we don't get an event on app start window.getGlobalOnlineStatus = () => window.globalOnlineStatus; window.log.info('background page reloaded'); window.log.info('environment:', window.getEnvironment()); let newVersion = false; window.document.title = window.getTitle(); const WhisperEvents = _.clone(Backbone.Events); window.Whisper = window.Whisper || {}; window.Whisper.events = WhisperEvents; window.log.info('Storage fetch'); void Storage.fetch(); function mapOldThemeToNew(theme: string) { switch (theme) { case 'dark': case 'light': return `classic-${theme}`; case 'android-dark': return 'classic-dark'; case 'android': case 'ios': case '': return 'classic-dark'; default: return theme; } } // using __unused as lodash is imported using _ ipcRenderer.on('native-theme-update', (__unused, shouldUseDarkColors) => { const shouldFollowSystemTheme = window.getSettingValue(SettingsKey.hasFollowSystemThemeEnabled); if (shouldFollowSystemTheme) { const theme = window.Events.getThemeSetting(); if (isThemeMismatched(theme, shouldUseDarkColors)) { const newTheme = getOppositeTheme(theme); void switchThemeTo({ theme: newTheme, mainWindow: true, usePrimaryColor: true, dispatch: window?.inboxStore?.dispatch, }); } } }); async function startJobRunners() { // start the job runners await runners.avatarDownloadRunner.loadJobsFromDb(); runners.avatarDownloadRunner.startProcessing(); await runners.configurationSyncRunner.loadJobsFromDb(); runners.configurationSyncRunner.startProcessing(); await runners.updateMsgExpiryRunner.loadJobsFromDb(); runners.updateMsgExpiryRunner.startProcessing(); await runners.fetchSwarmMsgExpiryRunner.loadJobsFromDb(); runners.fetchSwarmMsgExpiryRunner.startProcessing(); } // We need this 'first' check because we don't want to start the app up any other time // than the first time. And storage.fetch() will cause onready() to fire. let first = true; // eslint-disable-next-line @typescript-eslint/no-misused-promises Storage.onready(async () => { if (!first) { return; } first = false; // Update zoom window.updateZoomFactor(); // Ensure accounts created prior to 1.0.0-beta8 do have their // 'primaryDevicePubKey' defined. if (Registration.isDone() && !Storage.get('primaryDevicePubKey')) { await Storage.put('primaryDevicePubKey', getOurPubKeyStrFromCache()); } // These make key operations available to IPC handlers created in preload.js window.Events = { getPrimaryColorSetting: () => Storage.get('primary-color-setting', 'green'), setPrimaryColorSetting: async (value: any) => { await Storage.put('primary-color-setting', value); }, getThemeSetting: () => Storage.get('theme-setting', 'classic-dark'), setThemeSetting: async (value: any) => { await Storage.put('theme-setting', value); }, getHideMenuBar: () => Storage.get('hide-menu-bar'), setHideMenuBar: async (value: boolean) => { await Storage.put('hide-menu-bar', value); window.setAutoHideMenuBar(false); window.setMenuBarVisibility(!value); }, getSpellCheck: () => Storage.get('spell-check', true), setSpellCheck: async (value: boolean) => { await Storage.put('spell-check', value); }, shutdown: async () => { // Stop background processing AttachmentDownloads.stop(); // Stop processing incoming messages // TODOLATER stop polling opengroupv2 and swarm nodes // Shut down the data interface cleanly await Data.shutdown(); }, }; const currentVersion = window.getVersion(); const lastVersion = Storage.get('version'); newVersion = !lastVersion || currentVersion !== lastVersion; await Storage.put('version', currentVersion); if (newVersion) { window.log.info(`New version detected: ${currentVersion}; previous: ${lastVersion}`); await Data.cleanupOrphanedAttachments(); } const themeSetting = window.Events.getThemeSetting(); const newThemeSetting = mapOldThemeToNew(themeSetting); await window.Events.setThemeSetting(newThemeSetting); try { if (Registration.isDone()) { try { await LibSessionUtil.initializeLibSessionUtilWrappers(); } catch (e) { window.log.warn('LibSessionUtil.initializeLibSessionUtilWrappers failed with', e.message); // I don't think there is anything we can do if this happens throw e; } } await initialiseEmojiData(nativeEmojiData); await AttachmentDownloads.initAttachmentPaths(); await Promise.all([ getConversationController().load(), BlockedNumberController.load(), OpenGroupData.opengroupRoomsLoad(), loadKnownBlindedKeys(), ]); await startJobRunners(); } catch (error) { window.log.error( 'main_renderer: ConversationController failed to load:', error && error.stack ? error.stack : error ); } finally { void start(); } }); async function manageExpiringData() { await Data.cleanSeenMessages(); await Data.cleanLastHashes(); // eslint-disable-next-line @typescript-eslint/no-misused-promises setTimeout(manageExpiringData, 1000 * 60 * 60); } async function start() { void manageExpiringData(); window.dispatchEvent(new Event('storage_ready')); window.log.info('Cleanup: starting...'); const results = await Promise.all([Data.getOutgoingWithoutExpiresAt()]); // Combine the models const messagesForCleanup = results.reduce( (array, current) => array.concat((current as any).toArray()), [] ); window.log.info(`Cleanup: Found ${messagesForCleanup.length} messages for cleanup`); const idsToCleanUp: Array = []; await Promise.all( messagesForCleanup.map((message: MessageModel) => { const sentAt = message.get('sent_at'); if (message.hasErrors()) { return null; } window.log.info(`Cleanup: Deleting unsent message ${sentAt}`); idsToCleanUp.push(message.id); return null; }) ); if (idsToCleanUp.length) { await Data.removeMessagesByIds(idsToCleanUp); } window.log.info('Cleanup: complete'); window.log.info('listening for registration events'); WhisperEvents.on('registration_done', () => { window.log.info('[onboarding] handling registration event'); void connect(); }); function switchBodyToRtlIfNeeded() { const loc = getCrowdinLocale(); if (rtlLocales.includes(loc) && !document.getElementById('body')?.classList.contains('rtl')) { document.getElementById('body')?.classList.add('rtl'); } } function openInbox() { switchBodyToRtlIfNeeded(); const hideMenuBar = Storage.get('hide-menu-bar', true) as boolean; window.setAutoHideMenuBar(hideMenuBar); window.setMenuBarVisibility(!hideMenuBar); // eslint-disable-next-line more/no-then void getConversationController() .loadPromise() ?.then(() => { const container = document.getElementById('root'); const root = createRoot(container!); root.render(); }); } function showRegistrationView() { const container = document.getElementById('root'); const root = createRoot(container!); root.render(); switchBodyToRtlIfNeeded(); } DisappearingMessages.initExpiringMessageListener(); if (Registration.isDone() && !isSignInByLinking()) { await connect(); openInbox(); } else { const primaryColor = window.Events.getPrimaryColorSetting(); await switchPrimaryColorTo(primaryColor); showRegistrationView(); } window.addEventListener('focus', () => { Notifications.clear(); }); window.addEventListener('unload', () => { Notifications.fastClear(); }); // Set user's launch count. const prevLaunchCount = window.getSettingValue('launch-count'); const launchCount = !prevLaunchCount ? 1 : prevLaunchCount + 1; window.setTheme = async newTheme => { await window.Events.setThemeSetting(newTheme); }; window.toggleMenuBar = () => { const current = window.getSettingValue('hide-menu-bar'); if (current === undefined) { window.Events.setHideMenuBar(false); return; } window.Events.setHideMenuBar(!current); }; window.toggleSpellCheck = () => { const currentValue = window.getSettingValue('spell-check'); // if undefined, it means 'default' so true. but we have to toggle it, so false // if not undefined, we take the opposite const newValue = currentValue !== undefined ? !currentValue : false; window.Events.setSpellCheck(newValue); ToastUtils.pushRestartNeeded(); }; window.toggleMediaPermissions = async () => { const value = window.getMediaPermissions(); if (value === true) { const valueCallPermissions = window.getCallMediaPermissions(); if (valueCallPermissions) { window.log.info('toggleMediaPermissions : forcing callPermissions to false'); await window.toggleCallMediaPermissionsTo(false); } } if (value === false && isMacOS()) { window.askForMediaAccess(); } window.setMediaPermissions(!value); }; window.toggleCallMediaPermissionsTo = async enabled => { const previousValue = window.getCallMediaPermissions(); if (previousValue === enabled) { return; } if (previousValue === false) { // value was false and we toggle it so we turn it on if (isMacOS()) { window.askForMediaAccess(); } window.log.info('toggleCallMediaPermissionsTo : forcing audio/video to true'); // turning ON "call permissions" forces turning on "audio/video permissions" window.setMediaPermissions(true); } window.setCallMediaPermissions(enabled); }; // eslint-disable-next-line @typescript-eslint/no-misused-promises window.openFromNotification = async conversationKey => { window.showWindow(); if (conversationKey) { // do not put the messageId here so the conversation is loaded on the last unread instead await window.openConversationWithMessages({ conversationKey, messageId: null, }); } else { openInbox(); } }; await window.setSettingValue('launch-count', launchCount); // On first launch if (launchCount === 1) { // Initialise default settings await window.setSettingValue('hide-menu-bar', true); await window.setSettingValue(SettingsKey.settingsLinkPreview, false); } WhisperEvents.on('openInbox', () => { openInbox(); }); } let disconnectTimer: NodeJS.Timeout | null = null; function onOffline() { window.log.info('offline'); window.globalOnlineStatus = false; window.removeEventListener('offline', onOffline); window.addEventListener('online', onOnline); // We've received logs from Linux where we get an 'offline' event, then 30ms later // we get an online event. This waits a bit after getting an 'offline' event // before disconnecting the socket manually. disconnectTimer = global.setTimeout(disconnect, 1000); } function onOnline() { window.log.info('online'); window.globalOnlineStatus = true; window.removeEventListener('online', onOnline); window.addEventListener('offline', onOffline); if (disconnectTimer) { window.log.warn('Already online. Had a blip in online/offline status.'); clearTimeout(disconnectTimer); disconnectTimer = null; return; } if (disconnectTimer) { clearTimeout(disconnectTimer); disconnectTimer = null; } void connect(); } function disconnect() { window.log.info('disconnect'); // Clear timer, since we're only called when the timer is expired disconnectTimer = null; AttachmentDownloads.stop(); } let connectCount = 0; async function connect() { window.log.info('connect'); // Bootstrap our online/offline detection, only the first time we connect if (connectCount === 0 && navigator.onLine) { window.addEventListener('offline', onOffline); } if (connectCount === 0 && !navigator.onLine) { window.log.warn('Starting up offline; will connect when we have network access'); window.addEventListener('online', onOnline); onEmpty(); // this ensures that the loading screen is dismissed return; } if (!Registration.everDone()) { return; } connectCount += 1; Notifications.disable(); // avoid notification flood until empty setTimeout(() => { Notifications.enable(); }, 10 * 1000); // 10 sec setTimeout(() => { void queueAllCached(); }, 10 * 1000); // 10 sec await AttachmentDownloads.start({ logger: window.log, }); window.isOnline = true; } function onEmpty() { window.readyForUpdates(); Notifications.enable(); }