202 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
		
		
			
		
	
	
			202 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
| 
								 | 
							
								// tray = 系统托盘
							 | 
						|||
| 
								 | 
							
								import path from 'path';
							 | 
						|||
| 
								 | 
							
								import { Tray, Menu, app, dialog, nativeImage, BrowserWindow, Notification, ipcMain } from 'electron';
							 | 
						|||
| 
								 | 
							
								import type { IpcMainInvokeEvent } from 'electron';
							 | 
						|||
| 
								 | 
							
								import {_PATHS} from '../paths';
							 | 
						|||
| 
								 | 
							
								import {$env, isDev} from '../env';
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								const TrayIcons = {
							 | 
						|||
| 
								 | 
							
								  // update-begin--author:liaozhiyang---date:20250725---for:【JHHB-13】桌面应用消息通知
							 | 
						|||
| 
								 | 
							
								  normal: nativeImage.createFromPath(
							 | 
						|||
| 
								 | 
							
								    process.platform === 'win32'
							 | 
						|||
| 
								 | 
							
								      ? path.join(_PATHS.publicRoot, 'logo.png')
							 | 
						|||
| 
								 | 
							
								      : path.join(_PATHS.electronRoot, './icons/mac/tray-icon.png').replace(/[\\/]dist[\\/]/, '/')
							 | 
						|||
| 
								 | 
							
								  ),
							 | 
						|||
| 
								 | 
							
								  // update-end--author:liaozhiyang---date:20250725---for:【JHHB-13】桌面应用消息通知
							 | 
						|||
| 
								 | 
							
								  empty: nativeImage.createEmpty(),
							 | 
						|||
| 
								 | 
							
								};
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								// 创建托盘图标
							 | 
						|||
