Электрон - Краткое руководство

Почему Электрон?

Electron позволяет создавать настольные приложения с использованием чистого JavaScript, предоставляя среду выполнения с богатыми API-интерфейсами собственной (операционной системы).

Это не означает, что Electron является привязкой JavaScript к библиотекам графического пользовательского интерфейса (GUI). Вместо этого Electron использует веб-страницы в качестве своего графического интерфейса, поэтому вы также можете видеть его как минимальный браузер Chromium, управляемый JavaScript. Таким образом, все электронные приложения технически представляют собой веб-страницы, работающие в браузере, который может использовать API вашей ОС.

Кто использует электрон?

Github разработал Electron для создания текстового редактора Atom. Оба они были открыты в 2014 году. Electron используется многими компаниями, такими как Microsoft, Github, Slack и т. Д.

Electron использовался для создания ряда приложений. Ниже приведены несколько примечательных приложений -

  • Рабочий стол Slack
  • Настольное приложение Wordpress
  • Код Visual Studio
  • Редактор Caret Markdown
  • Приложение электронной почты Nylas
  • GitKraken git-клиент

Чтобы начать разработку с использованием Electron, вам необходимо установить Node и npm (менеджер пакетов узлов). Если у вас их еще нет, перейдите к настройке узла, чтобы установить узел в вашей локальной системе. Убедитесь, что node и npm установлены, выполнив следующие команды в вашем терминале.

node --version
npm --version

Приведенная выше команда сгенерирует следующий вывод -

v6.9.1
3.10.8

Всякий раз, когда мы создаем проект с помощью npm, нам нужно предоставить package.jsonфайл, в котором есть все подробности о нашем проекте. npm позволяет нам легко настроить этот файл. Давайте настроим наш девелоперский проект.

  • Запустите свой терминал / cmd, создайте новую папку с именем hello-world и откройте эту папку с помощью команды cd.

  • Теперь, чтобы создать файл package.json с помощью npm, используйте следующую команду.

npm init
  • Он запросит у вас следующую информацию -

Просто продолжайте нажимать Enter и введите свое имя в поле «Имя автора».

Создайте новую папку и откройте ее с помощью команды cd. Теперь выполните следующую команду, чтобы установить Electron глобально.

$ npm install -g electron-prebuilt

Как только он запустится, вы можете проверить, правильно ли установлен Electron, выполнив следующую команду -

$ electron --version

Вы должны получить результат -

v1.4.13

Теперь, когда мы настроили Electron, давайте перейдем к созданию нашего первого приложения с его использованием.

Electron берет главный файл, определенный в вашем файле package.json, и выполняет его. Этот основной файл создает окна приложений, которые содержат визуализированные веб-страницы и взаимодействие с собственным графическим интерфейсом пользователя (графический интерфейс пользователя) вашей операционной системы.

Когда вы запускаете приложение с помощью Electron, main processсоздано. Этот основной процесс отвечает за взаимодействие с собственным графическим интерфейсом пользователя операционной системы. Он создает графический интерфейс вашего приложения.

Просто запуск основного процесса не дает пользователям вашего приложения никакого окна приложения. Они создаются основным процессом в главном файле с помощью модуля BrowserWindow . Затем каждое окно браузера запускает собственноеrenderer process. Процесс рендеринга берет HTML-файл, который ссылается на обычные файлы CSS, файлы JavaScript, изображения и т. Д., И отображает его в окне.

Основной процесс может получить доступ к собственному графическому интерфейсу через модули, доступные непосредственно в Electron. Настольное приложение может получить доступ ко всем модулям Node, таким как модуль файловой системы, для обработки файлов, запроса на выполнение HTTP-вызовов и т. Д.

Разница между процессами Main и Renderer

Основной процесс создает веб-страницы путем создания экземпляров BrowserWindow . Каждый экземпляр BrowserWindow запускает веб-страницу в собственном процессе рендеринга. Когда экземпляр BrowserWindow уничтожается, соответствующий процесс рендеринга также завершается.

Основной процесс управляет всеми веб-страницами и соответствующими процессами визуализации. Каждый процесс рендеринга изолирован и заботится только о запущенной в нем веб-странице.

Мы создали package.jsonфайл для нашего проекта. Теперь мы создадим наше первое настольное приложение с помощью Electron.

Создайте новый файл с именем main.js . Введите в него следующий код -

const {app, BrowserWindow} = require('electron') 
const url = require('url') 
const path = require('path')  

let win  

function createWindow() { 
   win = new BrowserWindow({width: 800, height: 600}) 
   win.loadURL(url.format ({ 
      pathname: path.join(__dirname, 'index.html'), 
      protocol: 'file:', 
      slashes: true 
   })) 
}  

app.on('ready', createWindow)

Создайте еще один файл, на этот раз HTML-файл с именем index.html . Введите в него следующий код.

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Hello World!</title>
   </head>
   
   <body>
      <h1>Hello World!</h1>
      We are using node <script>document.write(process.versions.node)</script>,
      Chrome <script>document.write(process.versions.chrome)</script>,
      and Electron <script>document.write(process.versions.electron)</script>.
   </body>
</html>

Запустите это приложение, используя следующую команду -

$ electron ./main.js

Откроется новое окно. Это будет выглядеть так -

Как это приложение работает?

Мы создали основной файл и файл HTML. В основном файле используются два модуля - app и BrowserWindow . Модуль приложения используется для управления жизненным циклом событий вашего приложения, а модуль BrowserWindow используется для создания окон браузера и управления ими.

Мы определили функцию createWindow , в которой мы создаем новый BrowserWindow и присоединяем URL-адрес к этому BrowserWindow. Это HTML-файл, который отображается и отображается нам при запуске приложения.

