electron-node-red/main.js

409 wiersze
14 KiB
JavaScript
Czysty Zwykły widok Historia

2016-05-06 12:18:15 +00:00
2016-03-09 21:49:41 +00:00
'use strict';
// Some settings you can edit easily
const editable = true; // Set this to false to create a run only application - no editor/no console
const allowLoadSave = false; // set to true to allow omport and export of flow
const showMap = false; // set to true to add Worldmap to the menu
let flowfile = 'electronflow.json'; // default Flows file name - loaded at start
const urldash = "/ui/#/0"; // Start on the dashboard page
const urledit = "/red"; // url for the editor page
const urlconsole = "/console.htm"; // url for the console page
const urlmap = "/worldmap"; // url for the worldmap
// tcp port to use
//const listenPort = "18880"; // fix it if you like
const listenPort = parseInt(Math.random()*16383+49152) // or random ephemeral port
2016-10-30 22:06:48 +00:00
const os = require('os');
const fs = require('fs');
2018-08-20 20:16:39 +00:00
const url = require('url');
const path = require('path');
const http = require('http');
const express = require("express");
2016-03-09 21:49:41 +00:00
const electron = require('electron');
2018-08-20 20:16:39 +00:00
2016-03-09 21:49:41 +00:00
const app = electron.app;
const ipc = electron.ipcMain;
const dialog = electron.dialog;
2016-03-09 21:49:41 +00:00
const BrowserWindow = electron.BrowserWindow;
2016-06-02 11:44:55 +00:00
const {Menu, MenuItem} = electron;
// this should be placed at top of main.js to handle squirrel setup events quickly
if (handleSquirrelEvent()) { return; }
2016-10-30 22:06:48 +00:00
2016-03-09 21:49:41 +00:00
var RED = require("node-red");
var red_app = express();
// Add a simple route for static content served from 'public'
2018-08-20 20:16:39 +00:00
red_app.use("/",express.static("web"));
//red_app.use(express.static(__dirname +"/public"));
2016-03-09 21:49:41 +00:00
// Create a server
var server = http.createServer(red_app);
var userdir = __dirname;
if (editable) {
if (process.argv[1] && (process.argv[1] === "main.js")) {
userdir = __dirname;
2016-10-30 22:06:48 +00:00
}
else { // We set the user directory to be in the users home directory...
userdir = os.homedir() + '/.node-red';
if (!fs.existsSync(userdir)) {
fs.mkdirSync(userdir);
}
if (!fs.existsSync(userdir+"/"+flowfile)) {
fs.writeFileSync(userdir+"/"+flowfile, fs.readFileSync(__dirname+"/"+flowfile));
}
2016-10-30 22:06:48 +00:00
}
}
console.log("UserDir :",userdir);
2018-08-20 20:16:39 +00:00
// console.log("PORT",listenPort);
// Keep a global reference of the window objects, if you don't, the window will
// be closed automatically when the JavaScript object is garbage collected.
let mainWindow;
let conWindow;
2018-08-21 14:46:46 +00:00
let logBuffer = [];
let logLength = 250;
2016-10-30 22:06:48 +00:00
ipc.on('clearLogBuffer', function(event, arg) { logBuffer = []; });
2016-03-09 21:49:41 +00:00
// Create the settings object - see default settings.js file for other options
var settings = {
httpAdminRoot: "/red", // set to false to disable editor/deploy
2016-03-09 21:49:41 +00:00
httpNodeRoot: "/",
2016-10-30 22:06:48 +00:00
userDir: userdir,
2016-10-31 23:22:06 +00:00
flowFile: flowfile,
editorTheme: { projects:{ enabled:false } },
2018-08-20 20:16:39 +00:00
functionGlobalContext: { }, // enables global context
logging: {
websock: {
level: 'info',
metrics: false,
handler: function() {
return function(msg) {
if (editable) { // No logging if not editable
var ts = (new Date(msg.timestamp)).toISOString();
ts = ts.replace("Z"," ").replace("T"," ");
var line = ts+" : "+msg.msg;
logBuffer.push(line);
if (conWindow) { conWindow.webContents.send('debugMsg', line); }
if (logBuffer.length > logLength) { logBuffer.shift(); }
}
2018-08-20 20:16:39 +00:00
}
}
}
}
2016-03-09 21:49:41 +00:00
};
if (!editable) {
settings.httpAdminRoot = false;
settings.readOnly = true;
}
2016-03-09 21:49:41 +00:00
// Initialise the runtime with a server and settings
RED.init(server,settings);
// Serve the editor UI from /red
if (settings.httpAdminRoot !== false) {
red_app.use(settings.httpAdminRoot,RED.httpAdmin);
}
2016-03-09 21:49:41 +00:00
2018-08-20 20:16:39 +00:00
// Serve the http nodes UI from /
2016-03-09 21:49:41 +00:00
red_app.use(settings.httpNodeRoot,RED.httpNode);
2016-05-06 12:18:15 +00:00
// Create the Application's main menu
var template = [
// {label: "Application",
// submenu: [
// //{ role: 'about' },
// //{ type: "separator" },
// { role: 'togglefullscreen' },
// { role: 'quit' }
// ]},
{ label: 'Node-RED',
submenu: [
{ label: 'Import Flow',
click() { openFlow(); }
},
{ label: 'Save Flow As',
click() { saveFlow(); }
2018-08-20 20:16:39 +00:00
},
{ type: 'separator' },
{ label: 'Console',
accelerator: "Shift+CmdOrCtrl+C",
click() { createConsole(); }
},
{ label: 'Dashboard',
accelerator: "Shift+CmdOrCtrl+D",
click() { mainWindow.loadURL("http://localhost:"+listenPort+urldash); }
2016-12-29 11:56:09 +00:00
},
{ label: 'Editor',
accelerator: "Shift+CmdOrCtrl+E",
click() { mainWindow.loadURL("http://localhost:"+listenPort+urledit); }
2018-08-21 13:46:02 +00:00
},
{ label: 'Worldmap',
accelerator: "Shift+CmdOrCtrl+M",
click() { mainWindow.loadURL("http://localhost:"+listenPort+urlmap); }
2016-11-01 08:54:10 +00:00
},
{ type: 'separator' },
{ label: 'Documentation',
click() { electron.shell.openExternal('https://nodered.org/docs') }
},
{ label: 'Flows and Nodes',
click() { electron.shell.openExternal('https://flows.nodered.org') }
2018-08-21 16:25:48 +00:00
},
{ label: 'Discourse Forum',
click() { electron.shell.openExternal('https://discourse.nodered.org/') }
},
{ type: "separator" },
{ role: 'togglefullscreen' },
{ role: 'quit' }
]}
// ,{label: "Edit",
2018-08-21 16:25:48 +00:00
// submenu: [
// { label: "Undo", accelerator: "CmdOrCtrl+Z", selector: "undo:" },
// { label: "Redo", accelerator: "Shift+CmdOrCtrl+Z", selector: "redo:" },
// { type: "separator" },
// { label: "Cut", accelerator: "CmdOrCtrl+X", selector: "cut:" },
// { label: "Copy", accelerator: "CmdOrCtrl+C", selector: "copy:" },
// { label: "Paste", accelerator: "CmdOrCtrl+V", selector: "paste:" },
// { label: "Select All", accelerator: "CmdOrCtrl+A", selector: "selectAll:" }
// ]}
// ,{ label: 'View',
// submenu: [
// { label: 'Reload',
// accelerator: 'CmdOrCtrl+R',
// click(item, focusedWindow) { if (focusedWindow) { focusedWindow.reload(); }}
// },
// { label: 'Toggle Developer Tools',
// accelerator: process.platform === 'darwin' ? 'Alt+Command+I' : 'Ctrl+Shift+I',
// click(item, focusedWindow) { if (focusedWindow) { focusedWindow.webContents.toggleDevTools(); }}
// },
// { type: 'separator' },
// { role: 'resetzoom' },
// { role: 'zoomin' },
// { role: 'zoomout' },
// { type: 'separator' },
// { role: 'togglefullscreen' },
// { role: 'minimize' }
// ]}
2016-05-06 12:18:15 +00:00
];
if (!showMap) { template[0].submenu.splice(6,1); }
if (!editable) {
template[0].submenu.splice(3,1);
template[0].submenu.splice(4,1);
}
if (!allowLoadSave) { template[0].submenu.splice(0,2); }
let fileName = ""
function saveFlow() {
dialog.showSaveDialog({
filters:[{ name:'JSON', extensions:['json'] }],
defaultPath: fileName
}, function(file_path) {
if (file_path) {
2018-08-25 22:17:01 +00:00
var flo = JSON.stringify(RED.nodes.getFlows().flows);
fs.writeFile(file_path, flo, function(err) {
if (err) { dialog.showErrorBox('Error', err); }
2018-08-25 22:17:01 +00:00
else {
dialog.showMessageBox({
icon: "nodered.png",
message:"Flow file saved as\n\n"+file_path,
buttons: ["OK"]
});
}
});
}
});
}
function openFlow() {
dialog.showOpenDialog({ filters:[{ name:'JSON', extensions:['json']} ]},
function (fileNames) {
if (fileNames) {
//console.log(fileNames[0]);
fs.readFile(fileNames[0], 'utf-8', function (err, data) {
try {
var flo = JSON.parse(data);
if (Array.isArray(flo) && (flo.length > 0)) {
RED.nodes.setFlows(flo,"full");
fileName = fileNames[0];
}
else {
dialog.showErrorBox("Error", "Failed to parse flow file.\n\n "+fileNames[0]+".\n\nAre you sure it's a flow file ?");
}
}
catch(e) {
dialog.showErrorBox("Error", "Failed to load flow file.\n\n "+fileNames[0]);
}
});
}
}
)
}
2018-08-21 13:46:02 +00:00
2018-08-21 16:25:48 +00:00
// Create the console log window
2018-08-20 20:16:39 +00:00
function createConsole() {
2018-08-21 13:46:02 +00:00
if (conWindow) { conWindow.show(); return; }
2018-08-20 20:16:39 +00:00
// Create the hidden console window
conWindow = new BrowserWindow({
2018-08-21 16:25:48 +00:00
title: "Node-RED Console",
width: 800,
height: 600,
icon: path.join(__dirname, 'nodered.png'),
autoHideMenuBar: true
2018-08-20 20:16:39 +00:00
});
conWindow.loadURL(url.format({
pathname: path.join(__dirname, 'console.htm'),
protocol: 'file:',
slashes: true
}))
conWindow.webContents.on('did-finish-load', () => {
2018-08-21 14:46:46 +00:00
conWindow.webContents.send('logBuff', logBuffer);
2018-08-20 20:16:39 +00:00
});
2018-08-21 16:25:48 +00:00
conWindow.on('closed', () => {
2018-08-20 20:16:39 +00:00
conWindow = null;
});
//conWindow.webContents.openDevTools();
}
2016-03-09 21:49:41 +00:00
2018-08-21 16:25:48 +00:00
// Create the main browser window
2016-11-01 08:54:10 +00:00
function createWindow() {
2016-05-06 12:18:15 +00:00
mainWindow = new BrowserWindow({
title: "Node-RED",
2016-06-15 19:08:26 +00:00
//titleBarStyle: "hidden",
2016-05-06 12:18:15 +00:00
width: 1024,
height: 768,
icon: path.join(__dirname, 'nodered.png'),
fullscreenable: true,
autoHideMenuBar: true,
webPreferences: {
nodeIntegration: false
}
2016-05-06 12:18:15 +00:00
});
mainWindow.loadURL(`file://${__dirname}/load.html`);
//if (process.platform !== 'darwin') { mainWindow.setAutoHideMenuBar(true); }
2016-05-06 12:18:15 +00:00
2018-08-20 20:16:39 +00:00
mainWindow.webContents.on('did-get-response-details', function(event, status, newURL, originalURL, httpResponseCode) {
if ((httpResponseCode == 404) && (newURL == ("http://localhost:"+listenPort+urldash))) {
setTimeout(mainWindow.webContents.reload, 250);
2016-05-06 12:18:15 +00:00
}
Menu.setApplicationMenu(Menu.buildFromTemplate(template));
});
2018-08-20 20:16:39 +00:00
// mainWindow.webContents.on('did-finish-load', () => {
// console.log("LOADED DASHBOARD");
// });
2016-05-06 12:18:15 +00:00
mainWindow.webContents.on("new-window", function(e, url, frameName, disposition, options) {
// if a child window opens... modify any other options such as width/height, etc
// in this case make the child overlap the parent exactly...
var w = mainWindow.getBounds();
options.x = w.x;
options.y = w.y;
options.width = w.width;
options.height = w.height;
//re-use the same child name so all "2nd" windows use the same one.
//frameName = "child";
})
2018-08-21 16:25:48 +00:00
mainWindow.on('closed', () => {
2016-05-06 12:18:15 +00:00
mainWindow = null;
});
2018-08-21 16:25:48 +00:00
// Start the app full screen
//mainWindow.setFullScreen(true)
// Open the DevTools at start
2018-08-21 16:25:48 +00:00
//mainWindow.webContents.openDevTools();
2016-03-09 21:49:41 +00:00
}
// Called when Electron has finished initialization and is ready to create browser windows.
2018-08-20 20:16:39 +00:00
app.on('ready', createWindow );
2016-03-09 21:49:41 +00:00
// Quit when all windows are closed.
app.on('window-all-closed', function () {
2016-05-06 12:18:15 +00:00
// On OS X it is common for applications and their menu bar
// to stay active until the user quits explicitly with Cmd + Q
2018-08-20 20:16:39 +00:00
if (process.platform !== 'darwin') { app.quit(); }
2016-03-09 21:49:41 +00:00
});
2016-11-01 08:54:10 +00:00
app.on('activate', function() {
2016-05-06 12:18:15 +00:00
// On OS X it's common to re-create a window in the app when the
// dock icon is clicked and there are no other windows open.
if (mainWindow === null) {
createWindow();
2018-08-20 20:16:39 +00:00
mainWindow.loadURL("http://127.0.0.1:"+listenPort+urldash);
2016-05-06 12:18:15 +00:00
}
2016-03-09 21:49:41 +00:00
});
// Start the Node-RED runtime, then load the inital dashboard page
RED.start().then(function() {
server.listen(listenPort,"127.0.0.1",function() {
2018-08-20 20:16:39 +00:00
mainWindow.loadURL("http://127.0.0.1:"+listenPort+urldash);
});
});
2016-10-30 22:06:48 +00:00
///////////////////////////////////////////////////////
2016-10-30 22:06:48 +00:00
// All this Squirrel stuff is for the Windows installer
function handleSquirrelEvent() {
2018-08-21 16:25:48 +00:00
if (process.argv.length === 1) { return false; }
const path = require('path');
const ChildProcess = require('child_process');
const appFolder = path.resolve(process.execPath, '..');
const rootAtomFolder = path.resolve(appFolder, '..');
const updateDotExe = path.resolve(path.join(rootAtomFolder, 'Update.exe'));
const exeName = path.basename(process.execPath);
const spawn = function(command, args) {
let spawnedProcess, error;
try { spawnedProcess = ChildProcess.spawn(command, args, {detached: true}); }
catch (error) {}
return spawnedProcess;
};
const spawnUpdate = function(args) {
return spawn(updateDotExe, args);
};
const squirrelEvent = process.argv[1];
switch (squirrelEvent) {
case '--squirrel-install':
case '--squirrel-updated':
// Optionally do things such as:
// - Add your .exe to the PATH
// - Write to the registry for things like file associations and
// explorer context menus
// Install desktop and start menu shortcuts
spawnUpdate(['--createShortcut', exeName]);
setTimeout(app.quit, 1000);
return true;
case '--squirrel-uninstall':
// Undo anything you did in the --squirrel-install and
// --squirrel-updated handlers
// Remove desktop and start menu shortcuts
spawnUpdate(['--removeShortcut', exeName]);
setTimeout(app.quit, 1000);
return true;
case '--squirrel-obsolete':
// This is called on the outgoing version of your app before
// we update to the new version - it's the opposite of
// --squirrel-updated
app.quit();
return true;
}
2018-08-20 20:16:39 +00:00
}