Newer
Older
dxCard-admin / electron / utils / tray.ts
YFJ on 23 Sep 6 KB 项目推送
// 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);
        }
      },
    },
  ]);
}