introduce

Electron, official Introduction: Building cross-platform desktop applications using JavaScript, HTML, and CSS. Out of personal interest, I came into contact with Electron and began to try to develop some local gadgets. The following is an experience summary of the development process, easy to review and exchange.

use

Let’s build a simple electron application. Application source address: github.com/zhuxingmin/…

1. Project initialization

The project is built based on [email protected]. Run commands to generate the project

// Install create-react-app globally
npm install -g create-react-app

// Execute the command to generate the project
create-react-app electronApp

// Install the dependencies and start the project
yarn && yarn start
Copy the code

At this point, you just start a React app. The next step is to install the electron electron- Updater electron- Builder electron-is-dev library

yarn add electron electron-updater electron-builder electron-is-dev
Copy the code
The package name version role
electron ^ 9.2.1 Electron library
electron-builder ^ 22.8.0 Packaging library
electron-updater ^ 4.1.2 Update the library
electron-is-dev 2.0.0 Environmental library

2. The configuration package. Json

After installing the project dependencies, add the electron application configuration to package.json.

"version": "0.0.1"              // Set the application version number
"productName": "appName"        // Set the application name
"main": "main.js"               // Set the application entry file
"homepage": "."                 // Set the application root path
Copy the code

Add application commands, startup, and packaging to scripts.

"estart": "electron ."              / / start
"package-win": "electron-builder"   // Package-win (for Windows)
Copy the code

Added build configuration item, added configuration related to packaging. The main configurations are as follows:

attribute instructions
appId Application ID. The default value is the application installation path

On Windows, enter get-startapps in PowerShell to view the application ID
compression Apply the package compression type
nsis Configuration related to application package installation and uninstallation
files The range of files included in the application package
directories Apply the packaging address and output address
publish Application publishing configuration, currently used mainly as application update address
protocols Custom protocol, you can wake up the APP according to the protocol name
win Windows packages the configuration and generates the EXE file
"build": {
    In Windows, you can enter get-startapps in PowerShell to view the application ID
    "appId": "org.develar.zhuxingmin"./ / compress "store" | "normal" | "maximum"
    "compression": "store".// nsis installation and configuration
    "nsis": {
        "oneClick": false.// One-click install
        "allowToChangeInstallationDirectory": true.// Allow you to change the installation directory
        // The following configurations are not commonly used
        "guid": "haha".// Registry name
        "perMachine": true.// Whether to enable installation permissions (for this computer or the current user)
        "allowElevation": true.// Allow requests to be promoted. If false, the user must restart the installation program with the promoted permissions.
        "installerIcon": "xxx.ico".// Install the icon
        "uninstallerIcon": "xxx.ico".// Uninstall icon
        "installerHeaderIcon": "xxx.ico".// Install the header icon
        "createDesktopShortcut": true.// Create a desktop icon
        "createStartMenuShortcut": true.// Create the Start menu icon
        "shortcutName": "lalala" // Name of the icon
    },
    // Apply the files contained in the package
    "files": [
        "build/**/*"."main.js"."source/*"."service/*"."static/*"."commands/*"].// Apply the package address and output address
    "directories": {
        "app": ". /"."output": "dist"
    },
    // Publish configuration to work with automatic updates
    "publish": [{// "generic" | "github"
            "provider": "generic".// Static resource server
            "url": "Http://your server directory /latest.yml"}].// Custom protocols are used to wake up applications
    "protocols": [{"name": "myProtocol"."schemes": [
                "myProtocol"]}],// Windows package configuration
    "win": {
        "icon": "build/fav.ico".// Run permission
        / / "requireAdministrator" | "access to the administrator rights"
        / / "highestAvailable" | "the highest available permissions"
        "requestedExecutionLevel": "highestAvailable"."target": [{"target": "nsis"}]}},Copy the code

3. Write the entry filemain.js

As we all know, the entry file for a project based on react scaffolding is index.js, so we need to change the entry to main.js if we want to start the electron application after the above configuration is complete

  1. First, create a new directorymain.jsFile, and inpackage.jsonModify the application entry field in themainThe value ofmain.js
  2. throughelectronTo provide theBrowserWindowTo create a window instancemainWindow
  3. throughmainWindowInstance methodsloadURLTo load static resources
  4. Static resources can be loaded in two ways: development and production; Need to pass throughelectron-is-devJudge the current environment; If it is a development environment, you can open the debug portal and passhttp://localhost:3000/Load local resources (the react project starts at the default address) In a production environment, close the debug entry and find the project entry file through the local pathindex.html

The general code is as follows

