Skip to content

tdeheurles/ElectronTest

Repository files navigation

ElectronTest

This project main purpose is to present electron

- Electron
    - ipc (main & renderer)
    - Browser-Window
    - electron-packager

I also use it as a brief tutorial for some other technologies:

- ReactJS
- ReactBootstrap
- Rx
- Gulp
- ES6 (babel)

Summary


Electron main concepts

Electron is an io.js server that is installed (without admin rights) on the client machine in order to run an application. This application comes with a suite of helper to update, report bug and access the client machine.

When you start an electron application, a first process named main is started as an io.js webserver. This main process has access to all the io.js packages.

The views are built using HTML/Javascript (with all the library/framework...).
This Gui is built on a fork of chromium. So, like chrome/chromium, every window is started in a different process. The main process will manage these views. A view is named a renderer process.

When you start a view, you give it a URL. This URL can bring to a local server hosted by the main process or to any other WebSite. You can choose at any time if you prefer a local application, a remote application or an hybrid one.

Electron comes with an async messagerie service, ipc, to communicate from main to views. You can easily share data with this channel.

You can add security or restrain the different parts of your views by turning on or off some options.


This project main concept

This project shows a very simple view with two buttons.

  • The first one will spawn a view each time you click. These windows will use the ipc messagerie to receive a Rx stream of data to update. The update will be done with ReactJS.
  • The second one will show a notification.

The architecture is the following:

  • a node server will host the files (distinct from the electron app),
  • the electron app will run as a browser with additionnal capabilities.
Context

This project is built on windows 7 for windows. On other plateform just have a look ato the script in the main folder and adapt it to bash or something like that (few line of codes). It's very simple script commands (or open an issue if you want help).

At the moment, electron version is 0.31.2.


Go-Fast

  • install node.js
  • control that node path is in your env PATH
  • git clone https://github.com/tdeheurles/ElectronTest
  • in a PowerShell terminal : run-server.ps1
  • in another PowerShell terminal : run-electron.ps1
  • make a prayer
  • test the electron app

if you've got some problems, go to issues and to the server section.


Electron project

The official quick-start is here, and you can find the documentation here.

An electron project can be achieved with a minimum of 2 files :

app
 |--- main.js
 *--- package.json

The package.json is a standard node/iojs package file. We put some metadata and the dependencies.

{
  "name":    "ElectronTest",
  "version": "0.0.1",
  "main":    "main.js",
}

The main.js is the script that will start the application. A minimalist one will be of that kind:

const app           = require('app');
const BrowserWindow = require('browser-window');

// close the main process when all renderer process are closed
app.on('window-all-closed', function() {
  if (process.platform != 'darwin') {
    app.quit();
  }
});


// start a process using app (the code speaks by itself).
let mainWindow = null;
app.on('ready', function() {
  mainWindow = new BrowserWindow({width: 800, height: 600});
  mainWindow.loadUrl('http://www.google.com');
  mainWindow.on('closed', function() {
    mainWindow = null;
  });
});

If you want to host files (like an index.html) within the application, you just have to add them to the app directory and to change URL of the mainWindow to something like 'file://' + __dirname + '/index.html'

When the application is packaged, 50mb of other files need to be downloaded. All these files just need to be copy/paste to the user filesystem.

File exploration

We will now go through all the files of the project to cover briefly all the functionalities.
Do not hesitate to open an issue for any question.
Note that the code here should not be copy/paste as I added lots of comments that may break the code. Prefer to simply git clone

electron

Our electron app is composed of two files: main.js and package.json.

package.json :

{
  "name":    "ElectronTest",
  "version": "0.0.1",
  "main":    "main.js",
  "dependencies": {
    "rx": "^3.1.2"
  }
}

It's a very simple package.json with an entry pointing to the main.json file. I have added a dependency : Rx. This dependency need to be downloaded before running the application (electron won't download it when running).

main.js :
The main.js file is very simple with no idea of architecture.
You will find first some globals defining the server and port. jshint and use strict will restrict the writing of javascript.

/* jshint node: true*/
'use strict';

We can find a MAIN part that will start the application and the first renderer process.

const app = require('app');
const BrowserWindow = require('browser-window');

app.on('window-all-closed', function() {
  if (process.platform != 'darwin') {
    app.quit();
  }
});


