250 lines
7.5 KiB
Markdown
250 lines
7.5 KiB
Markdown
|
## Electron Quick Start
|
|||
|
```
|
|||
|
# 克隆这仓库
|
|||
|
$ git clone https://github.com/electron/electron-quick-start
|
|||
|
# 进入仓库
|
|||
|
$ cd electron-quick-start
|
|||
|
# 安装依赖库
|
|||
|
$ npm install
|
|||
|
# 运行应用
|
|||
|
$ npm start
|
|||
|
```
|
|||
|
|
|||
|
## 使用IPC进行GUI与原生APP进行通讯
|
|||
|
在main.js里添加下面的代码,从通道订阅消息:
|
|||
|
```
|
|||
|
var ipc = require('ipc');
|
|||
|
ipc.on('close-main-window', function () {
|
|||
|
app.quit();
|
|||
|
});
|
|||
|
```
|
|||
|
引入ipc模块后,通过通道订阅消息就变得很简单,on()方法设置订阅的通道名,定义回调函数。
|
|||
|
|
|||
|
渲染进程要通过通道发送消息,将下面代码加入index.js:
|
|||
|
```
|
|||
|
var ipc = require('ipc');
|
|||
|
var closeEl = document.querySelector('.close');
|
|||
|
closeEl.addEventListener('click', function () {
|
|||
|
ipc.send('close-main-window');
|
|||
|
});
|
|||
|
```
|
|||
|
同样,我们引入ipc模块,给关闭按钮的元素绑定一个click事件。当点击关闭按钮时,通过「close-main-window」通道的send()方法发送消息。
|
|||
|
|
|||
|
## 全局快捷键
|
|||
|
```
|
|||
|
var globalShortcut = require('global-shortcut');
|
|||
|
app.on('ready', function() {
|
|||
|
... // existing code from earlier
|
|||
|
globalShortcut.register('ctrl+shift+1', function () {
|
|||
|
mainWindow.webContents.send('global-shortcut', 0);
|
|||
|
});
|
|||
|
globalShortcut.register('ctrl+shift+2', function () {
|
|||
|
mainWindow.webContents.send('global-shortcut', 1);
|
|||
|
});
|
|||
|
});
|
|||
|
```
|
|||
|
```
|
|||
|
ipc.on('global-shortcut', function (arg) {
|
|||
|
var event = new MouseEvent('click');
|
|||
|
soundButtons[arg].dispatchEvent(event);
|
|||
|
});
|
|||
|
```
|
|||
|
## 保存用户配置
|
|||
|
使用nconf模块
|
|||
|
```
|
|||
|
npm install --save nconf
|
|||
|
```
|
|||
|
```
|
|||
|
|
|||
|
var nconf = require('nconf').file({file: getUserHome() + '/sound-machine-config.json'});
|
|||
|
|
|||
|
function saveSettings(settingKey, settingValue) {
|
|||
|
nconf.set(settingKey, settingValue);
|
|||
|
nconf.save();
|
|||
|
}
|
|||
|
|
|||
|
function readSettings(settingKey) {
|
|||
|
nconf.load();
|
|||
|
return nconf.get(settingKey);
|
|||
|
}
|
|||
|
|
|||
|
function getUserHome() {
|
|||
|
return process.env[(process.platform == 'win32') ? 'USERPROFILE' : 'HOME'];
|
|||
|
}
|
|||
|
|
|||
|
module.exports = {
|
|||
|
saveSettings: saveSettings,
|
|||
|
readSettings: readSettings
|
|||
|
};
|
|||
|
```
|
|||
|
## 系统托盘
|
|||
|
```
|
|||
|
var remote = require('remote');
|
|||
|
var Tray = remote.require('tray');
|
|||
|
var Menu = remote.require('menu');
|
|||
|
var path = require('path');
|
|||
|
|
|||
|
var trayIcon = null;
|
|||
|
|
|||
|
if (process.platform === 'darwin') {
|
|||
|
trayIcon = new Tray(path.join(__dirname, 'img/tray-iconTemplate.png'));
|
|||
|
}
|
|||
|
else {
|
|||
|
trayIcon = new Tray(path.join(__dirname, 'img/tray-icon-alt.png'));
|
|||
|
}
|
|||
|
|
|||
|
var trayMenuTemplate = [
|
|||
|
{
|
|||
|
label: 'Sound machine',
|
|||
|
enabled: false
|
|||
|
},
|
|||
|
{
|
|||
|
label: 'Settings',
|
|||
|
click: function () {
|
|||
|
ipc.send('open-settings-window');
|
|||
|
}
|
|||
|
},
|
|||
|
{
|
|||
|
label: 'Quit',
|
|||
|
click: function () {
|
|||
|
ipc.send('close-main-window');
|
|||
|
}
|
|||
|
}
|
|||
|
];
|
|||
|
var trayMenu = Menu.buildFromTemplate(trayMenuTemplate);
|
|||
|
trayIcon.setContextMenu(trayMenu);
|
|||
|
```
|
|||
|
|
|||
|
## 打包应用
|
|||
|
>摘自https://www.jianshu.com/p/f134878af30f
|
|||
|
|
|||
|
安装electron-package
|
|||
|
```
|
|||
|
npm install electron-package --save-dev
|
|||
|
```
|
|||
|
添加scrip命令 ,用于打包electron app。
|
|||
|
```
|
|||
|
"scripts": {
|
|||
|
"start": "electron .",
|
|||
|
"build": "electron-packager . hello_electron --platform=darwin --arch=x64 --ignore=node_modules/electron-*",
|
|||
|
},
|
|||
|
```
|
|||
|
**electron-packager命令格式**
|
|||
|
```
|
|||
|
electron-packager 项目目录 app名称 --platform=平台 --arch=架构 --ignore=要忽略的目录或文件
|
|||
|
arch
|
|||
|
ia32 , x64 , armv7l , all
|
|||
|
|
|||
|
plateform
|
|||
|
linux , win32 , darwin , mas , all
|
|||
|
|
|||
|
OS X (also known as darwin)
|
|||
|
Mac App Store (also known as mas)
|
|||
|
```
|
|||
|
执行命令npm run build,将得到如下结果
|
|||
|
|
|||
|
### electron-builder与electron-packager的区别
|
|||
|
使用electron-builder打包应用是安装包方式,而不想electron-packager打包之后直接是一个可文件夹,交给所有的文件暴露出来。由electron-builder打出的包更为轻量,并且可以打包出不暴露源码的setup安装程序
|
|||
|
|
|||
|
## 压缩源码
|
|||
|
为避免源代码泄露,可对源码进行压缩。
|
|||
|
|
|||
|
**安装electron-asar**
|
|||
|
```
|
|||
|
npm install electron-asar --save-dev
|
|||
|
```
|
|||
|
**添加scrip命令 ,用于压缩源代码。**
|
|||
|
```
|
|||
|
"scripts": {
|
|||
|
"start": "electron .",
|
|||
|
"build": "electron-packager . hello_electron --platform=darwin --arch=x64 --ignore=node_modules/electron-*",
|
|||
|
"package":"asar pack hello_electron-darwin-x64/hello_electron.app/Contents/Resources/app hello_electron-darwin-x64/hello_electron.app/Contents/Resources/app.asar"
|
|||
|
},
|
|||
|
```
|
|||
|
**asar 命令格式**
|
|||
|
|
|||
|
asar pack <dir> <output>
|
|||
|
执行npm run package将得到app.asar文件,此时可将app文件删除。
|
|||
|
|
|||
|
## Electron启动node.js服务器
|
|||
|
1.直接在index.html中启动外部的node.js服务器
|
|||
|
2.将原生的node.js服务器代码使用module.exports = () => {}导出,之后在electron的main.js中直接导入
|
|||
|
```
|
|||
|
app.server = require(__dirname + '/app/app')();
|
|||
|
```
|
|||
|
|
|||
|
## 不迁移项目就可以打包双版本的可行方案
|
|||
|
作者并未提供web开发的支持,但是提供了非常好的web打包支持。
|
|||
|
只要写好逻辑我们可以不用迁移项目就可以打包桌面项目和web项目。
|
|||
|
|
|||
|
process.env.IS_WEB是暴露的一个全局变量,我们可以在渲染进程中获取,项目在electron环境下,返回false。否则为true。于此,我们可以通过设置她的值来达到web dev的效果,也可以处理不同环境的不同逻辑,一些示例:
|
|||
|

