pc网页端调起客户端应用的那些事--electron

2023-12-19 21:20

本文主要是介绍pc网页端调起客户端应用的那些事--electron,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Date: 2020.04.11

前言

 在当下大前端的趋势之下,我还是跟随大流跳入了electron 开发的坑。不过这个也让我不再局限于网页业务的开发,并对大前端所代表的的内容有了新的认识。如果是在去年听到大前端这个词,我也只是只知其名不懂其义。

 为了开发客户端的新业务,这三个月我接触了很多新知识,从基本的业务开发,electron 的构建配置和打包,针对 mac 应用的签名,公证和盖章, pc 客户端的ota升级,再到 jenkins 自动化构建配置,CI集成,再到现在的pc网页端调起客户端应用等业务。这些内容提高了还是前端切图仔的我的眼界【捂脸.jpeg】

 不过,这个三个月一直忙于业务开发,学到了很多内容,一直没有进行过总结,这个让我觉得很害怕,害怕学到的新知识会被忘掉,因此,今天还是赶紧针对这些知识点做一下部分总结。今天先整理一下pc网页端调起客户端应用的知识点。

正文

背景:今天要讨论的是在网页端打开客户端页面的功能,也许大家比较少接触过在 pc 端调起 pc 客户端的应用的功能;但是在移动端网页调起手机应用的操作,大家或多或少有接触过;其实无论在 pc 还是网页;它们实现调起客户端的原理都是类似的。接下来我会分享一下 pc 网端调起 eletron 客户端的那些事。

part 1: 如何在浏览器调起基于electron 开发的应用(原理)

  1. 基于自定义的伪协议(URL Scheme 协议),实现应用的调起
  2. URL Scheme 协议【理解为用于实现在浏览器跳转应用的协议就可以啦】
mac

配置 URL Schemes,往info.plist文件里配置好scheme。

Mac os上浏览器启动本地app主要是通过注册 URL Schemes,每个打包好的app文件中都会存在一个info.plist文件;

mac 用户可以尝试在配置了URL Schemes的应用中,查看已经安装好的应用的 URL Schemes 配置,步骤如下:

  1. 在应用程序中,找到对应的应用
  2. 右键显示包内容
  3. 打开 contents 文件
  4. 点击查看 Info.plist 找到 URL types – URL Schemes;
  5. 再通过浏览器打开输入该协议即可实现在浏览器调起应用。

以 xcode 为例:
6. 打开 xcode 的 Info.plist 文件。可看到其中一个 URL Schemes 为 xcode.
在这里插入图片描述
7. 再打开浏览器,在地址栏输入xcode://,即可调起 xcode 应用。
在这里插入图片描述
在这里插入图片描述

window

window 的调起类似;

只不过不是往info.plist文件里配置scheme,而是通过往注册表的 HKCR (HKEY_CALSSES_ROOT) 目录下添加一条记录来完成该协议的注册。

示例:在注册表的HKEY_CALSSES_ROOT目录下,添加一个xcode 字段,并将值指向 xcode 的 exe 包的路径;即可实现如上面 mac 的方式在浏览器调起应用。

暂时没有window 电脑在身边,使用了网上一张图作为例子,侵删
在这里插入图片描述

part 2: 那么如何在开发应用的时候定义好协议呢(怎么做)

其实关于 electron 应用协议的注册,electron 已经有提供相应的 api 啦 (app.setAsDefaultProtocolClient),我们可以直接使用。在进行 app 的相关事件监听和处理时,同时进行协议的注册。

// 示例1:比如说想通过在浏览器输入 “协议://”,就可以实现调起该应用,则实现如下
app.setAsDefaultProtocolClient("你的协议");// 示例2:比如说想通过在浏览器输入 “xcode://”,按照下面进行协议的注册,就可以实现调起该应用
app.setAsDefaultProtocolClient("xcode");

另外,还需要在 package.json 的 build 字段中的 protocols 对象进行 schemes 配置。

在mac中,该配置字段主要是用在将 schemes的值打包进 Info.plist 文件的URL Schemes。【如果没有进行这个配置,在第一次安装应用但未启动,直接通过网页端调起该应用,好像是无效的,除非已手动打开过一次该应用,主要是因为app.setAsDefaultProtocolClient(“你的协议”)这个操作是写在主进程里面,需要执行一遍应用的逻辑,即启动过应用,才能完成该协议的注册】。

window 只需要使用 app.setAsDefaultProtocolClient 在注册表进行注册即可。

  • 查看协议是否设置成功
  1. 分别进行mac 和window的打包
  2. mac 查看包的 Info.plist (URL types – URL Schemes)
  3. window 查看注册表
  • 调用方式

可以尝试在浏览器直接打开 注册的协议://,就可以完成浏览器的调起

part 3: 调起应用时触发的钩子函数

  1. mac 通过 app.on(‘open-url’, fun)监听
  2. window 通过 app.on(“second-instance”,fun)监听

mac 和 window有区别,开发和生产有区别

macOS 下通过URL Schemes协议调起应用时