Мы использовали собственный объектный процесс Electron в нашем html-файле. Этот объект является расширением объекта процесса Node.js и включает в себя всеt=its функциональные возможности при добавлении многих других.

Пользовательский интерфейс приложений Electron построен с использованием HTML, CSS и JS. Таким образом, здесь мы можем использовать все доступные инструменты для интерфейсной веб-разработки. Вы можете использовать такие инструменты, как Angular, Backbone, React, Bootstrap и Foundation, для создания приложений.

Вы можете использовать Bower для управления этими интерфейсными зависимостями. Установите беседку, используя -

$ npm install -g bower

Теперь вы можете получить все доступные JS- и CSS-фреймворки, библиотеки, плагины и т. Д. С помощью bower. Например, чтобы получить последнюю стабильную версию начальной загрузки, введите следующую команду -

$ bower install bootstrap

Это загрузит bootstrap в bower_components . Теперь вы можете ссылаться на эту библиотеку в своем HTML. Давайте создадим простую страницу, используя эти библиотеки.

Давайте теперь установим jquery с помощью команды npm -

$ npm install --save jquery

Далее это потребуется в нашем файле view.js. У нас уже есть настройка main.js следующим образом:

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

app.on('ready', createWindow)

Откройте свой index.html файл и введите в него следующий код -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Hello World!</title>
      <link rel = "stylesheet" 
         href = "./bower_components/bootstrap/dist/css/bootstrap.min.css" />
   </head>
   
   <body>
      <div class = "container">
         <h1>This page is using Bootstrap and jQuery!</h1>
         <h3 id = "click-counter"></h3>
         <button class = "btn btn-success" id = "countbtn">Click here</button>
         <script src = "./view.js" ></script>
      </div>
   </body>
</html>

Создайте view.js и введите в него логику счетчика кликов -

let $ = require('jquery')  // jQuery now loaded and assigned to $
let count = 0
$('#click-counter').text(count.toString())
$('#countbtn').on('click', () => {
   count ++ 
   $('#click-counter').text(count)
})

Запустите приложение, используя следующую команду -

$ electron ./main.js

Приведенная выше команда сгенерирует вывод, как на следующем снимке экрана -

Вы можете создавать собственное приложение так же, как и веб-сайты. Если вы не хотите, чтобы пользователи были ограничены точным размером окна, вы можете использовать адаптивный дизайн и позволить пользователям гибко использовать ваше приложение.

Работа с файлами - очень важная часть создания настольного приложения. Почти все настольные приложения взаимодействуют с файлами.

Мы создадим форму в нашем приложении, которая будет принимать в качестве входных данных имя и адрес электронной почты. Эта форма будет сохранена в файл, и будет создан список, который покажет это как результат.

Настройте свой основной процесс, используя следующий код в main.js файл -

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

app.on('ready', createWindow)

Теперь откройте index.html файл и введите в него следующий код -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>File System</title>
      <link rel = "stylesheet" 
         href = "./bower_components/bootstrap/dist/css/bootstrap.min.css" />
      
      <style type = "text/css">
         #contact-list {
            height: 150px;
            overflow-y: auto;
         }
      </style>
   </head>
   
   <body>
      <div class = "container">
         <h1>Enter Names and Email addresses of your contacts</h1>
         <div class = "form-group">
            <label for = "Name">Name</label>
            <input type = "text" name = "Name" value = "" id = "Name" 
               placeholder = "Name" class = "form-control" required>
         </div>
         
         <div class = "form-group">
            <label for = "Email">Email</label>
            <input type = "email" name = "Email" value = "" id = "Email" 
               placeholder = "Email" class = "form-control" required>
         </div>
         
         <div class = "form-group">
            <button class = "btn btn-primary" id = "add-to-list">Add to list!</button>
         </div>
         
         <div id = "contact-list">
            <table class = "table-striped" id = "contact-table">
               <tr>
                  <th class = "col-xs-2">S. No.</th>
                  <th class = "col-xs-4">Name</th>
                  <th class = "col-xs-6">Email</th>
               </tr>
            </table>
         </div>
         
         <script src = "./view.js" ></script>
      </div>
   </body>
</html>

Теперь нам нужно обработать событие сложения. Мы сделаем это в нашемview.js файл.

Мы создадим функцию loadAndDisplayContacts (), которая будет изначально загружать контакты из файла. После создания функции loadAndDisplayContacts () мы создадим обработчик кликов на нашемadd to listкнопка. Это добавит запись как в файл, так и в таблицу.

В вашем файле view.js введите следующий код -

let $ = require('jquery')
let fs = require('fs')
let filename = 'contacts'
let sno = 0

$('#add-to-list').on('click', () => {
   let name = $('#Name').val()
   let email = $('#Email').val()

   fs.appendFile('contacts', name + ',' + email + '\n')

   addEntry(name, email)
})

function addEntry(name, email) {
   if(name && email) {
      sno++
      let updateString = '<tr><td>'+ sno + '</td><td>'+ name +'</td><td>' 
         + email +'</td></tr>'
      $('#contact-table').append(updateString)
   }
}

function loadAndDisplayContacts() {  
   
   //Check if file exists
   if(fs.existsSync(filename)) {
      let data = fs.readFileSync(filename, 'utf8').split('\n')
      
      data.forEach((contact, index) => {
         let [ name, email ] = contact.split(',')
         addEntry(name, email)
      })
   
   } else {
      console.log("File Doesn\'t Exist. Creating new file.")
      fs.writeFile(filename, '', (err) => {
         if(err)
            console.log(err)
      })
   }
}

loadAndDisplayContacts()

Теперь запустите приложение, используя следующую команду -

$ electron ./main.js