let atomScreen = null;
let mainWindow = null;
app.on('ready', function() {
  atomScreen = require('screen')

  mainWindow = new BrowserWindow({width: 800, height: 600});
  mainWindow.loadUrl("http://localhost:8123/index.html");

  mainWindow.openDevTools();

  mainWindow.on('closed', function() { mainWindow = null; });
});

The app is the object representing the main process.
The BrowserWindow is the object that represent the renderer process.

This code will shut down the app if all window are closed:

app.on('window-all-closed', function() {
  if (process.platform != 'darwin') {
    app.quit();
  }
});

We then start the application in a window of 800*600, ask to load the URL at localhost:8123, start the devtools (chromium tools to debug the GUI) and ask to delete the mainWindow instance if the user click on close. The atomScreen will be used after but need to be required after the app is ready.

let atomScreen = null;
let mainWindow = null;
app.on('ready', function() {
  atomScreen = require('screen')

  mainWindow = new BrowserWindow({width: 800, height: 600});
  mainWindow.loadUrl("http://localhost:8123/index.html");

  mainWindow.openDevTools();

  mainWindow.on('closed', function() {
    mainWindow = null;
  });
});

index.html

Here we will find the different elements that will be loaded by our app.
Each HTML is a very simple one with a css url and pointing to a different javascript file. For example:

<!DOCTYPE html>
<html>
<head lang="en">
    <meta charset="UTF-8">
    <title>ReactJS and ES6</title>
    <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.5/css/bootstrap.min.css">
</head>
<body>
    <script src="dist/app.js"></script>
</body>
</html>

So our electron app, when starting, opens that index.html. This html file downloads a css file from bootstrap and our script app.js.

This app.js is a transpilation in ES5 from our source file app.jsx written in ES6. You can see here how it's done.
The app.jsx is defined in ReactJS. The code can be read like HTML where you can easily add some javascript code.

// First we import some element using ES6 syntaxe
import React from 'react';
import Tile   from './Tile.jsx'     // Tile is a ReactJS object (see after)
import { Button, ButtonGroup,
         Grid, Row, Col,
         OverlayTrigger,
         Label
       }        from 'react-bootstrap';     // React-Bootstrap proposes lots of
                                            //   already prepared components
// The function React.render take two parameters
// - an HTML like element
// - the element where that generated HTML will be put (document.body)
//
// Here I just put a text Element in the first Row, second column. And a Tile element in the second Row, second column.
// You also can see that we have an easy responsive design achieved with ReactBootstrap.
React.render(
  <Grid>
    <Row>
      <Col xs={1} md={4}></Col>
      <Col xs={10} md={4} className="text-center">
        ElectronTest
      </Col>
      <Col xs={1} md={4}></Col>
    </Row>

    <Row>
      <Col xs={4} md={4}></Col>
      <Col xs={4} md={4}>
          <Tile Title="Create a RxWindow" />
      </Col>
      <Col xs={4} md={4}></Col>
    </Row>
  </Grid>,
  document.body
);

Then we can find our Tile.jsx object used in the app.jsx element:

// We import some dependencies
import React      from 'react';
import { Button } from 'react-bootstrap';

// here we create a class Tile
//   - that extends React.Component
//   - that will be exported as default (behind the scope of this tutorial)
export default class Tile extends React.Component {

    // we first define a function in the ES6 syntax.
    //   this code permits to avoid browserify to throw an error when trying
    //   to import an element that won't exist at transpilation time.
    //   For example, we will import ipc that will just exist at runtime,
    //     and only on electron browser
    executionTimeRequire = (name) => { return require(name) }

    // Here is the handler for button of this object
    createTile = (e) => {
        // We require ipc
        var ipc = this.executionTimeRequire('ipc')
        // We use that IPC to send a message to the main process
        //   on the 'create_someReactView' topic
        // Our message does not comport data (empty envelope)
        ipc.send('create_someReactView', null)
  }

  // Here is the code that define the visual of our object.
  // It's just a button
  render() {
    return (
      <Button onClick={this.createTile}>{this.props.Title}</Button>
     );
  }
}

IPC

So we sent a message to our main process on the create_someReactView topic.
Look at the corresponding code in the main process file main.js in the electron folder:

// We require ipc. Here, the code is not transpiled or browserified so we
// can just use `require`
const ipc_main = require('ipc')

