element-desktop/src/tray.ts

148 lines
4.9 KiB
TypeScript
Raw Normal View History

/*
2024-09-06 18:56:18 +02:00
Copyright 2024 New Vector Ltd.
Copyright 2017 Karl Glatz <karl@glatz.biz>
Copyright 2017 OpenMarket Ltd
2024-09-06 18:56:18 +02:00
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
import { app, Tray, Menu, nativeImage } from "electron";
import pngToIco from "png-to-ico";
import path from "path";
import fs from "fs";
import { v5 as uuidv5 } from "uuid";
2022-01-10 13:57:33 +01:00
import { _t } from "./language-helper";
2022-10-13 13:42:33 +02:00
let trayIcon: Tray | null = null;
export function hasTray(): boolean {
2022-12-15 12:00:58 +01:00
return trayIcon !== null;
}
export function destroy(): void {
if (trayIcon) {
trayIcon.destroy();
trayIcon = null;
}
}
export function isMonochrome(): boolean {
return global.store.get("monochromeIcon", process.platform === "linux");
}
export function refreshIcon(): void {
const monochrome = isMonochrome();
if (monochrome) {
trayIcon?.setImage(nativeImage.createFromPath(global.trayConfig.monochrome_icon_path));
} else {
trayIcon?.setImage(nativeImage.createFromPath(global.trayConfig.color_icon_path));
}
}
function toggleWin(): void {
if (global.mainWindow?.isVisible() && !global.mainWindow.isMinimized() && global.mainWindow.isFocused()) {
global.mainWindow.hide();
} else {
if (global.mainWindow?.isMinimized()) global.mainWindow.restore();
if (!global.mainWindow?.isVisible()) global.mainWindow?.show();
global.mainWindow?.focus();
}
}
interface IConfig {
2024-07-24 22:06:26 +02:00
color_icon_path: string; // eslint-disable-line camelcase
monochrome_icon_path: string; // eslint-disable-line camelcase
brand: string;
}
function getUuid(): string {
// The uuid field is optional and only needed on unsigned Windows packages where the executable path changes
// The hardcoded uuid is an arbitrary v4 uuid generated on https://www.uuidgenerator.net/version4
return global.vectorConfig["uuid"] || "eba84003-e499-4563-8e9d-166e34b5cc25";
}
export function create(config: IConfig): void {
// no trays on darwin
2022-12-15 12:00:58 +01:00
if (process.platform === "darwin" || trayIcon) return;
2024-07-24 22:06:26 +02:00
const defaultIcon = nativeImage.createFromPath(
isMonochrome() ? config.monochrome_icon_path : config.color_icon_path,
);
let guid: string | undefined;
if (process.platform === "win32" && app.isPackaged) {
// Providing a GUID lets Windows be smarter about maintaining user's tray preferences
// https://github.com/electron/electron/pull/21891
// Ideally we would only specify it for signed packages but determining whether the app is signed sufficiently
// is non-trivial. So instead we have an escape hatch that unsigned packages can iterate the `uuid` in
// config.json to prevent Windows refusing GUID-reuse if their executable path changes.
guid = uuidv5(`${app.getName()}-${app.getPath("userData")}`, getUuid());
}
// Passing guid=undefined on Windows will cause it to throw `Error: Invalid GUID format`
// The type here is wrong, the param must be omitted, never undefined.
trayIcon = guid ? new Tray(defaultIcon, guid) : new Tray(defaultIcon);
trayIcon.setToolTip(config.brand);
initApplicationMenu();
2022-12-15 12:00:58 +01:00
trayIcon.on("click", toggleWin);
let lastFavicon: string | null = null;
2022-12-15 12:00:58 +01:00
global.mainWindow?.webContents.on("page-favicon-updated", async function (ev, favicons) {
if (!favicons || favicons.length <= 0 || !favicons[0].startsWith("data:")) {
if (lastFavicon !== null) {
global.mainWindow?.setIcon(defaultIcon);
2022-10-13 13:42:33 +02:00
trayIcon?.setImage(defaultIcon);
lastFavicon = null;
}
return;
}
// No need to change, shortcut
if (favicons[0] === lastFavicon) return;
lastFavicon = favicons[0];
let newFavicon = nativeImage.createFromDataURL(favicons[0]);
// Windows likes ico's too much.
2022-12-15 12:00:58 +01:00
if (process.platform === "win32") {
try {
2022-12-15 12:00:58 +01:00
const icoPath = path.join(app.getPath("temp"), "win32_element_icon.ico");
fs.writeFileSync(icoPath, await pngToIco(newFavicon.toPNG()));
newFavicon = nativeImage.createFromPath(icoPath);
} catch (e) {
console.error("Failed to make win32 ico", e);
}
}
2022-10-13 13:42:33 +02:00
trayIcon?.setImage(newFavicon);
global.mainWindow?.setIcon(newFavicon);
});
2022-12-15 12:00:58 +01:00
global.mainWindow?.webContents.on("page-title-updated", function (ev, title) {
2022-10-13 13:42:33 +02:00
trayIcon?.setToolTip(title);
});
}
export function initApplicationMenu(): void {
if (!trayIcon) {
return;
}
const contextMenu = Menu.buildFromTemplate([
{
label: _t("action|show_hide"),
click: toggleWin,
},
2022-12-15 12:00:58 +01:00
{ type: "separator" },
{
label: _t("action|quit"),
2022-12-15 12:00:58 +01:00
click: function (): void {
app.quit();
},
},
]);
trayIcon.setContextMenu(contextMenu);
}