electron 窗口渲染的内容逻辑运行在renderer进程,窗口管理与应用管理逻辑运行在main进程,窗口之间的渲染内容的通讯无法直接进行,在多窗口应用中尤为突出,需要通过main进程进行中转。
electron-wpc通过封装窗口间通信过程,降低与业务耦合度,便捷化实现Electron窗口间通讯。
ps :觉得ok给个star鼓励下
功能|效果
- 双窗口间(renderer进程)消息发送及处理结果的获取;
a. 支持多对一窗口的半双工消息通信。
b. 基于EventEmitter+Promise特性实现窗口间消息的发送及结果主动返回;
c. 消息处理者(Provider Renderer)可主动向发送者(Resolver Renderer)抛出异常。
d. 支持消息的超时处理。
e. 集成简单。 - 多窗口间(renderer进程)广播通信;
a.支持一对多的单工消息通信。
b.支持广播成功或失败的消息回调。
c.支持广播前在main进程中对数据加工。
c.集成简单。
安装
npm install electron-wpc
运行示例
示例中的执行过程及结果在窗口控制台中打印。
npm install
npm run example
集成与应用
双窗口间(renderer进程)消息发送及处理结果的获取:
- 在main进程中注册Provider Window(作为内容提供者),并绑定TAG。(单个窗口可根据作用类型不同设置不同的tag)
const { registProviderWindow, unRegistProviderWindow } = require('electron-wpc');
//...
const winProvider = new BrowserWindow();
const TAG = 'tag_for_win_provider';
//...
registProviderWindow(winProvider, TAG);
//...销毁
winProvider.on('close',()=>{
unRegistProviderWindow(TAG);
})
- 在Provider Window 的Renderer进程中,注册相应的监听事件。
const { WPCProviderDelegate } = require('electron-wpc');
const providerDelegate = new WPCProviderDelegate();
//example :更新用户表的任务
providerDelegate.on('update_user_table', (resolve, reject, args) => {
//模拟耗时操作
setTimeout(() => {
const replyArgs = { result: `update user id:${args.id} success!` };
resolve(replyArgs);
}, 1000);
});
- 在resolver Window的Renderer进程中,在相应业务逻辑处发送消息,等待执行结果。
const { WPCResolverDelegate } = require('electron-wpc');
const TAG = 'tag_for_win_provider';//需填入目标Provider的tagId.
const resolverDelegate = new WPCResolverDelegate(TAG);//这里需要传入tag以指定Provider
// (可选)可设置超时时间,如果超时时间内没有返回结果,则将catch timeout error.
resolverDelegate.setTimeOut(6 * 1000);//default is 5s.
//发送消息
resolverDelegate.send('update_user_table')
.then(res => {
//结果返回
console.log('<-update_user_table res',res);
})
.catch(e => {
//处理失败
console.error('<-receive error msg:', e)
});
多窗口间(renderer进程)广播通信
- 在main进程注册广播通信事件id
const {registBrodcastTransfer,unRegistBrodcastTransfer} = require('electron-wpc');
//注册多窗口接收广播
registBrodcastTransfer("broadcast_key1");
//如消息数据需要在main进程处理,则可对数据进行加工
registBrodcastTransfer("broadcast_key2",(resolve,reject,args)=>{
// 对args进行加工
args.id = args.id+'xxx';
resolve(args);
});
//...销毁
app.on('close',()=>{
unRegistBrodcastTransfer("broadcast_key1","broadcast_key2");
})
- renderer进程发送广播至其他所有renderer进程
ipcRenderer.send("broadcast_key1", { id: 666 });
ipcRenderer.send("broadcast_key2", 'abcd');
// (可选)监听广播发送的结果。(需要在原key的基础上添加-reply作为新key)
ipcRenderer.once("broadcast_key2" + "-reply", (e, a) => {
console.log('=>send broadcast result:', a);
});
- renderer进程注册广播接收者
function exampleBroadcastReceiver() {
ipcRenderer.on('broadcast_key1', (event, args) => {
console.log('接收到其他窗口的广播1:', args);
});
ipcRenderer.on('broadcast_key2', (event, args) => {
console.log('接收到其他窗口的广播2:', args);
});
}