主实例会通过 open-url 事件监听应用被调起的事件

// 网页进行应用的调起后,会触发该事件
app.on('open-url', (event, urlStr) => {console.log("mac 准备执行网页端调起客户端逻辑");handleUrlFromWeb(urlStr); // 对 url 执行的处理逻辑_handleBeforeOpenWindowByWeb(); // 利用相应的参数,打开其他页面(这个是我这边的需求用到),代码略
});
  • macOS中,打开协议的参数的处理
// 进行处理网页传来 url 参数,参数自定义,以下为示例
// 示例调起应用的 url 为(应用URL Schemes为youapp):youapp:?name=名字&type=类型&shareId=1585876954860136091
const handleUrlFromWeb = (urlStr) => {const urlObj = new URL(urlStr);const { searchParams } = urlObj;const shareId = searchParams.get('shareId');const name = searchParams.get('name');const type = searchParams.get('type');/** code这部分代码为使用这些参数执行相应的逻辑*/
};
window 下通过URL Schemes协议调起应用时

分两种情况

  1. 应用处于打开状态,会触发 second-instance 事件并接收这个 URL。

  2. 应用处于未打开状态,在网页端通过浏览器调起应用之后不会触发 second-instance 事件;这个时候需要主动判断应用是否是从网页端调起,并主动触发 second-instance 事件;

在 window 里面判断是否是从网页端的标准:如果是通过url schema启动,其启动参数会超过1个

以下为在应用 ready 之后,判断是否是在 window 下主动调起客户端,并主动触发 second-instance 事件的实现逻辑;

/*** 添加对协议链接的处理, 用于实现网页调起客户端逻辑*/
// 当应用启动完成后,主动判断应用是否是从网页中调起
const _handleAfterReady = () => {// windows如果是通过url schema启动则发出时间处理// 启动参数超过1个才可能是通过url schema启动if (process.argv.length > 1) {if (!app.isReady()) {app.once("browser-window-created", () => {// app 未打开时,通过 open-url打开 app,此时可能还没 ready,需要延迟发送事件// 此段ready延迟无法触发 service/app/ open-url 处理,因为saga初始化需要时间app.emit("second-instance", null, process.argv);});} else {app.emit("second-instance", null, process.argv);}}
};

触发 second-instance 事件之后的具体逻辑与 mac 类似,不过针对获取到的参数的处理方式还是存在一些差异

app.on('second-instance', async (event, argv) => {// Windows 下通过协议URL启动时,URL会作为参数,所以需要在这个事件里处理if (process.platform === 'win32') {console.log("window 准备执行网页端调起客户端逻辑");handleArgvFromWeb(argv);_handleOpenWindowByWeb();}
});

处理window下的参数

const PROTOCOL = '注册的协议名'; // 用户自定义// window 系统中执行网页调起应用时,处理协议传入的参数
const handleArgvFromWeb = (argv) => {const prefix = `${PROTOCOL}:`;// 开发阶段,跳过前两个参数(`electron.exe .`)// 打包后,跳过第一个参数(`myapp.exe`)const offset = app.isPackaged ? 1 : 2;const url = argv.find((arg, i) => i >= offset && arg.startsWith(prefix));if (url) handleUrlFromWeb(url);
};

补充: 从网页调起客户端时,在macos 中,既会触发 open-url 事件,也会触发 second-instance 事件;所以在写 second-instance事件 处理逻辑时,需要将 mac 排除掉,避免重复处理(通过判断 process.platform === ‘win32’【我们的应用只有Mac和window】)

part 4: 将被调起应用的可能存在的状态

这个部分主要是针对本人当下所做的需求去进行分析: 调起应用,根据是否登录,再决定是否打开新窗口并执行相应的业务逻辑(可以打开多个窗口)。按需浏览,业务不同可以考虑跳过该part,直接看下一部分。

  1. 已打开,已登录

这个时候,直接在浏览器通过协议打开即可

  1. 已打开,未登录

浏览器通过协议打开,显示登录窗口,登录成功,显示主窗口个,并打开对应功能窗口(要注意多窗口的处理,用户有可能在多个网页中调起了应用)

  1. 未打开,已登录 和 未打开,未登录
    如果只是使用 app.setAsDefaultProtocolClient(“你的协议”)进行 scheme 协议的注册;那么刚安装完应用,首次需要用户主动打开应用,才能完成 scheme 协议的注册;

如果想要在应用刚安装完成,不要求用户首次主动打开应用,也支持scheme 协议(通过网页调起)调起应用,则需要做一下配置

  • mac 应用,在package.json 的 build 字段中的 protocols 对象进行 schemes 配置即可。打包完成后,会在应用的 info.plist 看到相应的协议配置

  • window 应用,在需要通过在 package.json 中的 nsis 字段中配置相应的安装或卸载相关的脚本路径;配置应用在安装时,主动往注册表写入协议;
    在这里插入图片描述

  • 浏览器判断 scheme 协议是否存在(判断本地是否有安装应用,没安装跳下载地址)

part 5: 关于卸载的那些事