Как только вы добавите в него несколько контактов, приложение будет выглядеть так:

Для большего fs module API callsобратитесь к руководству по файловой системе узла .

Теперь мы можем обрабатывать файлы с помощью Electron. Мы рассмотрим, как вызывать диалоговые окна сохранения и открытия (собственные) для файлов в главе о диалогах.

В предыдущей главе мы использовали узловой модуль fs. Теперь мы рассмотрим некоторые другие модули узлов, которые мы можем использовать с Electron.

Модуль ОС

Используя модуль ОС, мы можем получить много информации о системе, в которой работает наше приложение. Ниже приведены несколько методов, которые помогают при создании приложения. Эти методы помогают нам настроить приложения в соответствии с ОС, в которой они работают.

Старший Нет Описание функции
1

os.userInfo([options])

В os.userInfo()Метод возвращает информацию о текущем эффективном пользователе. Эта информация может быть использована для персонализации приложения для пользователя даже без явного запроса информации.

2

os.platform()

В os.platform()Метод возвращает строку, определяющую платформу операционной системы. Это можно использовать для настройки приложения в соответствии с операционной системой пользователя.

3

os.homedir()

В os.homedir()Метод возвращает домашний каталог текущего пользователя в виде строки. Как правило, конфигурации всех пользователей находятся в домашнем каталоге пользователя. Так что это можно использовать с той же целью для нашего приложения.

4

os.arch()

В os.arch()Метод возвращает строку, определяющую архитектуру ЦП операционной системы. Это можно использовать при работе на экзотических архитектурах для адаптации вашего приложения к этой системе.

5

os.EOL

Строковая константа, определяющая маркер конца строки для конкретной операционной системы. Это следует использовать всякий раз, когда заканчиваются строки в файлах в ОС хоста.

Используя тот же файл main.js и следующий HTML-файл, мы можем распечатать эти свойства на экране:

<html>
   <head>
      <title>OS Module</title>
   </head>
   
   <body>
      <script>
         let os = require('os')
         document.write('User Info: ' + JSON.stringify(os.userInfo()) + '<br>' + 
            'Platform: ' + os.platform() + '<br>' + 
            'User home directory: ' +  os.homedir() + '<br>' + 
            'OS Architecture: ' + os.arch() + '<br>')
      </script>
   </body>
</html>

Теперь запустите приложение, используя следующую команду -

$ electron ./main.js

Приведенная выше команда сгенерирует следующий вывод -

User Info: {"uid":1000,"gid":1000,"username":"ayushgp","homedir":"/home/ayushgp",
   "shell":"/usr/bin/zsh"}
Platform: linux
User home directory: /home/ayushgp
OS Architecture: x64

Сетевой модуль

Сетевой модуль используется в приложении для сетевой работы. С помощью этого модуля мы можем создавать как серверы, так и сокетные соединения. Как правило, для сетевых задач рекомендуется использовать модуль оболочки из npm, а не сетевой модуль.

В следующих таблицах перечислены наиболее полезные методы из модуля -

Старший Нет Описание функции
1

net.createServer([options][, connectionListener])

Создает новый TCP-сервер. Аргумент connectionListener автоматически устанавливается в качестве прослушивателя для события «соединение».

2

net.createConnection(options[, connectionListener])

Заводской метод, который возвращает новый net.Socket и подключается к предоставленному адресу и порту.

3

net.Server.listen(port[, host][, backlog][, callback])

Начните принимать соединения на указанном порту и хосте. Если хост не указан, сервер будет принимать соединения, направленные на любой адрес IPv4.

4

net.Server.close([callback])

Окончательно закрывается, когда все соединения завершаются и сервер генерирует событие «закрыть».

5

net.Socket.connect(port[, host][, connectListener])

Открывает соединение для данного сокета. Если указаны порт и хост, то сокет будет открыт как TCP-сокет.

Сетевой модуль также имеет несколько других методов. Чтобы получить более полный список, см. Это .

Теперь давайте создадим электронное приложение, которое использует сетевой модуль для создания подключений к серверу. Нам нужно будет создать новый файл,server.js -

var net = require('net');
var server = net.createServer(function(connection) { 
   console.log('Client Connected');
   
   connection.on('end', function() {
      console.log('client disconnected');
   });
   
   connection.write('Hello World!\r\n');
   connection.pipe(connection);
});

server.listen(8080, function() { 
   console.log('Server running on http://localhost:8080');
});

Используя тот же файл main.js, замените HTML-файл следующим:

<html>
   <head>
      <title>net Module</title>
   </head>
   
   <body>
      <script>
         var net = require('net');
         var client = net.connect({port: 8080}, function() {
            console.log('Connection established!');  
         });
         
         client.on('data', function(data) {
            document.write(data.toString());
            client.end();
         });
         
         client.on('end', function() { 
            console.log('Disconnected :(');
         });
      </script>
   </body>
</html>

Запустите сервер, используя следующую команду -

$ node server.js

Запустите приложение, используя следующую команду -

$ electron ./main.js

Приведенная выше команда сгенерирует следующий вывод -

Обратите внимание, что мы подключаемся к серверу автоматически и автоматически отключаемся.

У нас также есть несколько других узловых модулей, которые мы можем использовать непосредственно в интерфейсе с помощью Electron. Использование этих модулей зависит от сценария, в котором вы их используете.

Electron предоставляет нам 2 модуля IPC (Inter Process Communication), которые называются ipcMain и ipcRenderer.

В ipcMainМодуль используется для асинхронного обмена данными между основным процессом и процессами рендеринга. При использовании в основном процессе модуль обрабатывает асинхронные и синхронные сообщения, отправленные из процесса рендеринга (веб-страницы). Сообщения, отправленные от средства визуализации, будут отправлены в этот модуль.