// The first and second topics will trigger a function when a message is
// received. It's used to create different `renderer` process.  
ipc_main.on('create_someReactView', function(event, arg) {
  create_someReactView()
})
function create_someReactView(){
  // we first use the atomScreen generated at the application start
  //   to get the screen size
  const screen_size = atomScreen.getPrimaryDisplay().workAreaSize;

  // we define the width of our view
  //  (can be done more dynamically)
  const notification_width = 200
  const notification_height = 130

  // we prepare the position of our view at the bottom right
  const notification_x = screen_size.width - notification_width - 10
  const notification_y = screen_size.height - notification_height - 10

  // we instantiate our renderer process
  const notification = new BrowserWindow({
    'width':          notification_width,
    'height':         notification_height,
    'frame':          false,
    'show':           false,
    'x':              notification_x,
    'y':              notification_y
  });

  // We load an Url from our server  
  notification.loadUrl('http://${server_adress}:${server_port}/SomeReactView.html')

  // and we show our View
  notification.show()
}

SomeReactView.html is exactly the same as index.html. It just points to SomeReactView.js (that is transpiled from SomeReactView.jsx) :

import React  from 'react'
import { Grid, Row, Col, Button } from 'react-bootstrap'
import SomeViewModel from './SomeViewModel.js'

export default class SomeReactView extends React.Component {
    // ES6 constructor
    constructor(props) {
        super(props);

        // an initialiser object
        const someViewModel = new SomeViewModel([0,0,0, 0,0,0, 0,0,0,
                                                 0,0,0, 0,0,0, 0,0,0,
                                                 0,0,0, 0,0,0, 0,0,0,
                                                 0,0,0, 0,0,0, 0,0,0 ]);

        // we define the initial state of our view
        this.state = { someViewModel: someViewModel };
    }

    // same as before
    executionTimeRequire = (name) => { return require(name) }

    // We ask the component to do something just before appearing
    componentWillMount() {
        // We initiate our messaging tool
        const ipc = this.executionTimeRequire('ipc')

        // We prepare to receive updates:
        // When we receive data on the `update` topic
        ipc.on('update', (data) => {
            // we create a new ViewModel from the data
            let newViewModel = this.state.someViewModel
            newViewModel.elements = data
            // we set the state to that new viewModel
            // react will generate the new component from that state
            this.setState({someViewModel: newViewModel});
        })

        // We send a message on the 'give_it_to_me' topic (empty envelope)
        ipc.send('give_it_to_me', null)
    }

  // as for all ReactJS, the visual part of the element
  //   here we just have something like a table (that can
  //     have be put in another object)
  // We can notice the style added for the Gui. It's a html class that
  //   correspond to electron (need to be added on frameless view)
  render() {
    return (
      <Grid style={{'-webkit-app-region': 'drag'}}>
        <Row>
          <Col xs={1}>{this.state.someViewModel.elements[0]}</Col>
          <Col xs={1}>{this.state.someViewModel.elements[1]}</Col>
          <Col xs={1}>{this.state.someViewModel.elements[2]}</Col>
          <Col xs={1}>{this.state.someViewModel.elements[3]}</Col>
          <Col xs={1}>{this.state.someViewModel.elements[4]}</Col>
          <Col xs={1}>{this.state.someViewModel.elements[5]}</Col>
        </Row>
        [...]
      </Grid>
     );
  }
}

// We instantiate a SomeReactView (defined just before)
//   and we put it directly under the body of the html
//     (erasing everything defined inside)
React.render(
  <SomeReactView />,
  document.body
);

As a summary, we have started the main process of an electron application. That main process has started a first renderer process with the code at localhost:8123/index.html.
We then have clicked a button in that view. The trigger have sent an IPC message to the main process to request the creation of a new renderer process.
The second renderer process has started. Just before appearing (ComponentWillMount), it has sent a new IPC message to the main process. That last message was an empty envelope on the give_it_to_me topic.

Rx

We now look inside the main process code in the app.js file at the Rx section.

This code is separated in three elements.
The first one will permit to trigger an action on the reception of a message on the topic give_it_to_me:

// We know that the message is an empty envelope so we don't care of arg.
// event will be used later to find the author of the message with `event.sender`
ipc_main.on('give_it_to_me', function(event, arg) {
  console.log("stream requested")
  //[...]
})