mac 卸载存在的坑:暂时没有发现

macOS 相关的卸载会比较简单,只在卸载应用即可完成卸载

window 卸载存在的坑:electron 开发的应用已完成卸载,但在注册表通过app.setAsDefaultProtocolClient 注册的协议没有删除
问题

但是针对 window 的应用的卸载后会有坑,直接通过 app.setAsDefaultProtocolClient(“你的协议”)可以完成协议的写入,但是应用却不会在卸载时主动删除注册表的协议;

这个时候就会存在一个坑,用户在浏览器再输入相应协议进行应用的调起时,浏览器依旧会弹出相应的提示弹窗(这个图是在mac 截的,window类似),点击打开应用,弹窗会被收起,但实际上并不会打开应用,因为应用已经被卸载,只是注册表的记录没被删除,导致依旧会出现弹窗。
在这里插入图片描述
这种情况其实并不会影响正常功能的使用了,但是测试人员说这个不是很美观。好吧,不美观,就再处理一下。

解决

这种情况应该就应用卸载的逻辑做一下配置才行。还记得上一下小部分,有说过可以通过 nsis 去进行应用的安装和卸载的相关配置(在我们应用中,window打包使用的是 electron-builder);

在 electron-builder 的说明文档中 nsis 模块的部分,有提供8个配置入口(customHeader, preInit, customInit, customUnInit, customInstall, customUnInstall, customRemoveFiles, customInstallMode),我将它们理解为插槽,用于针对安装和卸载去进行定制化配置

使用示例如下:

!macro customHeader!system "echo '' > ${BUILD_RESOURCES_DIR}/customHeader"
!macroend!macro preInit; This macro is inserted at the beginning of the NSIS .OnInit callback!system "echo '' > ${BUILD_RESOURCES_DIR}/preInit"
!macroend!macro customInit!system "echo '' > ${BUILD_RESOURCES_DIR}/customInit"
!macroend!macro customInstall!system "echo '' > ${BUILD_RESOURCES_DIR}/customInstall"
!macroend!macro customInstallMode# set $isForceMachineInstall or $isForceCurrentInstall # to enforce one or the other modes.
!macroend

我们的需求是在应用卸载时主动删除注册表的相关的协议,所以 installer.nsh 逻辑如下:

!macro customUnInstallDeleteRegKey HKCR "你注册的协议"
!macroend 

以上即在应用卸载时完成协议的删除

其他补充
  • HKCR 表示的 HKEY_CLASSES_ROOT 的在注册表的简写 [即注册表的路径]

  • nsis钩子详细说明:https://www.electron.build/configuration/nsis
    在这里插入图片描述

  • 参考写法(卸载程序):https://stackoom.com/question/3C5eL/nsis%E5%8D%B8%E8%BD%BD%E7%A8%8B%E5%BA%8F%E6%9C%AA%E5%88%A0%E9%99%A4%E7%94%B5%E5%AD%90%E5%BA%94%E7%94%A8%E7%A8%8B%E5%BA%8F%E7%9A%84%E6%B3%A8%E5%86%8C%E8%A1%A8-nsh%E8%84%9A%E6%9C%AC

这篇关于pc网页端调起客户端应用的那些事--electron的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/513747

相关文章

MobaXterm远程登录工具功能与应用小结

《MobaXterm远程登录工具功能与应用小结》MobaXterm是一款功能强大的远程终端软件,主要支持SSH登录,拥有多种远程协议,实现跨平台访问,它包括多会话管理、本地命令行执行、图形化界面集成和... 目录1. 远程终端软件概述1.1 远程终端软件的定义与用途1.2 远程终端软件的关键特性2. 支持的

使用Java实现获取客户端IP地址

《使用Java实现获取客户端IP地址》这篇文章主要为大家详细介绍了如何使用Java实现获取客户端IP地址,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 首先是获取 IP,直接上代码import org.springframework.web.context.request.Requ

5分钟获取deepseek api并搭建简易问答应用

《5分钟获取deepseekapi并搭建简易问答应用》本文主要介绍了5分钟获取deepseekapi并搭建简易问答应用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1、获取api2、获取base_url和chat_model3、配置模型参数方法一:终端中临时将加

JavaScript中的isTrusted属性及其应用场景详解

《JavaScript中的isTrusted属性及其应用场景详解》在现代Web开发中,JavaScript是构建交互式应用的核心语言,随着前端技术的不断发展,开发者需要处理越来越多的复杂场景,例如事件... 目录引言一、问题背景二、isTrusted 属性的来源与作用1. isTrusted 的定义2. 为

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex

Python手搓邮件发送客户端

《Python手搓邮件发送客户端》这篇文章主要为大家详细介绍了如何使用Python手搓邮件发送客户端,支持发送邮件,附件,定时发送以及个性化邮件正文,感兴趣的可以了解下... 目录1. 简介2.主要功能2.1.邮件发送功能2.2.个性签名功能2.3.定时发送功能2. 4.附件管理2.5.配置加载功能2.6.

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys