-
Notifications
You must be signed in to change notification settings - Fork 157
/
Copy pathsystemTray.ts
104 lines (90 loc) · 3.06 KB
/
systemTray.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
// @ts-ignore
import getos from 'getos'
import path from 'path'
import { app, screen, BrowserWindow, Menu, KeyboardEvent, Rectangle, Tray as ElectronTray } from 'electron'
import { capitalize } from '../../resources/utils'
const isMacOS = process.platform === 'darwin'
let isUbuntu23OrGreater = false
if (process.platform === 'linux') {
try {
getos((error: Error, osInfo: any) => {
if (error) {
console.error('Could not determine Linux version', error)
} else {
if (osInfo.dist === 'Ubuntu' && osInfo.release) {
const majorVersion = parseInt(osInfo.release.split('.')[0], 10)
isUbuntu23OrGreater = majorVersion >= 23
}
}
})
} catch (error) {
console.error('Could not determine Linux version', error)
}
}
const delaySettingContextMenu = () => !isMacOS && !isUbuntu23OrGreater
export type SystemTrayEventHandlers = {
click: () => void
clickShow: () => void
clickHide: () => void
}
export class SystemTray {
private clickHandlers: SystemTrayEventHandlers
private electronTray?: ElectronTray
constructor(clickHandlers: SystemTrayEventHandlers) {
this.clickHandlers = clickHandlers
}
init(mainWindow: BrowserWindow) {
// Electron Tray can only be instantiated when the app is ready
this.electronTray = new ElectronTray(path.join(__dirname, isMacOS ? './IconTemplate.png' : './Icon.png'))
this.electronTray.on('click', (_event: KeyboardEvent, bounds: Rectangle) => {
const mainWindowBounds = mainWindow.getBounds()
const currentDisplay = screen.getDisplayMatching(bounds)
const trayClickDisplay = screen.getDisplayMatching(mainWindowBounds)
if (trayClickDisplay.id !== currentDisplay.id) {
this.setContextMenu('show', { switchScreen: true })
}
this.clickHandlers.click()
})
}
setContextMenu(
type: string,
{ displaySummonShortcut = false, accelerator = 'Alt+/', switchScreen = false }
) {
const separatorMenuItem = {
label: 'Frame',
click: () => {},
type: 'separator'
}
const menuItemLabelMap = {
hide: 'Dismiss',
show: 'Summon'
}
const label = menuItemLabelMap[type as keyof typeof menuItemLabelMap]
const eventName = `click${capitalize(type)}`
const actionMenuItem: Electron.MenuItemConstructorOptions = {
label,
click: () => this.clickHandlers[eventName as keyof typeof this.clickHandlers](),
toolTip: `${label} Frame`
}
const quitMenuItem = {
label: 'Quit',
click: () => app.quit()
}
if (displaySummonShortcut) {
actionMenuItem.accelerator = accelerator
actionMenuItem.registerAccelerator = false
}
const menu = Menu.buildFromTemplate([actionMenuItem, separatorMenuItem, quitMenuItem])
if (switchScreen) {
this.electronTray?.setContextMenu(menu)
} else {
setTimeout(() => this.electronTray?.setContextMenu(menu), delaySettingContextMenu() ? 200 : 0)
}
}
closeContextMenu() {
this.electronTray?.closeContextMenu()
}
setTitle(title: string) {
this.electronTray?.setTitle(title)
}
}