В ipcRendererМодуль используется для асинхронной связи от процесса рендеринга к основному процессу. Он предоставляет несколько методов, позволяющих отправлять синхронные и асинхронные сообщения из процесса рендеринга (веб-страницы) в основной процесс. Вы также можете получать ответы от основного процесса.

Мы создадим главный процесс и процесс рендеринга, которые будут отправлять друг другу сообщения, используя указанные выше модули.

Создайте новый файл с именем main_process.js со следующим содержанием -

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')
const {ipcMain} = require('electron')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

// Event handler for asynchronous incoming messages
ipcMain.on('asynchronous-message', (event, arg) => {
   console.log(arg)

   // Event emitter for sending asynchronous messages
   event.sender.send('asynchronous-reply', 'async pong')
})

// Event handler for synchronous incoming messages
ipcMain.on('synchronous-message', (event, arg) => {
   console.log(arg) 

   // Synchronous event emmision
   event.returnValue = 'sync pong'
})

app.on('ready', createWindow)

Теперь создайте новый index.html файл и добавьте в него следующий код.

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Hello World!</title>
   </head>
   
   <body>
      <script>
         const {ipcRenderer} = require('electron')

         // Synchronous message emmiter and handler
         console.log(ipcRenderer.sendSync('synchronous-message', 'sync ping')) 

         // Async message handler
         ipcRenderer.on('asynchronous-reply', (event, arg) => {
            console.log(arg)
         })

         // Async message sender
         ipcRenderer.send('asynchronous-message', 'async ping')
      </script>
   </body>
</html>

Запустите приложение, используя следующую команду -

$ electron ./main_process.js

Приведенная выше команда сгенерирует следующий вывод -

// On your app console
Sync Pong
Async Pong

// On your terminal where you ran the app
Sync Ping
Async Ping

Рекомендуется не выполнять вычисления тяжелых / блокирующих задач в процессе рендеринга. Всегда используйте IPC, чтобы делегировать эти задачи основному процессу. Это помогает поддерживать темп вашего приложения.

Очень важно, чтобы любое приложение было удобным для пользователя. В результате вам не следует создавать диалоговые окна с помощью вызовов alert (). Electron предоставляет довольно хороший интерфейс для выполнения задачи создания диалоговых окон. Давайте посмотрим на это.

Электрон обеспечивает dialog модуль, который мы можем использовать для отображения собственных системных диалогов для открытия и сохранения файлов, предупреждений и т. д.

Давайте сразу перейдем к примеру и создадим приложение для отображения простых текстовых файлов.

Создайте новый файл main.js и введите в него следующий код -

const {app, BrowserWindow} = require('electron') 
const url = require('url') 
const path = require('path') 
const {ipcMain} = require('electron')  

let win  

function createWindow() { 
   win = new BrowserWindow({width: 800, height: 600}) 
   win.loadURL(url.format ({ 
      pathname: path.join(__dirname, 'index.html'), 
      protocol: 'file:', 
      slashes: true 
   })) 
}  

ipcMain.on('openFile', (event, path) => { 
   const {dialog} = require('electron') 
   const fs = require('fs') 
   dialog.showOpenDialog(function (fileNames) { 
      
      // fileNames is an array that contains all the selected 
      if(fileNames === undefined) { 
         console.log("No file selected"); 
      
      } else { 
         readFile(fileNames[0]); 
      } 
   });
   
   function readFile(filepath) { 
      fs.readFile(filepath, 'utf-8', (err, data) => { 
         
         if(err){ 
            alert("An error ocurred reading the file :" + err.message) 
            return 
         } 
         
         // handle the file content 
         event.sender.send('fileData', data) 
      }) 
   } 
})  
app.on('ready', createWindow)

Этот код будет открывать диалоговое окно открытия всякий раз, когда наш основной процесс получает сообщение openFile от процесса рендеринга. Это сообщение перенаправит содержимое файла обратно в процесс рендеринга. Теперь нам нужно будет распечатать содержимое.

Теперь создайте новый index.html файл со следующим содержанием -

<!DOCTYPE html> 
<html> 
   <head> 
      <meta charset = "UTF-8"> 
      <title>File read using system dialogs</title> 
   </head> 
   
   <body> 
      <script type = "text/javascript"> 
         const {ipcRenderer} = require('electron') 
         ipcRenderer.send('openFile', () => { 
            console.log("Event sent."); 
         }) 
         
         ipcRenderer.on('fileData', (event, data) => { 
            document.write(data) 
         }) 
      </script> 
   </body> 
</html>

Теперь всякий раз, когда мы запускаем наше приложение, появляется собственное открытое диалоговое окно, как показано на следующем снимке экрана -

Как только мы выберем файл для отображения, его содержимое будет отображаться в окне приложения -

Это был лишь один из четырех диалогов, которые предоставляет Electron. Однако все они имеют одинаковое использование. Как только вы научитесь это делать с помощьюshowOpenDialog, то вы можете использовать любой другой диалог.

Диалоги с той же функциональностью:

  • showSaveDialog ([browserWindow,] параметры [, обратный вызов])
  • showMessageDialog ([browserWindow,] параметры [, обратный вызов])
  • showErrorDialog (заголовок, содержание)

В настольных приложениях есть два типа меню: application menu(на верхней панели) и context menu(контекстное меню). В этой главе мы узнаем, как их создать.

Мы будем использовать два модуля - Menu и MenuItem . Обратите внимание, что модули Menu и MenuItem доступны только в основном процессе. Для использования этих модулей в процессе рендеринга вам понадобится удаленный модуль. Мы столкнемся с этим, когда создадим контекстное меню.