The second will create a pattern for a sequence.
For that we create a source by using Rx.Observable.timer(100,16). At this point, we have an infinite sequence espaced by 16ms that will start after 100ms.
Then we map each ping with a value. In this case, I return an array of two random elements. The first element will be the cell to update in the SomeReactView, the other will be the number that will appear:

  console.log("stream requested")
  const Rx = require('rx')
  const source = Rx.Observable
                 .timer(100,16)
                 .map(function(x) {
                    return [  
                      Math.floor((Math.random() * 36) + 1),
                      Math.floor((Math.random() * 100) + 1)
                    ]});

For a better understanding :

-------------------------------------------------------------> time
<--- 100ms ---> | <- 16ms -> | <- 16ms -> | <- 16ms -> |
                |            |            |            |
             [25,99]      [12,47]      [01,76]      [14,28]

Finally we subscribe to this source and tell what we want to do when:

  • we receive a value
  • an error is thrown
  • the sequence is terminated (here it won't happen)
  source.subscribe(
    function (x) {
      cache[x[0]] = x[1]
      event.sender.send('update', cache)
    },
    function (err) {
        console.log('Error: ' + err);
    },
    function () {
        console.log('Completed');
    }
  );

Each click on the main view button should generate a new GUI with a lot of numbers changing fast. The CPU will determine the number of GUI that can be generated (it's a simple non optimized HelloWorld here).

packaging

The project comes with two main parts: electron and the server.
We will here have a look at the generation of the code from the source for each one. The script are very static for now (no config).

main path

In the main path you will find a run script for each mini project : run-electron.ps1 and run-server.ps1.
build-windows-package.ps1 and npm-install-for-electron.bat are used by run-electron.ps1.

run-electron.ps1
# PowerShell script

# We define a reference to the main folder
$main_path = "$pwd"

# We statically define where electron will be run
$electron_runtime = "$main_path\electron\packaging\electrontest-win32-x64"

# We download a version of electron and get a cache
.\build-windows-package.ps1

# We install the npm dependencies for node
#   (the command is a bit special for electron)
.\npm-install-for-electron.bat

# We move to the electron folder and execute the program
cd $electron_runtime
.\electrontest.exe
build-windows-package.ps1
# npm proposes a packager for electron named `electron-packager`.
#   This one won't generate a unique exe file but will download all the
#   required files from some configuration
npm install electron-packager -g

$sourcedir          = "$pwd\electron\src"  # Where are our sources
$appname            = "electrontest"       # The name that will appear everywhere
$platform           = "win32"              # windows/mac/linux
$arch               = "x64"                # architecture
$electron_version   = "0.31.2"             # the version of electron you want (no latest)
$app_version        = "0.0.0"              # the version for our app
$output_directory   = "$pwd\electron\packaging"        # path for our package
$cache_directory    = "$pwd\electron\packaging\cache"  # path for our cache

# We give all parameters and add --overwrite to generate completely everytime
electron-packager               `
    $sourcedir                  `
    $appname                    `
    --platform=$platform        `
    --arch=$arch                `
    --version=$electron_version `
    --app-version=$app_version  `
    --out=$output_directory     `
    --cache=$cache_directory    `
    --overwrite
npm-install.bat
# stop log
@echo off

# move to the app folder inside the packaged electron
cd  electron\packaging\electrontest-win32-x64\resources\app

# set variable needed by electron for adding npm package
set npm_config_disturl=https://atom.io/download/atom-shell
set npm_config_target=0.31.2
set npm_config_arch=x64
set HOME=~/.electron-gyp

# We install (with addition of the paramters to add package if needed)
#   ie: npm-install.bat gulp --save-dev
npm install %*
run-server.ps1

This one is just a wrapper to the one in the server folder (you can use the same command inside each folder). It's very simple commands here.

# main folder

cd server

.\run-server.ps1
# server path

# install the npm (according to package.json)
npm install

# launch gulp (according to gulpfile.js)
npm run gulp
package.json
{
  "name": "electrontest",
  "version": "1.0.0",
  "description": "",
  "main": "gulpfile.js",     "//": "HERE WE TELL NPM THAT THE ENTRYPOINT IS GULP",
  "scripts": {
    "gulp": "gulp",          "//": "OUR `npm run gulp` CORRESPONDS TO THAT LINE",
  },
  "author": "",
  "license": "ISC",
  "dependencies": {          "//": "THESE DEPENDENCIES WILL BE INCLUDED IN OUR BUNDLE FILES",
    "react": "^0.13.3",
    "react-bootstrap": "^0.25.0",
    "bootstrap": "^3.3.5",
    "electron-toaster": "^1.0.8",
    "node-notifier": "^4.2.3"
  },
  "devDependencies": {       "//": "THESE DEPENDENCIES DOES NOT AFFECT THE BUNDLE FILES",
    "babelify": "^6.3.0",
    "browserify": "^11.0.1",
    "gulp": "^3.9.0",
    "gulp-webserver": "^0.9.1",
    "vinyl-source-stream": "^1.1.0"
  }
}
gulpfile.js

gulp is an helper that can be used to automate tasks for web stuff. It's in the same category as webpack The main idea here is to take the sources, transpile them from ReactJS and ES6 to the ES5 the browser knows.

first we get some dependencies
// gulp will manage this process. everything is organised in task.
// Each task can be triggered from the CLI with `gulp run taskname`
//   the `default` is automatically triggered if no name is given
const gulp        = require('gulp');                 

// browserify is the one that will transform our multiple files in only one
// this task is done according to the require in the code
//   It's the one that we get around with the function
//      executionTimeRequire = (name) => { return require(name) };
//      defined in Tile.jsx and SomeReactView.jsx
const browserify  = require('browserify');

// Babel is a transpiler
// This package permit us to use ES6 and ES7 features in our code.
// It's the one that also transpile our ReactJS jsx files in js files
const babelify    = require('babelify');

// This one manages the process for the transpilation
const source      = require('vinyl-source-stream');

// Webserver, as its name says, will serve our files. It's simpler in this project
// than using nginx or something else
const webserver   = require('gulp-webserver');


// TASK DEFINITIONS
// ================
// Here we define 4 gulp tasks. Each one corresponds to a part of the project.

// The first one generates the app.js file that corresponds to our first renderer process
gulp.task('generate-app', function () {
  // We first give to browserify the entry point of `tree` of files
  //     (It will follow automatically the require)
  //     debug tells him to generate a map file
  //      ==> (you will be able to look to the now transpiled class from the browser)
  return browserify({entries: './src/app.jsx', extensions: ['.jsx'], debug: true})
    // here is the command for babel. stage defines the features we want to use
    //   stages are the development progress of the feature. Here I use early stage features
    //   function declaration is one of them => ie: "() => { }"
    .transform(babelify, { stage: 0 })
    .bundle()
    //   This line asks gulp to continue when an error is thrown in the transpilation process
    //     ie: you forget a `,` and you save. If this line is not present, gulp stops
    .on('error', function(err) { console.error(err); this.emit('end'); })
    //   The name of the generated file
    .pipe(source('app.js'))
    // the folder where to put our generated file
    .pipe(gulp.dest('public/dist'));
});

// this task is in the same form
gulp.task('generate-someReactView', function () {
  return browserify({entries: './src/SomeReactView.jsx', extensions: ['.jsx'], debug: true})
    .transform(babelify, { stage: 0 })
    .bundle()
    .on('error', function(err) { console.error(err); this.emit('end'); })
    .pipe(source('SomeReactView.js'))
    .pipe(gulp.dest('public/dist'));
});

// this task is in the same form
gulp.task('generate-someViewModel', function () {
  return browserify({entries: './src/SomeViewModel.js', extensions: ['.js'], debug: true})
    .transform(babelify, { stage: 0 })
    .bundle()
    .on('error', function(err) { console.error(err); this.emit('end'); })
    .pipe(source('SomeViewModel.js'))
    .pipe(gulp.dest('public/dist'));
});

// Here we just copy paste the html files from src to public
gulp.task('copy-html', function () {
   gulp.src('./src/**/*.html')
   .pipe(gulp.dest('./public'));
});




// ==== MAIN
// =============
// the default task is the main one, it will be triggered by our `npm run gulp` in the `run-server.ps1`
// the default task starts by triggering the four tasks defined zbovr
gulp.task(
  'default',
  ['generate-app', 'generate-someViewModel', 'copy-html', 'generate-someReactView'],
  function () {
    // then this two lines ask to look for any change in our files (at save time).
    // ie: I look for any jsx in any path under src, and trigger my generate-tasks
    gulp.watch('./src/**/*.jsx', ['generate-app', 'generate-someViewModel', 'generate-someReactView' ]);
    gulp.watch('./src/**/*.html', ['copy-html']);

    // this code corresponds to the server
    // I give him port, index file.
    gulp.src('public')
        .pipe(webserver({
          port: 8123,
          fallback: "index.html"
        }));
});

Issues

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published