乐趣区

Electron-使用-nodenotifier-弹窗提示

背景介绍

使用 Electron 做跨平台客户端开发时,无论是开发工具类的应用,还是聊天类的应用,消息提醒都是一个常见的需求。

Electron 提供了通知的方式。可以直接使用 HTML5 Notification API 来发送通知。

node-notifier 是一个跨平台的 Node.js 跨平台消息发送模块,本文介绍 Electron 集成 node-notifier 的过程以及遇到的问题和解决办法。并且因为在 Windows 下使用没有遇到什么大的问题,所以以下内容主要针对 Mac 系统。

环境搭建

  1. 使用 electron-webpack-quick-start 初始化项目 electron-notifier
  2. 添加 node-notifier 依赖:yarn add node-notifier
  3. demo 代码

    // main/index.js
    import {ipcMain} from "electron"
    const notifier = require("node-notifier");
    
    ipcMain.on("notify", (evt, data) => {const { title, message} = data;
        console.log("notify:", title, message);
        notifier.notify(
          {
            title,
            message
          },
          (err, res) => {if (err) {console.log("error:", err);
            }
          }
        );
    });
    
    // renderer/index.js
    const {ipcRenderer} = require("electron");
    ipcRenderer.send("notify", {
      tite: "title",
      message: "This is a message"
    });
  4. 运行 yarn run dev 弹出消息提示:
  5. 运行 yarn run dist 打包,得到打包好的 .dmg 文件,打开运行,发现并没有消息提示。

问题排查

  1. 因为需要记录打包后应用的日志信息,引入 electron-log

    // main/index.js
    const log = require("electron-log");
    // 重写 console 的方法,日志会在控制台输出,并且输出到本地文件。具体用法参考官方文档
    Object.assign(console, log.functions);
    // 获取本地日志文件路径
    console.log('日志文件:', log.transports.file.getFile())
  2. 重新执行打包命令,并运行打包生成的应用,得到对应的日志信息
  3. 通过控制台打印的文件路径或者 ~/Library/Logs/{app name}/{process type}.log 目录下找到 main.log, 查看其中的内容。可以看到有对应的报错信息:

    error: Error: Command failed: ... com.electron.electron-notifier.bcYDBs -title "title" -message "message" -timeout "10" -json "true" ...
    
    No Info.plist file in application bundle or no NSPrincipalClass in the Info.plist file, exiting
    ...
  4. 报错解读:

    • 前一段是使用传入的参数调用 node-notifier
    • 后一段是说:缺少 application bundleInfo.plist 文件或者 Info.plist 文件中 缺少 NSPrincipalClass 字段
  5. 问题排查

    • 通过“显示包内容”查看打包好的应用程序,发现 Info.plist 已经存在
    • 参考 apple 开发者文档 中的说明,NSPrincipalClass 字段定义了一个 bundle 的主类的名称。对于应用程序来说,缺省情况下这个名字就是应用程序的名字。这里我的理解是 package.json 中的 name 已经指定应用的名称
  6. 通过上面的分析,初步判断问题并不是提示所说的“缺少 NSPrincipalClass
  7. 下面是一通 google。经测试都不行

    • No Info.plist file in application bundle or no NSPrincipalClass in the Info.plist file on MacOS
    • Windows 10 not getting toasts
  8. 无奈,只能更多的去翻看官方文档

    • Electron 文档: 应用程序打包
    • node-notifier#within-electron-packaging
  9. 根据文档的解释,asar 打包以后不能运行 node-notifier 中的二进制文件,该目录需要在打包时进行 unpack 处理,原文如下:

    Due to the way asar works, you cannot execute a binary from within an `asar`. As a simple solution, when packaging the app into an asar please make sure you `--unpack` the `vendor/` folder of `node-notifier`, so the module still has access to the notification binaries.
  10. 因为项目中用到 electron-builder 进行打包,翻看其文档,添加以下代码: 通过 asarUnpack 指定不打包到 asar 的文件. 如果按照 Electron 官方文档的打包方式,可以使用:asar pack app app.asar --unpack ./node_modules/node-notifier/vendor/**

    // package.json
    "build": {
        "appId": "com.codinglife.electron-notifier",
        "asar": true,
        "asarUnpack": ["./node_modules/node-notifier/vendor/**"]
      }
  11. 重新运行打包命令,可以在 app.asar 同目录看到对应的 unpacked 目录。
  12. 运行打包好的应用,可完美弹出对应的消息提示
  13. 另外,也可以通过不打包 asar 的方式使用,配置如下:

    "build": {
        "appId": "com.codinglife.electron-notifier",
        "asar": false
    }
  14. 打包后的应用中,源文件全部放在 app 目录下。

总结

  1. asar 是一种压缩文件,Electron 使用它来加快 require 代码的速度和隐藏源代码。Electron 实现了部分的 Node.js API 方法来操作 asar 文件,但也存在以下几个问题:

    • asar 中的文件是只读的
    • 部分 API 需要解压到临时目录才能使用其中的文件,增加了开销
    • 不能执行其中的二进制文件
  2. 多翻文档,多 google
  3. electron-builder 有对 Electron 应用打包,配置,升级等操作的处理,解决了大部分打包,升级安装过程的问题,强烈安利

参考文档

  • electron-webpack-quick-start
  • node-notifier
  • electron-log
  • apple 开发者文档
  • electron-builder
退出移动版