Теперь давайте создадим новый main.js файл для основного процесса -

const {app, BrowserWindow, Menu, MenuItem} = require('electron')
const url = require('url')
const path = require('path')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

const template = [
   {
      label: 'Edit',
      submenu: [
         {
            role: 'undo'
         },
         {
            role: 'redo'
         },
         {
            type: 'separator'
         },
         {
            role: 'cut'
         },
         {
            role: 'copy'
         },
         {
            role: 'paste'
         }
      ]
   },
   
   {
      label: 'View',
      submenu: [
         {
            role: 'reload'
         },
         {
            role: 'toggledevtools'
         },
         {
            type: 'separator'
         },
         {
            role: 'resetzoom'
         },
         {
            role: 'zoomin'
         },
         {
            role: 'zoomout'
         },
         {
            type: 'separator'
         },
         {
            role: 'togglefullscreen'
         }
      ]
   },
   
   {
      role: 'window',
      submenu: [
         {
            role: 'minimize'
         },
         {
            role: 'close'
         }
      ]
   },
   
   {
      role: 'help',
      submenu: [
         {
            label: 'Learn More'
         }
      ]
   }
]

const menu = Menu.buildFromTemplate(template)
Menu.setApplicationMenu(menu)
app.on('ready', createWindow)

Здесь мы строим меню по шаблону. Это означает, что мы предоставляем меню функции в формате JSON, а она позаботится обо всем остальном. Теперь нам нужно установить это меню как меню приложений.

Теперь создайте пустой HTML-файл с именем index.html и запустите это приложение, используя -

$ electron ./main.js

В обычном положении меню приложения вы увидите меню, основанное на указанном выше шаблоне.

Мы создали это меню из основного процесса. Давайте теперь создадим контекстное меню для нашего приложения. Сделаем это в нашем HTML-файле -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Menus</title>
   </head>
   
   <body>
      <script type = "text/javascript">
         const {remote} = require('electron')
         const {Menu, MenuItem} = remote

         const menu = new Menu()

         // Build menu one item at a time, unlike
         menu.append(new MenuItem ({
            label: 'MenuItem1',
            click() { 
               console.log('item 1 clicked')
            }
         }))
         
         menu.append(new MenuItem({type: 'separator'}))
         menu.append(new MenuItem({label: 'MenuItem2', type: 'checkbox', checked: true}))
         menu.append(new MenuItem ({
            label: 'MenuItem3',
            click() {
               console.log('item 3 clicked')
            }
         }))

         // Prevent default action of right click in chromium. Replace with our menu.
         window.addEventListener('contextmenu', (e) => {
            e.preventDefault()
            menu.popup(remote.getCurrentWindow())
         }, false)
      </script>
   </body>
</html>

Мы импортировали модули Menu и MenuItem с помощью удаленного модуля; затем мы создали меню и добавили к нему наши пункты меню один за другим. Кроме того, мы предотвратили действие по умолчанию - щелчок правой кнопкой мыши в Chrome и заменили его нашим меню.

Создание меню в Electron - очень простая задача. Теперь вы можете прикрепить к этим элементам свои обработчики событий и обрабатывать события в соответствии с вашими потребностями.

Панель задач - это меню за пределами окна вашего приложения. В MacOS и Ubuntu он находится в правом верхнем углу экрана. В Windows он находится в правом нижнем углу. Мы можем создавать меню для нашего приложения на панели задач с помощью Electron.

Создать новый main.jsфайл и добавьте в него следующий код. Подготовьте файл png для использования в качестве значка на панели задач.

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

app.on('ready', createWindow)

После настройки основного окна браузера мы создадим новый index.html файл со следующим содержанием -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Menus</title>
   </head>
   <body>
      <script type = "text/javascript">
         const {remote} = require('electron')
         const {Tray, Menu} = remote
         const path = require('path')

         let trayIcon = new Tray(path.join('','/home/ayushgp/Desktop/images.png'))

         const trayMenuTemplate = [
            {
               label: 'Empty Application',
               enabled: false
            },
            
            {
               label: 'Settings',
               click: function () {
                  console.log("Clicked on settings")
               }
            },
            
            {
               label: 'Help',
               click: function () {
                  console.log("Clicked on Help")
               }
            }
         ]
         
         let trayMenu = Menu.buildFromTemplate(trayMenuTemplate)
         trayIcon.setContextMenu(trayMenu)
      </script>
   </body>
</html>

Мы создали лоток с помощью субмодуля Tray. Затем мы создали меню, используя шаблон, и прикрепили его к нашему объекту в трее.

Запустите приложение, используя следующую команду -

$ electron ./main.js

Когда вы запустите указанную выше команду, проверьте в панели задач значок, который вы использовали. Я использовал смайлик для своего приложения. Приведенная выше команда сгенерирует следующий вывод -

Electron предоставляет собственный API уведомлений только для MacOS. Поэтому мы не собираемся использовать это, вместо этого мы будем использовать модуль npm под названием node-notifier . Это позволяет нам уведомлять пользователей в Windows, MacOS и Linux.

Установите модуль node-notifier в папку вашего приложения, используя следующую команду в этой папке -

$ npm install --save node-notifier

Давайте теперь создадим приложение, у которого есть кнопка, которая будет генерировать уведомление каждый раз, когда мы нажимаем на эту кнопку.

Создать новый main.js файл и введите в него следующий код -

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

app.on('ready', createWindow)

