Electron

作者: hellomyshadow | 来源:发表于2019-04-14 22:33 被阅读0次

node GUI

1. NW.js和Electron都是基于NodeJs的GUI框架,使用HTML、CSS、JS来构建UI,处理与用户的交互
2. NW.js和Electron专注于开发桌面跨平台应用,而不是web服务器
3. NW.js的原称为Node-Webkit,目前由因特尔公司维护,而Electron由Github维护;
4. NW.js和Electron都使用了开源浏览器Chromium,并使用NodeJs访问系统、文件、网络...

Electron

1. Electron的原理:将Chromium和Node.js整合到一个运行时环境中,且可以跨平台打包;
    1. Chromium是一个谷歌开源浏览器,可以调用所有前端相关的API,且不需要考虑兼容性问题;
    2. 浏览器并不能访问原生的资源,但Node.Js的API支持在页面中和操作系统进行交互。
2. npm install -g electron:全局安装Electron
    1. 可以在Github上下载Electron的demo:electron-quick-start
    2. 进入demo目录,执行 cnpm install 安装依赖,执行 npm start 运行demo
3. electron-forge:相当于Electron的脚手架,可以快速创建、运行、打包项目;
    1. npm install -g electron-forge:全局安装
    2. electron-forge init myDemo:创建项目,默认使用 npm i 安装依赖;
    3. 如果执行失败,可以删除项目中的node_modules,执行 cnpm i,手动安装依赖;
4. 手动搭建项目:创建目录,在目录中创建main.js和index.html,执行cnpm init;
    1. main.js就是主进程,项目运行的入口文件;
    2. 一个应用只有一个主进程,但可以有多个渲染进程,一个窗口就表示一个渲染进程;
    3. electron .:运行项目。
5. 拖放打开文件
    <textarea id="container"></textarea>
    1. 调用H5的拖动API,获取拖放文件的路径
    let box = document.querySelector('#container')
    box.ondragenter=box.ondragover=box.ondragleave = (e) => {
        return false; --> 组织默认行为
    }
    box.ondrop=(e) => {
        e.preventDefault(); --> 阻止默认行为
        let path = e.dataTransfer.files[0].path; --> 获取文件的路径
    }
    2. 使用Node.js的fs模块读取文件,赋值给box.value:let fs = require('fs')
6. 相关模块
    1. Electron中的模块分为:主进程模块,渲染进程模块,公用模块;
    2. app:主进程模块,控制应用的生命周期,写法比较固定,注册固定的几个事件;
    3. BrowserWindow:主进程模块,窗口相关的模块;
    let win = BrowserWindow.getFocusedWindow();  -->获取当前窗口的对象
    win.center();  -->把窗口移动到屏幕中央
    4. remote:渲染进程的模块,用于间接调用部分主进程的模块,如BrowserWindow
    5. Menu:主进程的菜单模块,包括顶部菜单、右键菜单、系统托盘的右键菜单。
7. Menu菜单功能必须在窗口加载完成之后才能执行;
    app.on('ready', {
        ...... //创建窗口BrowserWindow
        require('./main/menu.js') //加载创建菜单的js模块
    })
    1. 虽然渲染进程也能通过remote模块调用Menu模块,实现自定义的顶部菜单,但可能会闪现默认
    的顶部菜单,所以还是建议在主进程中实现顶部菜单。

进程间通信IPC

1. 被Electron直接运行的脚本(package.json中指定的main节点)称为主进程,有且只有一个;
2. 用于展示的web界面都运行在一个独立的进程中,称为渲染进程;
3. 进程间通信的相关模块:ipcMain(主进程)、ipcRenderer(渲染进程)

主进程与渲染进程

1. 渲染进程向主进程发送异步消息:ipcRenderer.send('事件名1', 消息内容)
    1. 主进程中监听消息:ipcMain.on('事件名1', (event, arg) => { ... })
    2. 主进程给渲染进程回复消息:event.sender.send('事件名2', 消息内容)
    3. 渲染进程监听回复的消息:ipcRenderer.on('事件名2', (event, arg) => {})
2. 渲染进程向主进程发送同步消息:
    1. let msg = ipcRenderer.sendSync('事件名1', 消息内容)
    2. 与异步消息不同,主进程收到同步消息,必须回复,否则渲染进程会阻塞;
    ipcMain.on('事件名1', (event, arg) => {
        event.returnValue = 'sync reply' ---> 回复给渲染进程
    }) ----> sendSync()的返回值就是主进程回复的消息内容
3. 主进程把数据挂载到node的全局对象global上,渲染进程通过 getGlobal() 获取数据
    1. 主进程:global.username = 'abc';
    2. 渲染进程:remote.getGlobal('username');
    3. 这种方式污染了全局对象global,并不建议使用。

渲染进程与渲染进程

1. 渲染进程无法直接通信,最简单的方式是通过localStorage实现;
2. 通过BrowserWindow和webContents模块实现:
    1. webContents是一个事件发出者,负责渲染并控制网页,也是BrowserWindow的属性;
