kopia lustrzana https://github.com/dceejay/electron-node-red
335 wiersze
10 KiB
JavaScript
Executable File
335 wiersze
10 KiB
JavaScript
Executable File
|
|
'use strict';
|
|
|
|
// Some settings you can edit easily
|
|
// Flows file name
|
|
const flowfile = 'electronflow.json';
|
|
// Start on the dashboard page
|
|
const urldash = "/ui/#/0";
|
|
// url for the editor page
|
|
const urledit = "/red";
|
|
// url for the console page
|
|
const urlconsole = "/console.htm";
|
|
// url for the worldmap page
|
|
const urlmap = "/worldmap";
|
|
// tcp port to use
|
|
//const listenPort = "18880"; // fix it just because
|
|
const listenPort = parseInt(Math.random()*16383+49152) // or random ephemeral port
|
|
|
|
const os = require('os');
|
|
const url = require('url');
|
|
const path = require('path');
|
|
const electron = require('electron');
|
|
|
|
const app = electron.app;
|
|
const BrowserWindow = electron.BrowserWindow;
|
|
const {Menu, MenuItem} = electron;
|
|
|
|
// this should be placed at top of main.js to handle squirrel setup events quickly
|
|
if (handleSquirrelEvent()) { return; }
|
|
|
|
var http = require('http');
|
|
var express = require("express");
|
|
var RED = require("node-red");
|
|
|
|
// Create an Express app
|
|
var red_app = express();
|
|
|
|
// Add a simple route for static content served from 'public'
|
|
red_app.use("/",express.static("web"));
|
|
//red_app.use(express.static(__dirname +"/public"));
|
|
|
|
// Create a server
|
|
var server = http.createServer(red_app);
|
|
|
|
var userdir;
|
|
if (process.argv[1] && (process.argv[1] === "main.js")) {
|
|
userdir = __dirname;
|
|
}
|
|
else { // We set the user directory to be in the users home directory...
|
|
const fs = require('fs');
|
|
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));
|
|
}
|
|
}
|
|
console.log("Setting UserDir to ",userdir);
|
|
// console.log("DIR",__dirname);
|
|
// 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;
|
|
|
|
// Create the settings object - see default settings.js file for other options
|
|
var settings = {
|
|
httpAdminRoot: "/red",
|
|
httpNodeRoot: "/",
|
|
userDir: userdir,
|
|
flowFile: flowfile,
|
|
editorTheme: { projects:{ enabled:true } },
|
|
functionGlobalContext: { }, // enables global context
|
|
logging: {
|
|
websock: {
|
|
level: 'info',
|
|
metrics: false,
|
|
handler: function() {
|
|
return function(msg) {
|
|
if (conWindow) { conWindow.webContents.send('debugMsg', msg); }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
// Initialise the runtime with a server and settings
|
|
RED.init(server,settings);
|
|
|
|
// Serve the editor UI from /red
|
|
red_app.use(settings.httpAdminRoot,RED.httpAdmin);
|
|
|
|
// Serve the http nodes UI from /
|
|
red_app.use(settings.httpNodeRoot,RED.httpNode);
|
|
|
|
// Create the Application's main menu
|
|
var template = [{
|
|
label: "Application",
|
|
submenu: [
|
|
{ role: 'about' },
|
|
{ type: "separator" },
|
|
{ role: 'quit' }
|
|
]}, {
|
|
label: 'Node-RED',
|
|
submenu: [
|
|
{ label: 'Console',
|
|
accelerator: "Shift+CmdOrCtrl+C",
|
|
click() { createConsole(); }
|
|
},
|
|
{ label: 'Dashboard',
|
|
accelerator: "Shift+CmdOrCtrl+D",
|
|
click() { mainWindow.loadURL("http://localhost:"+listenPort+urldash); }
|
|
},
|
|
{ label: 'Editor',
|
|
accelerator: "Shift+CmdOrCtrl+E",
|
|
click() { mainWindow.loadURL("http://localhost:"+listenPort+urledit); }
|
|
},
|
|
{ label: 'Worldmap',
|
|
accelerator: "Shift+CmdOrCtrl+M",
|
|
click() { mainWindow.loadURL("http://localhost:"+listenPort+urlmap); }
|
|
},
|
|
{ type: 'separator' },
|
|
{ label: 'Documentation',
|
|
click() { require('electron').shell.openExternal('https://nodered.org/docs') }
|
|
},
|
|
{ label: 'Flows and Nodes',
|
|
click() { require('electron').shell.openExternal('https://flows.nodered.org') }
|
|
},
|
|
{ label: 'Discourse Forum',
|
|
click() { require('electron').shell.openExternal('https://discourse.nodered.org/') }
|
|
}
|
|
]}, {
|
|
label: "Edit",
|
|
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' }
|
|
]}
|
|
];
|
|
|
|
// function openFlow() {
|
|
// dialog.showOpenDialog(function (fileNames) {
|
|
// if (fileNames === undefined) {
|
|
// console.log("No file selected");
|
|
// }
|
|
// else {
|
|
// console.log(fileNames[0]);
|
|
// //readFile(fileNames[0]);
|
|
// }
|
|
// });
|
|
// }
|
|
|
|
function createConsole() {
|
|
if (conWindow) { conWindow.show(); return; }
|
|
// Create the hidden console window
|
|
conWindow = new BrowserWindow({
|
|
title:"Node-RED Console", width:800, height:600, frame:true, show:true
|
|
});
|
|
//conWindow.loadURL("http://localhost:"+listenPort+urlconsole);
|
|
conWindow.loadURL(url.format({
|
|
pathname: path.join(__dirname, 'console.htm'),
|
|
protocol: 'file:',
|
|
slashes: true
|
|
}))
|
|
conWindow.webContents.on('did-finish-load', () => {
|
|
//console.log("LOADED CONSOLE");
|
|
conWindow.webContents.send('debugMsg', "Ready");
|
|
});
|
|
conWindow.on('closed', function() {
|
|
conWindow = null;
|
|
});
|
|
//conWindow.webContents.openDevTools();
|
|
}
|
|
|
|
function createWindow() {
|
|
// Create the browser window.
|
|
mainWindow = new BrowserWindow({
|
|
autoHideMenuBar: true,
|
|
webPreferences: {
|
|
nodeIntegration: false
|
|
},
|
|
title: "Node-RED",
|
|
fullscreenable: true,
|
|
//titleBarStyle: "hidden",
|
|
width: 1024,
|
|
height: 768,
|
|
icon: __dirname + "/nodered.png"
|
|
});
|
|
|
|
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);
|
|
}
|
|
Menu.setApplicationMenu(Menu.buildFromTemplate(template));
|
|
});
|
|
|
|
// mainWindow.webContents.on('did-finish-load', () => {
|
|
// console.log("LOADED DASHBOARD");
|
|
// });
|
|
|
|
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";
|
|
})
|
|
|
|
// Open the DevTools.
|
|
//mainWindow.webContents.openDevTools();
|
|
|
|
// Emitted when the window is closed.
|
|
mainWindow.on('closed', function() {
|
|
mainWindow = null;
|
|
});
|
|
}
|
|
|
|
// Called when Electron has finished initialization and is ready to create browser windows.
|
|
app.on('ready', createWindow );
|
|
|
|
// Quit when all windows are closed.
|
|
app.on('window-all-closed', function () {
|
|
// On OS X it is common for applications and their menu bar
|
|
// to stay active until the user quits explicitly with Cmd + Q
|
|
if (process.platform !== 'darwin') { app.quit(); }
|
|
});
|
|
|
|
app.on('activate', function() {
|
|
// 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();
|
|
mainWindow.loadURL("http://127.0.0.1:"+listenPort+urldash);
|
|
}
|
|
});
|
|
|
|
// Start the Node-RED runtime, then load the inital page
|
|
RED.start().then(function() {
|
|
server.listen(listenPort,"127.0.0.1",function() {
|
|
mainWindow.loadURL("http://127.0.0.1:"+listenPort+urldash);
|
|
});
|
|
});
|
|
|
|
///////////////////////////////////////////////////////
|
|
// All this Squirrel stuff is for the Windows installer
|
|
function handleSquirrelEvent() {
|
|
if (process.argv.length === 1) {
|
|
return false;
|
|
}
|
|
|
|
const ChildProcess = require('child_process');
|
|
const path = require('path');
|
|
|
|
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;
|
|
}
|
|
}
|