|
|||
|

|
|||
|

|
|||
|
|
|||
|
## 打开新窗口的“最佳”做法
|
|||
|
1.使用webview, allowpopups变量控制是否拦截新弹出的窗口
|
|||
|
|
|||
|
下面的例子,是webview允许.open/.showModalDialog/.showModelessDialog的例子:
|
|||
|
|
|||
|
electron的index.html:(重点是参数allowpopups)
|
|||
|
```
|
|||
|
<webview id="foo" src="https://newsn.net/test.html" allowpopups style="width:100%; height:360px;"></webview>
|
|||
|
```
|
|||
|
原文:https://newsn.net/say/electron-webview-window-open.html
|
|||
|
|
|||
|
2.因为是webpack配置,入口只有index.html ,所以打开新窗口,一般会再配置一个入口。其实还有一种更佳的做法。
|
|||
|
```
|
|||
|
>>> 主进程 定义好监听事件
|
|||
|
ipc.on('newPage', function(e) {
|
|||
|
const modalPath = process.env.NODE_ENV === 'development'
|
|||
|
? 'http://localhost:9080/#/newPage'
|
|||
|
: `file://${__dirname}/index.html#newPage`
|
|||
|
let win = new BrowserWindow({
|
|||
|
width: 1024,
|
|||
|
height: 724,
|
|||
|
webPreferences: {
|
|||
|
webSecurity: false
|
|||
|
}
|
|||
|
})
|
|||
|
win.on('close', function() {
|
|||
|
win = null
|
|||
|
})
|
|||
|
win.loadURL(modalPath)
|
|||
|
|
|||
|
})
|
|||
|
|
|||
|
>>> router/index.js 定义路由
|
|||
|
// import 你的新页面 .vue 文件
|
|||
|
{
|
|||
|
path: '/newPage',
|
|||
|
name: 'newPage',
|
|||
|
component: newPage,
|
|||
|
}
|
|||
|
|
|||
|
》》》配置完成 任意进程调用ipc.send('newPage') 完美解决
|
|||
|
```
|
|||
|
3.
|
|||
|
```
|
|||
|
document.getElementById("youtube").onclick = function(){
|
|||
|
|
|||
|
youtubeWindow = new BrowserWindow ({width: 1000, height:800})
|
|||
|
|
|||
|
youtubeWindow.loadURL("https://youtube.com/");
|
|||
|
|
|||
|
youtubeWindow.on("close", function(){
|
|||
|
youtubeWindow = null;
|
|||
|
})
|
|||
|
}
|
|||
|
|
|||
|
document.getElementById("local-list").onclick = function(){
|
|||
|
|
|||
|
localListWindow = new BrowserWindow ({width: 1000, height:800})
|
|||
|
|
|||
|
localListWindow.loadURL(`file://${__dirname}/local-list.html`);
|
|||
|
|
|||
|
localListWindow.on("close", function(){
|
|||
|
localListWindow = null;
|
|||
|
})
|
|||
|
}
|
|||
|
```
|