| 
								 | 
							
								export function createTray(win: BrowserWindow) {
							 | 
						|||
| 
								 | 
							
								  const tray = new Tray(TrayIcons.normal);
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  const TrayUtils = useTray(tray, win);
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  tray.setToolTip($env.VITE_GLOB_APP_TITLE! + (isDev ? ' (开发环境)' : ''));
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  // 左键托盘图标显示主窗口
							 | 
						|||
| 
								 | 
							
								  tray.on('click', () => TrayUtils.showMainWindow());
							 | 
						|||
| 
								 | 
							
								  // 右键托盘图标显示托盘菜单
							 | 
						|||
| 
								 | 
							
								  tray.on('right-click', () => showTrayContextMenu());
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  function showTrayContextMenu() {
							 | 
						|||
| 
								 | 
							
								    const trayContextMenu = getTrayMenus(win, TrayUtils);
							 | 
						|||
| 
								 | 
							
								    // 弹出托盘菜单,不使用 setContextMenu 方法是因为要实时更新菜单内容
							 | 
						|||
| 
								 | 
							
								    tray.popUpContextMenu(trayContextMenu);
							 | 
						|||
| 
								 | 
							
								  }
							 | 
						|||
| 
								 | 
							
								}
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								export function useTray(tray: Tray, win: BrowserWindow) {
							 | 
						|||
| 
								 | 
							
								  let isBlinking = false;
							 | 
						|||
| 
								 | 
							
								  let blinkTimer: NodeJS.Timeout | null = null;
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  function showMainWindow() {
							 | 
						|||
| 
								 | 
							
								    win.show();
							 | 
						|||
| 
								 | 
							
								  }
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  // 开始闪动
							 | 
						|||
| 
								 | 
							
								  function startBlink() {
							 | 
						|||
| 
								 | 
							
								    isBlinking = true;
							 | 
						|||
| 
								 | 
							
								    tray.setImage(TrayIcons.empty);
							 | 
						|||
| 
								 | 
							
								    blinkTimer = setTimeout(() => {
							 | 
						|||
| 
								 | 
							
								      tray.setImage(TrayIcons.normal);
							 | 
						|||
| 
								 | 
							
								      setTimeout(() => {
							 | 
						|||
| 
								 | 
							
								        if (isBlinking) {
							 | 
						|||
| 
								 | 
							
								          startBlink();
							 | 
						|||
| 
								 | 
							
								        }
							 | 
						|||
| 
								 | 
							
								      }, 500);
							 | 
						|||
| 
								 | 
							
								    }, 500);
							 | 
						|||
| 
								 | 
							
								  }
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  // 结束闪动
							 | 
						|||
| 
								 | 
							
								  function stopBlink() {
							 | 
						|||
| 
								 | 
							
								    isBlinking = false;
							 | 
						|||
| 
								 | 
							
								    if (blinkTimer) {
							 | 
						|||
| 
								 | 
							
								      clearTimeout(blinkTimer);
							 | 
						|||
| 
								 | 
							
								      blinkTimer = null;
							 | 
						|||
| 
								 | 
							
								    }
							 | 
						|||
| 
								 | 
							
								    tray.setImage(TrayIcons.normal);
							 | 
						|||
| 
								 | 
							
								  }
							 | 
						|||
| 
								 | 
							
								  ipcMain.on('tray-flash', (event: IpcMainInvokeEvent) => {
							 | 
						|||
| 
								 | 
							
								    // 仅在 Windows 系统中闪烁
							 | 
						|||
| 
								 | 
							
								    if (process.platform === 'win32') {
							 | 
						|||
| 
								 | 
							
								      startBlink();
							 | 
						|||
| 
								 | 
							
								    }
							 | 
						|||
| 
								 | 
							
								  });
							 | 
						|||
| 
								 | 
							
								  ipcMain.on('tray-flash-stop', (event: IpcMainInvokeEvent) => {
							 | 
						|||
| 
								 | 
							
								    // 仅在 Windows 系统中停止闪烁
							 | 
						|||
| 
								 | 
							
								    if (process.platform === 'win32') {
							 | 
						|||
| 
								 | 
							
								      stopBlink();
							 | 
						|||
| 
								 | 
							
								    }
							 | 
						|||
| 
								 | 
							
								  });
							 | 
						|||
| 
								 | 
							
								  win.on('focus', () => {
							 | 
						|||
| 
								 | 
							
								    stopBlink();
							 | 
						|||
| 
								 | 
							
								  });
							 | 
						|||
| 
								 | 
							
								  // 发送桌面通知
							 | 
						|||
| 
								 | 
							
								  function sendDesktopNotice() {
							 | 
						|||
| 
								 | 
							
								    // 判断是否支持桌面通知
							 | 
						|||
| 
								 | 
							
								    if (!Notification.isSupported()) {
							 | 
						|||
| 
								 | 
							
								      // todo 实际开发中不需要提示,直接返回或者换一种提示方式
							 | 
						|||
| 
								 | 
							
								      dialog.showMessageBoxSync(win, {
							 | 
						|||
| 
								 | 
							
								        type: 'error',
							 | 
						|||
| 
								 | 
							
								        title: '错误',
							 | 
						|||
| 
								 | 
							
								        message: '当前系统不支持桌面通知',
							 | 
						|||
| 
								 | 
							
								      });
							 | 
						|||
| 
								 | 
							
								      return;
							 | 
						|||
| 
								 | 
							
								    }
							 | 
						|||
| 
								 | 
							
								    const ins = new Notification({
							 | 
						|||
| 
								 | 
							
								      title: '通知标题',
							 | 
						|||
| 
								 | 
							
								      body: '通知内容第一行\n通知内容第二行',
							 | 
						|||
| 
								 | 
							
								      // icon: TrayIcons.normal.resize({width: 32, height: 32}),
							 | 
						|||
| 
								 | 
							
								    });
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								    ins.on('click', () => {
							 | 
						|||
| 
								 | 
							
								      dialog.showMessageBoxSync(win, {
							 | 
						|||
| 
								 | 
							
								        type: 'info',
							 | 
						|||
| 
								 | 
							
								        title: '提示',
							 | 
						|||
| 
								 | 
							
								        message: '通知被点击',
							 | 
						|||
| 
								 | 
							
								      });
							 | 
						|||
| 
								 | 
							
								    });
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								    ins.show();
							 | 
						|||
| 
								 | 
							
								  }
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  return {
							 | 
						|||
| 
								 | 
							
								    showMainWindow,
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								    startBlink,
							 | 
						|||
| 
								 | 
							
								    stopBlink,
							 | 
						|||
| 
								 | 
							
								    isBlinking: () => isBlinking,
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								    sendDesktopNotice,
							 | 
						|||
| 
								 | 
							
								  };
							 | 
						|||
| 
								 | 
							
								}
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								const MenuIcon = {
							 | 
						|||
| 
								 | 
							
								  exit: nativeImage
							 | 
						|||
| 
								 | 
							
								    .createFromDataURL(
							 | 
						|||
| 
								 | 
							
								      ''
							 | 
						|||
| 
								 | 
							
								    )
							 | 
						|||
| 
								 | 
							
								    .resize({
							 | 
						|||
| 
								 | 
							
								      width: 16,
							 | 
						|||
| 
								 | 
							
								      height: 16,
							 | 
						|||
| 
								 | 
							
								    }),
							 | 
						|||
| 
								 | 
							
								};
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								// 设置托盘菜单
							 | 
						|||
