mirror of
https://github.com/johannesjo/super-productivity.git
synced 2026-01-23 02:36:05 +00:00
384 lines
12 KiB
TypeScript
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/johannesjo/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}`,
|
|
];
|
|
log(`🕘 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
|
|
function createMainWin(): void {
|
|
mainWin = 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);
|
|
});
|
|
};
|