You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
516 lines
15 KiB
TypeScript
516 lines
15 KiB
TypeScript
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<string>) {
|
|
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<string> = [];
|
|
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(<SessionInboxView />);
|
|
});
|
|
}
|
|
|
|
function showRegistrationView() {
|
|
const container = document.getElementById('root');
|
|
const root = createRoot(container!);
|
|
root.render(<SessionRegistrationView />);
|
|
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();
|
|
}
|