const { BrowserWindow } = require("electron");
const url = require("url");
const isDev = require('electron-is-dev');
mainWindow = new BrowserWindow({
    width: 1200.// Initial width
    height: 800.// Initial height
    minWidth: 1200.minHeight: 675.autoHideMenuBar: true.// Hide the app's built-in menu bar
    titleBarStyle: false.// Hide the app's built-in title bar
    resizable: true.// Allow the window to stretch
    frame: false.// Hide the border
    transparent: true.// The background is transparent
    backgroundColor: "none".// No background color
    show: false.// No display by default
    hasShadow: false.// Apply no shadow
    modal: true.// Whether the window is a child of the disabled parent window
    webPreferences: {
      devTools: isDev,     // Whether to enable the debugging function
      nodeIntegration: true.// The node environment is integrated by default}});const config = dev
    ? "http://localhost:3000/"
    : url.format({
        pathname: path.join(__dirname, "./build/index.html"),
        protocol: "file:".slashes: true}); mainWindow.loadURL(config);Copy the code

4. Project initiation

With the pre-project completed, run the command configured above to start the electron application

   // Start the React app, which is running at "http://localhost:3000/"
   yarn start 
   // Restart the electron application. The electron application will load the React application in the entry file 'main.js' via mainwindow.loadURL (config)
   yarn estart 
Copy the code

File directory

At this point, a simple electron application has been started, and the renderings are shown below (a screenshot of the sample project).

As a client application, it is a little more complicated to update than our web development, as illustrated by the following application update example.

5. Apply updates

The electron client updates are different from web pages in that it downloads the update package locally and then overwrites the source file to achieve the update effect.

First step, install the dependencies

yarn add electron-updater electron-builder
Copy the code

The application uses the API provided by the electron updater to find and compare the latest. Yml file at the server address configured above. If the version number is updated, the application starts downloading the resource and returns information about the download progress. After the download is complete, users can be prompted automatically or manually, the application has been updated, please restart to complete the update (the update can be done without feeling, after the download of the update package, you can not prompt, the next time the client will be automatically updated)

/ / main process
const { autoUpdater } = require("electron-updater");
const updateUrl = "Remote server directory where the application resides"
const message = {
    error: "Check update error".checking: "Checking for updates...".updateAva: "New version detected, downloading...".updateNotAva: "I'm using the latest version. I don't need to update it."}; autoUpdater.setFeedURL(updateUrl); autoUpdater.on("error".(error) = > {
    sendUpdateMessage("error", message.error);
});
autoUpdater.on("checking-for-update".() = > {
    sendUpdateMessage("checking-for-update", message.checking);
});
autoUpdater.on("update-available".(info) = > {
    sendUpdateMessage("update-available", message.updateAva);
});
autoUpdater.on("update-not-available".(info) = > {
    sendUpdateMessage("update-not-available", message.updateNotAva);
});
// Update the download progress event
autoUpdater.on("download-progress".(progressObj) = > {
    mainWindow.webContents.send("downloadProgress", progressObj);
});
autoUpdater.on("update-downloaded".function (event, releaseNotes, releaseName, releaseDate, updateUrl, quitAndUpdate) {
    ipcMain.on("isUpdateNow".(e, arg) = > {
        // Exit the application and update after receiving a confirmation message from the rendering process
        autoUpdater.quitAndInstall();
    });
    // Ask whether to update immediately
    mainWindow.webContents.send("isUpdateNow");
});
ipcMain.on("checkForUpdate".() = > {
    // Check for updates
    autoUpdater.checkForUpdates();
});

function sendUpdateMessage(type, text) {
  // Notifies the rendering process of the updated message event
  mainWindow.webContents.send("message", { text, type });
}

Copy the code
// The render process
const { ipcRenderer } = window.require("electron");

// Send a request to check for updates
ipcRenderer.send("checkForUpdate");

// Set the listening channel to check for updates

// Listen to check for update events
ipcRenderer.on("message".(event, data) = > {
  console.log(data)
});

// Monitor the download progress
ipcRenderer.on("downloadProgress".(event, data) = > {
  console.log("downloadProgress: ", data);
});

// Listen for updates to start
ipcRenderer.on("isUpdateNow".(event, data) = > {
   // After the user clicks OK, the update is sent back to the main process
   ipcRenderer.send("isUpdateNow");
});
Copy the code

The main steps for applying updates

  1. In the main process, the API is used to obtain whether updates are available on the remote server
  2. Compare the version number of the update package to determine whether to update
  3. If the comparison result needs to be updated, download the update package and return to the current download progress
  4. After the download is complete, the developer can choose to be prompted automatically, manually or not (the app will automatically update the next startup).

This shows how, on the page (the rendering process), it communicates with the main process to check for updates. In practice, interaction between the main process and the rendering process is essential if we need to use background capabilities or native functionality. So how do they interact?

Before looking at the following code snippet, let’s take a look at the electron main process and the rendering process. In simple terms, everything executed through main.js belongs to the main process, and everything else is the rendering process.

6. Common interaction between the main process and the rendering process

// Used in the main process
const { ipcMain } = require("electron");

// Use in the render process
const { ipcRenderer } = window.require("electron");
Copy the code

Methods a

The rendering process sends the request and listens for the callback channel

IpcRenderer. Send (channel, someRequestParams); ipcRenderer.on(`${channel}-reply`.(event, result) = >{
    // Receive the result from the main process
})
Copy the code

The main process listens for the request and returns the result

ipcMain.on(channel, (event, someRequestParams) = > {
    The equestparams are taken by someRequestParams. The equestparams are taken by someRequestParams
    event.reply(`${channel}-reply`, result)
})
Copy the code

Way 2

Rendering process

const result = await ipcRenderer.invoke(channel, someRequestParams);
Copy the code

Main process:

ipcMain.handle(channel, (event, someRequestParams) = > {
    The equestparams are taken by someRequestParams. The equestparams are taken by someRequestParams
    return result
});
Copy the code

Mode 3: The rendering process notifies the main process, and mode 3: the main process notifies the rendering process

The main process

/* * Use 'BrowserWindow' to initialize instance 'mainWindow' */ 
mainWindow.webContents.send(channel, something)
Copy the code

Rendering process

ipcRenderer.on(channel, (event, something) = > {
    // do something
})
Copy the code

The above application update uses the first method.

There are other communication methods, such as postMessage and sendTo, which can be used according to the specific scenario.

7. Application wake up (linkage with other applications)

In addition to running the electron application by double-clicking on the icon, it can also be launched via a protocol link (browser address bar or command line). This allows us to wake up the application in the form of a link on a web page or other application. Zhuxingmin ://? A =1&b=2&c=3 ‘Custom protocol name: zhuxingmin’ parameter: a=1&b=2&c=3 ‘.

We can use parameters to make the app jump to a page or make the app do some functional action or whatever.

const path = require('path');
const { app } = require('electron');

// Obtain the single instance lock
const gotTheLock = app.requestSingleInstanceLock();

// If the instance fails to be obtained, it proves that the instance is running and exits directly
if(! gotTheLock) { app.quit(); }const args = [];
// In the case of a development environment, you need to add the absolute path of the script to the parameter
if(! app.isPackaged) { args.push(path.resolve(process.argv[1]));
}
// Add a '--' to ensure that the following parameters are not processed by Electron
args.push(The '-');
const PROTOCOL = 'zhuxingmin';
// Set a custom protocol
app.setAsDefaultProtocolClient(PROTOCOL, process.execPath, args);

// If there are no other instances when the protocol is opened, the current instance is the master instance and the parameters are processed
handleArgv(process.argv);

// When other instances are started, the primary instance receives the startup parameter 'argv' of the other instance through the second-instance event
app.on('second-instance'.(event, argv) = > {
  if (process.platform === 'win32') {
    // When Windows starts with a protocol URL, the URL is taken as an argument, so it needs to be handled in this eventhandleArgv(argv); }});// When macOS starts with a protocol URL, the primary instance receives the URL via an open-URL event
app.on('open-url'.(event, urlStr) = > {
  handleUrl(urlStr);
});

function handleArgv(argv) {
  const prefix = `${PROTOCOL}: `;
  const offset = app.isPackaged ? 1 : 2;
  const url = argv.find((arg, i) = > i >= offset && arg.startsWith(prefix));
  if (url) handleUrl(url);
}

function handleUrl(urlStr) {
  // myapp://? a=1&b=2
  let paramArr = urlStr.split("?") [1].split("&");
  const params = {};
  paramArr.forEach((item) = > {
      if (item) {
        const [key, value] = item.split("="); params[key] = value; }});/** { a: 1, b: 2 } */ 
}
Copy the code

8. Reference documents

  1. Automatic updates
  2. The custom protocol wakes up the Electron application
  3. Electron main process and rendering process

9. Complete project code

Github.com/zhuxingmin/…


Nanjing S300 Cloud Information Technology Co., LTD. (CH300) was founded on March 27, 2014. It is a mobile Internet enterprise rooted in Nanjing, currently located in Nanjing and Beijing. After 7 years of accumulation, the cumulative valuation has reached 5.2 billion times, and has been favored by many high-quality investment institutions at home and abroad, such as Sequoia Capital, SAIC Industrial Fund, etc. S300 Cloud is an excellent domestic independent third-party SaaS service provider of auto transaction and finance, which is based on artificial intelligence and takes the standardization of auto transaction pricing and auto financial risk control as the core product.

Welcome to join s300 Cloud and witness the booming development of the automobile industry together. Look forward to walking hand in hand with you! Java development, Java internship, PHP internship, testing, testing, product manager, big data, algorithm internship, hot recruitment… Website: www.sanbaiyun.com/ Resume: [email protected], please indicate from 😁