super-productivity/electron/start-app.ts
2026-01-05 14:45:06 +01:00

384 lines
12 KiB
TypeScript

import { initIpcInterfaces } from './ipc-handler';
import electronLog, { info, log } from 'electron-log/main';
import {
App,
app,
BrowserWindow,
globalShortcut,
ipcMain,
powerMonitor,
protocol,
} from 'electron';
import { join } from 'path';
import { initDebug } from './debug';
import electronDl from 'electron-dl';
import { IPC } from './shared-with-frontend/ipc-events.const';
import { initBackupAdapter } from './backup';
import { initLocalFileSyncAdapter } from './local-file-sync';
import { initFullScreenBlocker } from './full-screen-blocker';
import { CONFIG } from './CONFIG';
import { lazySetInterval } from './shared-with-frontend/lazy-set-interval';
import { initIndicator } from './indicator';
import { quitApp, showOrFocus } from './various-shared';
import { createWindow } from './main-window';
import { IdleTimeHandler } from './idle-time-handler';
import { destroyOverlayWindow } from './overlay-indicator/overlay-indicator';
import {
initializeProtocolHandling,
processPendingProtocolUrls,
} from './protocol-handler';
import { getIsQuiting, setIsLocked } from './shared-state';
const ICONS_FOLDER = __dirname + '/assets/icons/';
const IS_MAC = process.platform === 'darwin';
// const DESKTOP_ENV = process.env.DESKTOP_SESSION;
// const IS_GNOME = DESKTOP_ENV === 'gnome' || DESKTOP_ENV === 'gnome-xorg';
const IS_DEV = process.env.NODE_ENV === 'DEV';
let isShowDevTools: boolean = IS_DEV;
let customUrl: string;
let isDisableTray = false;
let forceDarkTray = false;
let wasUserDataDirSet = false;
if (IS_DEV) {
log('Starting in DEV Mode!!!');
}
const appIN: App = app;
let mainWin: BrowserWindow;
let idleTimeHandler: IdleTimeHandler;
export const startApp = (): void => {
// Initialize protocol handling
initializeProtocolHandling(IS_DEV, app, () => mainWin);
// Handle single instance lock
const gotTheLock = app.requestSingleInstanceLock();
if (!gotTheLock) {
app.quit();
return;
}
// LOAD IPC STUFF
initIpcInterfaces();
electronLog.initialize();
app.commandLine.appendSwitch('enable-speech-dispatcher');
// work around for #4375
// https://github.com/super-productivity/super-productivity/issues/4375#issuecomment-2883838113
// https://github.com/electron/electron/issues/46538#issuecomment-2808806722
app.commandLine.appendSwitch('gtk-version', '3');
// NOTE: needs to be executed before everything else
process.argv.forEach((val) => {
if (val && val.includes('--disable-tray')) {
isDisableTray = true;
log('Disable tray icon');
}
if (val && val.includes('--force-dark-tray')) {
forceDarkTray = true;
log('Force dark mode for tray icon');
}
if (val && val.includes('--user-data-dir=')) {
const customUserDir = val
.replace('--user-data-dir=', '')
.trim()
.replace(/[\/\\]+$/, ''); // Remove trailing slashes
log('Using custom directory for user data', customUserDir);
app.setPath('userData', customUserDir);
wasUserDataDirSet = true;
}
if (val && val.includes('--custom-url=')) {
customUrl = val.replace('--custom-url=', '').trim();
log('Using custom url', customUrl);
}
if (val && val.includes('--dev-tools')) {
isShowDevTools = true;
}
});
// TODO remove at one point in the future and only leave the directory setting part
// Special handling for snaps, since default user folder will cause problems when updating
if (
!wasUserDataDirSet &&
process.platform === 'linux' &&
process.env.SNAP &&
process.env.SNAP_USER_COMMON
) {
// COPY LEGACY SNAP DATA TO COMMON DIRECTORY
// -----------------------------------------
const appName = app.getName();
const commonDir = process.env.SNAP_USER_COMMON;
const newPath = join(commonDir, '.config', appName);
// SET COMMON DIRECTORY AS USER DATA DIRECTORY
// -------------------------------------------
// set userDa dir to common data to avoid the data being accessed by the update process
app.setPath('userData', newPath);
app.setAppLogsPath();
}
initDebug({ showDevTools: isShowDevTools }, IS_DEV);
// NOTE: opening the folder crashes the mas build
if (!IS_MAC) {
electronDl({
openFolderWhenDone: true,
onCompleted: (file) => {
if (mainWin) {
mainWin.webContents.send(IPC.ANY_FILE_DOWNLOADED, file);
}
},
});
}
// Allow invalid certificates for jira requests
appIN.on('certificate-error', (event, webContents, url, err, certificate, callback) => {
log(err);
event.preventDefault();
callback(true);
});
// APP EVENT LISTENERS
// -------------------
appIN.on('ready', () => createMainWin());
appIN.on('ready', () => initBackupAdapter());
appIN.on('ready', () => initLocalFileSyncAdapter());
appIN.on('ready', () => initFullScreenBlocker(IS_DEV));
if (!isDisableTray) {
appIN.on('ready', createIndicator);
}
appIN.on('activate', () => {
// On OS X it's common to re-create a window in the app when the
// dock icon is clicked and there are no other windows open.
if (mainWin === null) {
createMainWin();
} else {
showApp();
}
});
appIN.on('ready', () => {
// Initialize idle time handler
idleTimeHandler = new IdleTimeHandler();
let suspendStart: number;
// Prevent overlapping async idle checks.
// lazySetInterval schedules the next tick regardless of whether the previous
// check finished. Our idle detection on Wayland may spawn external commands
// (gdbus/dbus-send/xprintidle/loginctl) which can take close to or longer than
// the poll interval. Without this guard, multiple checks can run concurrently,
// causing timeouts and subsequent 0ms readings, which looks like “only one
// idle event was ever sent”. This ensures at most one check runs at a time.
let isCheckingIdle = false;
const sendIdleMsgIfOverMin = (
idleTime: number,
): { sent: boolean; reason?: string } => {
// sometimes when starting a second instance we get here although we don't want to
if (!mainWin) {
info(
'special case occurred when trackTimeFn is called even though, this is a second instance of the app',
);
return { sent: false, reason: 'no-window' };
}
if (getIsQuiting()) {
return { sent: false, reason: 'quitting' };
}
if (idleTime <= CONFIG.MIN_IDLE_TIME) {
return { sent: false, reason: 'below-threshold' };
}
mainWin.webContents.send(IPC.IDLE_TIME, idleTime);
return { sent: true };
};
// --------IDLE HANDLING---------
let consecutiveFailures = 0;
// init time tracking interval
log(
`🚀 Starting idle time tracking (interval: ${CONFIG.IDLE_PING_INTERVAL}ms, threshold: ${CONFIG.MIN_IDLE_TIME}ms)`,
);
const stopIdleChecks: () => void = lazySetInterval(async (): Promise<void> => {
// Skip if a previous check is still in flight
if (isCheckingIdle) {
return;
}
isCheckingIdle = true;
const startTime = Date.now();
try {
const idleTime = await idleTimeHandler.getIdleTime();
const checkDuration = Date.now() - startTime;
consecutiveFailures = 0;
const sendResult = sendIdleMsgIfOverMin(idleTime);
const actionSummary = sendResult.sent
? 'sent'
: `skipped:${sendResult.reason ?? 'unknown'}`;
const logParts = [
`idle=${idleTime}ms`,
`method=${idleTimeHandler.currentMethod}`,
`duration=${checkDuration}ms`,
`threshold=${CONFIG.MIN_IDLE_TIME}ms`,
`action=${actionSummary}`,
];
electronLog.debug(`🕘 Idle check (${logParts.join(', ')})`);
} catch (error) {
consecutiveFailures += 1;
log('💥 Error getting idle time, falling back to powerMonitor:', error);
if (consecutiveFailures >= 3) {
stopIdleChecks();
}
} finally {
isCheckingIdle = false;
}
}, CONFIG.IDLE_PING_INTERVAL);
// --------END IDLE HANDLING---------
powerMonitor.on('suspend', () => {
log('powerMonitor: System suspend detected');
setIsLocked(true);
suspendStart = Date.now();
mainWin.webContents.send(IPC.SUSPEND);
});
powerMonitor.on('lock-screen', () => {
log('powerMonitor: Screen lock detected');
setIsLocked(true);
suspendStart = Date.now();
mainWin.webContents.send(IPC.SUSPEND);
});
powerMonitor.on('resume', () => {
const idleTime = Date.now() - suspendStart;
log(`powerMonitor: System resume detected. Idle time: ${idleTime}ms`);
setIsLocked(false);
sendIdleMsgIfOverMin(idleTime);
mainWin.webContents.send(IPC.RESUME);
});
powerMonitor.on('unlock-screen', () => {
const idleTime = Date.now() - suspendStart;
log(`powerMonitor: Screen unlock detected. Idle time: ${idleTime}ms`);
setIsLocked(false);
sendIdleMsgIfOverMin(idleTime);
mainWin.webContents.send(IPC.RESUME);
});
protocol.registerFileProtocol('file', (request, callback) => {
const pathname = decodeURI(request.url.replace('file:///', ''));
callback(pathname);
});
});
appIN.on('will-quit', () => {
// un-register all shortcuts.
globalShortcut.unregisterAll();
});
appIN.on('before-quit', () => {
log('App before-quit: cleaning up resources');
// Clean up overlay window before quitting
destroyOverlayWindow();
// Remove all IPC listeners to prevent memory leaks
ipcMain.removeAllListeners();
// Clear any pending timeouts/intervals
if (global.gc) {
global.gc();
}
});
appIN.on('window-all-closed', () => {
log('Quit after all windows being closed');
// Force quit the app
app.quit();
// If app doesn't quit within 2 seconds, force exit
setTimeout(() => {
log('Force exiting app as it did not quit properly');
app.exit(0);
}, 2000);
});
process.on('uncaughtException', (err) => {
console.log(err);
process.exit(333);
});
// AUTO-UPDATER
// ------------
// appIN.on('ready', () => {
// // init auto-updates
// log.info('INIT AUTO UPDATES');
// // log.info(autoUpdater.getFeedURL());
// autoUpdater.logger = log;
// autoUpdater.logger.transports.file.level = 'info';
// autoUpdater.checkForUpdatesAndNotify();
// });
//
// autoUpdater.on('update-downloaded', (ev, info) => {
// log(ev);
// // Wait 5 seconds, then quit and install
// // In your application, you don't need to wait 5 seconds.
// // You could call autoUpdater.quitAndInstall(); immediately
// setTimeout(function() {
// autoUpdater.quitAndInstall();
// }, 5000)
// });
// HELPER FUNCTIONS
// ----------------
// eslint-disable-next-line prefer-arrow/prefer-arrow-functions
function createIndicator(): void {
initIndicator({
app,
showApp,
quitApp,
ICONS_FOLDER,
forceDarkTray,
});
}
// eslint-disable-next-line prefer-arrow/prefer-arrow-functions
async function createMainWin(): Promise<void> {
mainWin = await createWindow({
app,
IS_DEV,
ICONS_FOLDER,
quitApp,
customUrl,
});
// Process any pending protocol URLs after window is created
setTimeout(() => {
processPendingProtocolUrls(mainWin);
}, 1000);
}
// eslint-disable-next-line prefer-arrow/prefer-arrow-functions
function showApp(): void {
showOrFocus(mainWin);
}
// required for graceful closing
// @see: https://github.com/electron/electron/issues/5708
process.on('exit', () => {
setTimeout(() => {
log('Quit after process exit');
app.quit();
}, 100);
});
};