3. 渲染进程1向主进程发送消息,主进程创建渲染进程2,再向渲染进程2发送消息;
    ipcMain.on('事件名1', (event, arg) => { --> 主进程监听消息事件
        win = new BrowserWindow({width:400, height:300})
        win.loadURL(path.join('file:', __dirname, '../news.html'))
        win.webContents.on('did-finish-load', () => { --> 窗口加载完毕
            win.webContents.send('事件名2', arg) --> 向渲染进程2发送事件
        }) ---> 新的渲染进程中注册'事件名2'
    })
4. 渲染进程2向渲染进程1回复消息,则需要把渲染进程1的Id也发送给渲染进程2
    ipcMain.on('事件名1', (event, arg) => {
        //必须在新窗口加载之前获取当前窗口的Id,否则获取的新窗口的Id
        let winId = BrowserWindow.getFocusedWindow().id
        win = new BrowserWindow({width:400, height:300})
        win.loadURL(path.join('file:', __dirname, '../news.html'))
        win.webContents.on('did-finish-load', () => {
            win.webContents.send('事件名2', arg, winId)
        })
    })
5. 在新窗口中注册事件,向渲染进程1回复消息,在渲染进程1中监听回复事件
    ipcRenderer.on('事件名2', (event, arg, winId) => {
        let win = BrowserWindow.fromId(winId) -->获取渲染进程1的窗口对象
        win.webContents.send('事件名3', 'reply the msg') -->回复消息
    })

shell

1. shell模块可以操作用户的默认浏览器,调用资源管理器;
2. 调用用户的默认浏览器,打开指定网址:shell.openExternal('https://...')
3. shell.showItemInFolder()/openItem():资源管理器的相关操作;
4. <webview>:与HTML的<iframe>相似,但webview与应用程序不在同一个进程;
    1. webview没有渲染进程的权限,并且与应用之间的交互都是异步的,保证应用的安全性不受嵌入
    内容的影响;
    2. <webview src="https://www.baidu.com"></webview>

dialog

1. dialog模块:不仅可以弹出信息提示框,也可以实现本地文件的打开与保存;
2. 信息提示框
    1. dialog.showErrorBox('title', 'content'):错误信息提示框;
    2. dialog.showMessageBox({}, (index)=>{...}):信息提示框;
    3. 异步变同步:let index = dialog.showMessageBox({})
3. 打开文件/目录:dialog.showOpenDialog({}, (path) => {...})
    1. 打开功能只是回调文件/目录的路径,并不能获取文件/目录的内容;
    2. 操作文件/目录的功能,需要通过nodejs实现。
4. 保存文件:dialog.showSaveDialog({}, (path) => {...})
    1. showSaveDialog()只回调保存路径,并不是真的保存,不会在本地生成任何文件;
    2. 保存功能需要借助nodejs实现,通过fs模块把内容写入本地。
5. 打印当前窗口:BrowserWindow.getFocusedWindow().webContents.print()
6. app.quit():退出应用,Electron使用的是单例模式,此时不必考虑性能问题。

系统托盘

1. Tray模块:主进程的系统托盘模块;
2. 实现系统托盘:let tray = new Tray(path.join(__dirname, 'icon.png'))
3. 设置托盘名称:tray.setToolTip('title'),鼠标悬停在托盘图标上弹出;
4. 创建托盘的右键菜单
    1. Menu模块创建菜单:let menu = Menu.buildFromTemplate([{...}, {...}])
    2. 设置托盘的右键菜单:tray.setContextMenu(menu)
5. 关闭窗口,则隐藏到系统托盘;双击系统托盘,则重新打开窗口
    1. 关闭窗口
    let win = BrowserWindow.getFocusedWindow() --> 获取当前的窗口对象
    win.on('close', (e) => { --> 监听关闭按钮的事件
        if(win.isFocused()) { -->点击托盘右键菜单上的'退出',则关闭窗口
            win = null
        } else {
            e.preventDefault() --> 阻止默认的关闭事件
            win.hide() --> 隐藏窗口到系统托盘
        }
    })
    2. app.quit()会回调'close'事件,所以需要判断当前关闭的窗口是否拥有焦点;
    3. 如果当前窗口拥有焦点,则隐藏窗口;如果没有焦点,表示窗口已经处于隐藏状态,则关闭窗口
    4. 打开窗口
    tray.on('double-click', () => { --> 监听系统托盘的双击事件
        win.show()
    })
6. 托盘图标的闪烁:设置定时器setInterval(),每隔一段时间就重新设置一次图标。

消息通知

1. Electron的消息通知是通过H5的API实现的,右下方弹出通知框的效果;
    let option = {
        title: '通知标题', body: '通知内容', icon: '通知图标的路径'
    }
    let notification = new window.Notification(option.title, option)
    notification.onclick = ()=>{...} --> 通知框的点击事件
2. 监听网络变化:也是通过H5实现
    1. 网络正常:window.addEventListener('online', () => { ... })
    2. 网络断开:window.addEventListener('offline', () => { ... })