Давайте теперь создадим нашу веб-страницу и скрипт, который будет запускать уведомление. Создать новыйindex.html файл со следующим кодом -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Menus</title>
   </head>
   
   <body>
      <button type = "button" id = "notify" name = "button">
         Click here to trigger a notification!</button>
      <script type = "text/javascript">
         const notifier = require('node-notifier')
         const path = require('path');
         
         document.getElementById('notify').onclick = (event) => {
            notifier.notify ({
               title: 'My awesome title',
               message: 'Hello from electron, Mr. User!',
               icon: path.join('','/home/ayushgp/Desktop/images.png'),  // Absolute path 
                  (doesn't work on balloons)
               sound: true,  // Only Notification Center or Windows Toasters
               wait: true    // Wait with callback, until user action is taken 
               against notification
            
            }, function (err, response) {
               // Response is response from notification
            });

            notifier.on('click', function (notifierObject, options) {
               console.log("You clicked on the notification")
            });

            notifier.on('timeout', function (notifierObject, options) {
               console.log("Notification timed out!")
            });
         }
      </script>
   </body>
</html>

В notify метод позволяет нам передать его objectwithинформация, такая как заголовок, сообщение, миниатюра и т. д., которая помогает нам настроить уведомление. Мы также можем установить некоторые прослушиватели событий для уведомления.

Теперь запустите приложение, используя следующую команду -

$ electron ./main.js

Когда вы нажмете на созданную нами кнопку, вы увидите собственное уведомление от вашей операционной системы, как показано на следующем снимке экрана -

Мы также обработали события, при которых пользователь нажимает на уведомление или время ожидания уведомления истекает. Эти методы помогают нам сделать приложение более интерактивным, если оно работает в фоновом режиме.

Тег webview используется для встраивания «гостевого» контента, такого как веб-страницы, в ваше приложение Electron. Этот контент содержится в контейнере веб-просмотра. Встроенная страница в вашем приложении определяет, как этот контент будет отображаться.

Веб-просмотр выполняется в отдельном процессе, чем ваше приложение. Чтобы обеспечить защиту от вредоносного содержимого, веб-просмотр не имеет тех же разрешений, что и ваша веб-страница. Это защищает ваше приложение от встроенного контента. Все взаимодействия между вашим приложением и встроенной страницей будут асинхронными.

Давайте рассмотрим пример, чтобы понять встраивание внешней веб-страницы в наше приложение Electron. Мы встроим веб-сайт tutorialspoint в наше приложение с правой стороны. Создать новыйmain.js файл со следующим содержанием -

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

app.on('ready', createWindow)

Теперь, когда мы настроили наш основной процесс, давайте создадим HTML-файл, в который будет встроен веб-сайт tutorialspoint. Создайте файл с именем index.html со следующим содержимым -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Menus</title>
   </head>
   
   <body>
      <div>
         <div>
            <h2>We have the website embedded below!</h2>
         </div>
         <webview id = "foo" src = "https://www.tutorialspoint.com/" style = 
            "width:400px; height:480px;">
            <div class = "indicator"></div>
         </webview>
      </div>
      
      <script type = "text/javascript">
         // Event handlers for loading events.
         // Use these to handle loading screens, transitions, etc
         onload = () => {
            const webview = document.getElementById('foo')
            const indicator = document.querySelector('.indicator')

            const loadstart = () => {
               indicator.innerText = 'loading...'
            }

            const loadstop = () => {
               indicator.innerText = ''
            }

            webview.addEventListener('did-start-loading', loadstart)
            webview.addEventListener('did-stop-loading', loadstop)
         }
      </script>
   </body>
</html>

Запустите приложение, используя следующую команду -

$ electron ./main.js

Приведенная выше команда сгенерирует следующий вывод -

Тег webview можно использовать и для других ресурсов. У элемента webview есть список генерируемых им событий, перечисленных в официальной документации. Вы можете использовать эти события для улучшения функциональности в зависимости от того, что происходит в веб-просмотре.

Всякий раз, когда вы встраиваете сценарии или другие ресурсы из Интернета, рекомендуется использовать webview. Это рекомендуется, поскольку это дает большие преимущества в плане безопасности и не мешает нормальному поведению.

Захват аудио и видео являются важными характеристиками, если вы создаете приложения для совместного использования экрана, голосовых заметок и т. Д. Они также полезны, если вам требуется приложение для захвата изображения профиля.

Мы будем использовать HTML5 API getUserMedia для захвата аудио- и видеопотоков с помощью Electron. Давайте сначала настроим наш основной процесс вmain.js файл следующим образом -

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')

let win

// Set the path where recordings will be saved
app.setPath("userData", __dirname + "/saved_recordings")

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

app.on('ready', createWindow)

Теперь, когда мы настроили наш основной процесс, давайте создадим HTML-файл, который будет захватывать этот контент. Создайте файл с именемindex.html со следующим содержанием -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>Audio and Video</title>
   </head>
   
   <body>
      <video autoplay></video>
      <script type = "text/javascript">
         function errorCallback(e) {
            console.log('Error', e)
         }

         navigator.getUserMedia({video: true, audio: true}, (localMediaStream) => {
            var video = document.querySelector('video')
            video.src = window.URL.createObjectURL(localMediaStream)
            video.onloadedmetadata = (e) => {
               // Ready to go. Do some stuff.
            };
         }, errorCallback)
      </script>
   </body>
</html>

Вышеупомянутая программа сгенерирует следующий вывод -

Теперь у вас есть поток как с веб-камеры, так и с микрофона. Вы можете отправить этот поток по сети или сохранить его в желаемом формате.

Взгляните на документацию MDN для захвата изображений, чтобы получить изображения с вашей веб-камеры и сохранить их. Это было сделано с помощью HTML5 getUserMedia API. Вы также можете захватить рабочий стол пользователя с помощью модуля desktopCapturer, который поставляется с Electron. Давайте теперь посмотрим на пример того, как получить экранный поток.

Используйте тот же файл main.js, что и выше, и отредактируйте файл index.html, чтобы он имел следующий контент:

desktopCapturer.getSources({types: ['window', 'screen']}, (error, sources) => {
   if (error) throw error
   for (let i = 0; i < sources.length; ++i) {
      if (sources[i].name === 'Your Window Name here!') {
         navigator.webkitGetUserMedia({
            audio: false,
            video: {
               mandatory: {
                  chromeMediaSource: 'desktop',
                  chromeMediaSourceId: sources[i].id,
                  minWidth: 1280,
                  maxWidth: 1280,
                  minHeight: 720,
                  maxHeight: 720
               }
            }
         }, handleStream, handleError)
         return
      }
   }
})

function handleStream (stream) {
   document.querySelector('video').src = URL.createObjectURL(stream)
}

function handleError (e) {
   console.log(e)
}

Мы использовали модуль desktopCapturer, чтобы получить информацию о каждом открытом окне. Теперь вы можете захватывать события определенного приложения или всего экрана в зависимости от имени, которое вы передаете вышеуказанномуif statement. Это будет передавать в ваше приложение только то, что происходит на этом экране.

Вы можете обратиться к этому вопросу StackOverflow, чтобы подробно понять его использование.

Обычно мы запоминаем определенные ярлыки для всех приложений, которые мы ежедневно используем на нашем ПК. Чтобы ваши приложения казались интуитивно понятными и легкодоступными для пользователя, вы должны разрешить пользователю использовать ярлыки.

Мы будем использовать модуль globalShortcut для определения ярлыков в нашем приложении. Обратите внимание, чтоAccelerators- это строки, которые могут содержать несколько модификаторов и кодов клавиш, объединенные знаком +. Эти ускорители используются для определения сочетаний клавиш в нашем приложении.

Рассмотрим пример и создадим ярлык. Для этого мы будем следовать примеру диалоговых окон, где мы использовали открытое диалоговое окно для открытия файлов. Мы зарегистрируемCommandOrControl+O ярлык, чтобы открыть диалоговое окно.

Наши main.jsкод останется прежним. Так что создайте новыйmain.js файл и введите в него следующий код -

const {app, BrowserWindow} = require('electron')
const url = require('url')
const path = require('path')
const {ipcMain} = require('electron')

let win

function createWindow() {
   win = new BrowserWindow({width: 800, height: 600})
   win.loadURL(url.format ({
      pathname: path.join(__dirname, 'index.html'),
      protocol: 'file:',
      slashes: true
   }))
}

ipcMain.on('openFile', (event, path) => {
   const {dialog} = require('electron')
   const fs = require('fs')
   dialog.showOpenDialog(function (fileNames) {
         
      // fileNames is an array that contains all the selected
      if(fileNames === undefined)
         console.log("No file selected")
      else
         readFile(fileNames[0])
   })

   function readFile(filepath){
      fs.readFile(filepath, 'utf-8', (err, data) => {
         if(err){
            alert("An error ocurred reading the file :" + err.message)
            return
         }
         
         // handle the file content
         event.sender.send('fileData', data)
      })
   }
})

app.on('ready', createWindow)

Этот код будет открывать открытое диалоговое окно всякий раз, когда наш основной процесс получает сообщение openFile от процесса рендеринга. Раньше это диалоговое окно появлялось при каждом запуске приложения. Давайте теперь ограничим его открытием только при нажатииCommandOrControl+O.

Теперь создайте новый index.html файл со следующим содержанием -

<!DOCTYPE html>
<html>
   <head>
      <meta charset = "UTF-8">
      <title>File read using system dialogs</title>
   </head>
   
   <body>
      <p>Press CTRL/CMD + O to open a file. </p>
      <script type = "text/javascript">
         const {ipcRenderer, remote} = require('electron')
         const {globalShortcut} = remote
         globalShortcut.register('CommandOrControl+O', () => {
            ipcRenderer.send('openFile', () => {
               console.log("Event sent.");
            })
            
            ipcRenderer.on('fileData', (event, data) => {
               document.write(data)
            })
         })
      </script>
   </body>
</html>

Мы зарегистрировали новый ярлык и передали обратный вызов, который будет выполняться всякий раз, когда мы нажимаем на этот ярлык. Мы можем отменить регистрацию ярлыков, когда они нам не нужны.

Теперь, когда приложение откроется, мы получим сообщение об открытии файла с помощью только что определенного ярлыка.

Эти ярлыки можно сделать настраиваемыми, позволяя пользователю выбирать свои собственные ярлыки для определенных действий.

Переменные среды управляют конфигурацией и поведением приложения без изменения кода. Определенное поведение Electron управляется переменными среды, поскольку они инициализируются раньше, чем флаги командной строки и код приложения.

Существует два типа переменных среды, закодированных в электроне: Production variables и Development variables.

Производственные переменные

Следующие переменные среды предназначены для использования во время выполнения в упакованных приложениях Electron.

Старший Нет Переменная и описание
1

GOOGLE_API_KEY

Electron включает в себя жестко запрограммированный ключ API для выполнения запросов к веб-службе геокодирования Google. Поскольку этот ключ API включен в каждую версию Electron, он часто превышает квоту использования.

Чтобы обойти это, вы можете предоставить свой собственный ключ API Google в среде. Поместите следующий код в свой основной файл процесса перед открытием любых окон браузера, которые будут делать запросы геокодирования:

process.env.GOOGLE_API_KEY = 'YOUR_KEY_HERE'
2

ELECTRON_RUN_AS_NODE

Запускает процесс как обычный процесс Node.js.

3

ELECTRON_FORCE_WINDOW_MENU_BAR (Linux Only)

Не используйте строку глобального меню в Linux.

Переменные развития

Следующие переменные среды предназначены в первую очередь для целей разработки и отладки.

Старший Нет Переменная и описание
1

ELECTRON_ENABLE_LOGGING

Выводит на консоль внутренний журнал Chrome.

2

ELECTRON_ENABLE_STACK_DUMPING

Печатает трассировку стека на консоль при сбое Electron.

3

ELECTRON_DEFAULT_ERROR_MODE

Показывает диалоговое окно сбоя Windows при сбое Electron.

Чтобы установить любую из этих переменных среды как истинную, установите ее в консоли. Например, если вы хотите включить ведение журнала, используйте следующие команды -

Для Windows

> set ELECTRON_ENABLE_LOGGING=true

Для Linux

$ export ELECTRON_ENABLE_LOGGING=true

Обратите внимание, что вам нужно будет устанавливать эти переменные среды каждый раз при перезагрузке компьютера. Если вы не хотите этого делать, добавьте эти строки в свой.bashrc файлы.

У нас есть два процесса, которые запускают наше приложение - основной процесс и процесс рендеринга.

Поскольку процесс рендеринга выполняется в окне браузера, мы можем использовать Chrome Devtools для его отладки. Чтобы открыть DevTools, используйте сочетание клавиш «Ctrl + Shift + I» или клавишу <F12>. Вы можете узнать, как использовать инструменты разработчика здесь .

Когда вы откроете DevTools, ваше приложение будет выглядеть так, как показано на следующем снимке экрана -

Отладка основного процесса

DevTools в окне браузера Electron может только отлаживать JavaScript, который выполняется в этом окне (т. Е. На веб-страницах). Для отладки JavaScript, который выполняется в основном процессе, вам нужно будет использовать внешний отладчик и запустить Electron с помощью переключателя --debug или --debug-brk .

Electron будет прослушивать сообщения протокола отладчика V8 на указанном порту; к этому порту необходимо подключить внешний отладчик. Порт по умолчанию - 5858.

Запустите свое приложение, используя следующее -

$ electron --debug = 5858 ./main.js

Теперь вам понадобится отладчик, поддерживающий протокол отладчика V8. Для этого можно использовать VSCode или node-Inspector. Например, давайте выполним следующие шаги и настроим VSCode для этой цели. Выполните следующие действия, чтобы настроить его -

Загрузите и установите VSCode . Откройте свой проект Electron в VSCode.

Добавить файл .vscode/launch.json со следующей конфигурацией -

{
   "version": "1.0.0",
   "configurations": [
      {
         "name": "Debug Main Process",
         "type": "node",
         "request": "launch",
         "cwd": "${workspaceRoot}",
         "runtimeExecutable": "${workspaceRoot}/node_modules/.bin/electron",
         "program": "${workspaceRoot}/main.js"
      }
   ]
}

Note - Для Windows используйте "${workspaceRoot}/node_modules/.bin/electron.cmd" за runtimeExecutable.

Установите несколько точек останова в main.jsи начните отладку в представлении отладки. Когда вы попадете в точки останова, экран будет выглядеть примерно так:

Отладчик VSCode очень мощный и поможет вам быстро исправить ошибки. У вас также есть другие варианты, напримерnode-inspector для отладки электронных приложений.

Упаковка и распространение приложений - неотъемлемая часть процесса разработки настольного приложения. Поскольку Electron - это кроссплатформенный фреймворк для разработки настольных приложений, упаковка и распространение приложений для всех платформ также должны быть удобными.

Электронное сообщество создало проект electronic-packager, который позаботится об этом и для нас. Это позволяет нам упаковывать и распространять наше приложение Electron с пакетами для ОС (.app, .exe и т. Д.) Через JS или CLI.

Поддерживаемые платформы

Electron Packager работает на следующих хост-платформах -

  • Windows (32/64 бит)
  • OS X
  • Linux (x86 / x86_64)

Он генерирует исполняемые файлы / пакеты для следующих целевых платформ -

  • Windows (также известная как win32, для 32/64 бит)
  • OS X (также известный как darwin) / Mac App Store (также известный как mas)
  • Linux (для архитектур x86, x86_64 и armv7l)

Установка

Установите электронный упаковщик, используя -

# for use in npm scripts
$ npm install electron-packager --save-dev

# for use from cli
$ npm install electron-packager -g

Упаковка приложений

В этом разделе мы увидим, как запустить упаковщик из командной строки. Основная форма команды -

electron-packager <sourcedir> <appname> --platform=<platform> --arch=<arch> [optional flags...]

Это будет -

  • Найдите или загрузите правильный выпуск Electron.

  • Используйте эту версию Electron для создания приложения в <output-folder> / <appname> - <platform> - <arch>.

--platform и --archможет быть опущен в двух случаях. Если вы укажете--allвместо этого будут созданы пакеты для всех допустимых комбинаций целевых платформ / архитектур. В противном случае будет создан единый комплект для платформы / архитектуры хоста.

Мы использовали следующие ресурсы, чтобы узнать больше об Electron. Мы упоминали об этом при создании этого руководства.

Самый важный ресурс - это документация Electron. В документации подробно описаны почти все особенности и особенности фреймворка. Их достаточно, чтобы вы смогли создать приложение.

Есть также несколько очень хороших примеров Electron, представленных в репозитории электронных образцов приложений .

Видео ресурсы

Настольные приложения с веб-языками

Быстрая разработка кроссплатформенных настольных приложений с использованием JavaScript и Electron

Сообщения в блоге

Создание настольного приложения с помощью Electron

Создайте музыкальный плеер с React и Electron

Создание вашего первого настольного приложения с помощью HTML, JS и Electron

Создание кроссплатформенных настольных узловых приложений с помощью Electron