| 
								 | 
							
								function getTrayMenus(win: BrowserWindow, TrayUtils: ReturnType<typeof useTray>) {
							 | 
						|||
| 
								 | 
							
								  const {startBlink, stopBlink, sendDesktopNotice} = TrayUtils;
							 | 
						|||
| 
								 | 
							
								  const isBlinking = TrayUtils.isBlinking();
							 | 
						|||
| 
								 | 
							
								
							 | 
						|||
| 
								 | 
							
								  return Menu.buildFromTemplate([
							 | 
						|||
| 
								 | 
							
								    ...(isDev
							 | 
						|||
| 
								 | 
							
								      ? [
							 | 
						|||
| 
								 | 
							
								        {
							 | 
						|||
| 
								 | 
							
								          label: '开发工具',
							 | 
						|||
| 
								 | 
							
								          submenu: [
							 | 
						|||
| 
								 | 
							
								            {
							 | 
						|||
| 
								 | 
							
								              label: '以下菜单仅显示在开发环境',
							 | 
						|||
| 
								 | 
							
								              sublabel: '当前为开发环境',
							 | 
						|||
| 
								 | 
							
								              enabled: false,
							 | 
						|||
| 
								 | 
							
								            },
							 | 
						|||
| 
								 | 
							
								            {type: 'separator'},
							 | 
						|||
| 
								 | 
							
								            {
							 | 
						|||
| 
								 | 
							
								              label: '切换 DevTools',
							 | 
						|||
| 
								 | 
							
								              click: () => win.webContents.toggleDevTools(),
							 | 
						|||
| 
								 | 
							
								            },
							 | 
						|||
| 
								 | 
							
								            {
							 | 
						|||
| 
								 | 
							
								              label: `托盘图标${isBlinking ? '停止' : '开始'}闪烁`,
							 | 
						|||
| 
								 | 
							
								              sublabel: '模拟新消息提醒',
							 | 
						|||
| 
								 | 
							
								              click: () => (isBlinking ? stopBlink() : startBlink()),
							 | 
						|||
| 
								 | 
							
								            },
							 | 
						|||
| 
								 | 
							
								            {
							 | 
						|||
| 
								 | 
							
								              label: '发送桌面通知示例',
							 | 
						|||
| 
								 | 
							
								              click: () => sendDesktopNotice(),
							 | 
						|||
| 
								 | 
							
								            },
							 | 
						|||
| 
								 | 
							
								          ],
							 | 
						|||
| 
								 | 
							
								        },
							 | 
						|||
| 
								 | 
							
								        {type: 'separator'},
							 | 
						|||
| 
								 | 
							
								      ]
							 | 
						|||
| 
								 | 
							
								      : ([] as any)),
							 | 
						|||
| 
								 | 
							
								    {
							 | 
						|||
| 
								 | 
							
								      label: '显示主窗口',
							 | 
						|||
| 
								 | 
							
								      // 文件图标
							 | 
						|||
| 
								 | 
							
								      icon: TrayIcons.normal.resize({width: 16, height: 16}),
							 | 
						|||
| 
								 | 
							
								      click: () => win.show(),
							 | 
						|||
| 
								 | 
							
								    },
							 | 
						|||
| 
								 | 
							
								    {type: 'separator'},
							 | 
						|||
| 
								 | 
							
								    {
							 | 
						|||
| 
								 | 
							
								      label: '退出',
							 | 
						|||
| 
								 | 
							
								      // base64图标
							 | 
						|||
| 
								 | 
							
								      icon: MenuIcon.exit,
							 | 
						|||
| 
								 | 
							
								      click: () => {
							 | 
						|||
| 
								 | 
							
								        // 弹出是否确认退出提示框
							 | 
						|||
| 
								 | 
							
								        const choice = dialog.showMessageBoxSync(win, {
							 | 
						|||
| 
								 | 
							
								          type: 'question',
							 | 
						|||
| 
								 | 
							
								          title: '提示',
							 | 
						|||
| 
								 | 
							
								          message: '确定要退出应用吗?',
							 | 
						|||
| 
								 | 
							
								          buttons: ['退出', '取消'],
							 | 
						|||
| 
								 | 
							
								          defaultId: 1,
							 | 
						|||
| 
								 | 
							
								          cancelId: 1,
							 | 
						|||
| 
								 | 
							
								          noLink: true,
							 | 
						|||
| 
								 | 
							
								        });
							 | 
						|||
| 
								 | 
							
								        // 用户选择了退出,直接 exit
							 | 
						|||
| 
								 | 
							
								        if (choice === 0) {
							 | 
						|||
| 
								 | 
							
								          // global.isQuitting = true;
							 | 
						|||
| 
								 | 
							
								          app.exit(0);
							 | 
						|||
| 
								 | 
							
								        }
							 | 
						|||
| 
								 | 
							
								      },
							 | 
						|||
| 
								 | 
							
								    },
							 | 
						|||
| 
								 | 
							
								  ]);
							 | 
						|||
| 
								 | 
							
								}
							 |