其他模块

1. globalShortcut模块:主进程模块,注册全局快捷键,在窗口创建之前注册;
    app.on('ready', () => {
        ... //注册全局快捷键 ---> ... //创建窗口 ---> ... //设置菜单
    })
    1. 注册全局快捷键:globalShortcut.register('ctrl+e', () => { ... })
    2. 判断是否注册成功:globalShortcut.isRegistered('ctrl+e')
    3. 窗口关闭时要取消全局快捷键:globalShortcut.unregister('ctrl+e')
    app.on('will-quit', () => { globalShortcut.unregister('ctrl+e') })
2. clipboard模块:公用模块,剪切板事件
    1. 复制操作:clipboard.writeText('复制内容'),系统的Ctrl+C
    2. 手动获取复制的内容:let text = clipboard.readText(),系统的Ctrl+V
    3. 除了读取文本,剪切板还能读取HTML、Image、RTF...
3. nativeImage模块:公用模块,用于读取、传递图片
    1. nativeImage.createEmpty():创建一个空的nativeImage对象;
    2. nativeImage.createFromPath('路径'):根据图片路径,创建nativeImage对象
4. 剪切板无法直接读取图片,需要借助nativeImage模块;
    let image = nativeImage.createFromPath('static/back.jpg')
    clipboard.writeImage(image) --> 复制图片
    let data = clipboard.readImage() --> 读取复制的图片,返回nativeImage对象
    let src = data.toDataURL() --> 转为图片的Base64编码
    1. 创建一个<img />:let img = new Image()
    2. 设置src属性:img.src = src
    3. 插入到HTML中:document.body.appendChild(img)

Electron-Vue

1. Electron-Vue:Electron结合Vue开发单页面桌面应用;
    1. 安装环境:cnpm install electron -g、cnpm install vue-cli -g
    2. 创建项目:vue init simulatedgreg/electron-vue my-project
    3. 下载依赖:cd my-project ---> cnpm install
    4. 运行项目:npm run dev
2. 在main.js中默认引入了electron,在Vue组件中通过this.$electron能直接使用;
    1. 主进程模块:src/main/index.js
    2. 渲染进程模块:src/renderer目录下的所有Vue组件
3. 在Vue组件中使用Node.js的模块:const path = require('path');
4. 打包后的electron应用无法找到图片
    1. 对于static目录下的图片:<img :src="'./static/img/a.png'" />
    2. 对于电脑本地的图片,直接使用图片的绝对路径,另外还要关闭窗口的安全模式
    new BrowserWindow({ webPreferences: { webSecurity: false } });

窗口操作

1. frame:false:无边框化,去除顶部菜单、最小化、最大化和关闭;
    let win = new BrowserWindow({..., useContentSize:true, frame:false})
    1. 无边框化之后,窗口的可拖拽区也随之消失,需要手动设置拖拽区,鼠标可以拖动窗口;
    2. 设置拖拽区的CSS属性:-webkit-app-region: drag
    body{ -webkit-app-region: drag } --> 设置整个窗口为拖拽区
    3. 拖拽区内的按钮必须标记为no-draggable,否则将不可点击;
    button{ -webkit-app-region: no-drag; }
2. win.setMenu(null):只隐藏顶部菜单;
3. 最大化、最小化、隐藏、关闭窗口
    1. win.hide():隐藏窗口; win.isVisible():窗口是否可见; win.close():关闭窗口
    2. win.maximize():最大化窗口,如果窗口尚未显示,这也将会显示,但不会有焦点;
    3. win.unmaximize():取消窗口最大化;  win.isMaximized():判断窗口是否最大化;
    5. win.minimize():最小化窗口;  win.isMinimized():判断窗口是否最小化;
    6. win.restore():将窗口从最小化状态恢复到以前的状态。

NeDB

1. NeDB:Nodejs实现的一个NoSQL嵌入式数据库;
    1. 与Android上的Sqlite类似,都是嵌入式数据库,一般用于本地数据的持久化;
    2. 不同的是,Sqlite是MySql类型的,使用SQL语句操作,而NeDB的操作类似于MongoDB;
    3. 可以充当内存数据库,也可以用来实现本地存储,甚至可以在浏览器中使用。
2. 使用方式:
    1. 安装操作模块:cnpm install nedb --save
    2. 在src/renderer目录下创建datastore.js,import Datastore from 'nedb'
    import path from 'path'  import {remote} from 'electron'
    export default new Datasotre({
        autoload: true,
        filename: path.join(remote.app.getPath('userData'), '/data.db')
    })
    3. 设置NeDB保存在userData目录中,该目录空间专门为应用程序所保留,不会被篡改;
    4. 在main.js中,把数据库操作对象绑定到原型上:
    import db from './datastore'  --> Vue.prototype.$db = db

相关文章

网友评论

    本文标题:Electron

    本文链接:https://www.haomeiwen.com/subject/kuuewqtx.html