main.js 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. 'use strict'
  2. import { app, BrowserWindow, ipcMain } from 'electron';
  3. import * as path from 'path';
  4. import { format as formatUrl } from 'url';
  5. import { exec } from 'child_process';
  6. app.allowRendererProcessReuse = false;
  7. const isDevelopment = process.env.NODE_ENV !== 'production';
  8. const platform = process.platform;
  9. let mainWindow = null, subWindow1 = null, serWindow1;
  10. let random = null;
  11. const version = app.getVersion();
  12. let win1 = null, win2 = null, about = null, esptools = null;
  13. let tmp = null, appIcon = null, menu = null, trayImg = null;
  14. // ------------ Keep only one instance of the app running ------------ //
  15. // const singleLock = app.requestSingleInstanceLock();
  16. // !singleLock ? app.quit() : '';
  17. const gotTheLock = app.requestSingleInstanceLock()
  18. if (!gotTheLock) {
  19. app.quit()
  20. } else {
  21. app.on('second-instance', (event, commandLine, workingDirectory) => {
  22. if (mainWindow) {
  23. if (mainWindow.isMinimized()) mainWindow.restore()
  24. mainWindow.focus()
  25. }
  26. })
  27. // Create myWindow, load the rest of the app, etc...
  28. app.whenReady().then(() => {
  29. })
  30. }
  31. // ------------ run auto launch & add ca cert in MacOS ------------ //
  32. if (platform === 'darwin') {
  33. app.dock.hide();
  34. const sudo = require('sudo-prompt');
  35. const AutoLaunch = require('auto-launch');
  36. const opts = {
  37. name: "CocoRoboVDeskTop",
  38. isHidden: false,
  39. mac: { useLaunchAgent: true }
  40. };
  41. autoLaunch(opts);
  42. exec(`"${path.join(__dirname, '..', 'verifyCert.bash')}"`, (err, stdout) => {
  43. if (!err && stdout.indexOf('No') > -1)
  44. sudo.exec(
  45. `security add-trusted-cert -d -r trustRoot -k "/Library/Keychains/System.keychain" "${path.join(__dirname, '..', 'ca.crt')}"`,
  46. { name: 'CocoRobo X Uploader' },
  47. err => console.log(err)
  48. );
  49. else if (err)
  50. console.error(err);
  51. });
  52. /**
  53. * Auto launch when computer boot.
  54. * @opts {object} Auto lanuch's options
  55. * @returns {void} void
  56. */
  57. function autoLaunch(opts) {
  58. const autoLaunch = new AutoLaunch(opts);
  59. autoLaunch
  60. .isEnabled()
  61. .then(enabled => enabled ? '' : autoLaunch.enable())
  62. .catch(err => console.error(err));
  63. return;
  64. }
  65. }
  66. /**
  67. * Reboot the app.
  68. * @returns {void} void
  69. */
  70. function reboot() {
  71. app.relaunch();
  72. app.exit();
  73. return;
  74. }
  75. /**
  76. * Get windows id & return a list.
  77. * @param {string} data Incoming message
  78. * @returns {Array<number>} An id list
  79. */
  80. function getWindowsId(data) {
  81. const result = [];
  82. if (data === "already") {
  83. const windows = BrowserWindow.getAllWindows();
  84. for (let i = windows.length - 1; i >= 0; i--)
  85. result.push(windows[i].id);
  86. platform === 'darwin' ? result.reverse() : '';
  87. }
  88. return result;
  89. }
  90. // ------------ IpcMain eventlisteners ------------ //
  91. ipcMain.on('tmp', (e, path) => tmp = path);
  92. ipcMain.on('version', (e, data) => e.returnValue = data === "already" ? version : '');
  93. ipcMain.on('relaunch', reboot);
  94. ipcMain.on('windowsid', (e, data) => e.returnValue = getWindowsId(data));
  95. ipcMain.on('download', (event, url, filePath) => {
  96. const request = require('request')
  97. const progress = require('request-progress')
  98. progress(request(url))
  99. .on('progress', (state) => {
  100. event.sender.send('download-progress', state.percent)
  101. })
  102. .on('error', (err) => {
  103. event.sender.send('download-error', err)
  104. })
  105. .pipe(fs.createWriteStream(filePath))
  106. .on('finish', () => {
  107. event.sender.send('download-finished')
  108. })
  109. })
  110. // ------------ When app is ready ------------ //
  111. app.on('ready', () => {
  112. //createSubWindow1();
  113. //createSubWindow2();
  114. mainWindow = createMainWindow();
  115. });
  116. // when user acitvate the app, it's useful in mac only
  117. app.on('activate', () => {
  118. // on macOS it is common to re-create a window even after all windows have been closed
  119. mainWindow === null ? mainWindow = createMainWindow() : null;
  120. });
  121. function createMainWindow() {
  122. let window = new BrowserWindow({ autoHideMenuBar: true, show: false, webPreferences: { nodeIntegration: true }, width: 1280, height: 920, minWidth: 601, minHeight: 670 });
  123. let grantedDeviceThroughPermHandler;
  124. window.webContents.session.on('select-usb-device', (event, details, callback) => {
  125. //Add events to handle devices being added or removed before the callback on
  126. //`select-usb-device` is called.
  127. window.webContents.session.on('usb-device-added', (event, device) => {
  128. console.log('usb-device-added FIRED WITH', device)
  129. //Optionally update details.deviceList
  130. })
  131. window.webContents.session.on('usb-device-removed', (event, device) => {
  132. console.log('usb-device-removed FIRED WITH', device)
  133. //Optionally update details.deviceList
  134. })
  135. event.preventDefault()
  136. if (details.deviceList && details.deviceList.length > 0) {
  137. const deviceToReturn = details.deviceList.find((device) => {
  138. // if (!grantedDeviceThroughPermHandler || (device.deviceId != grantedDeviceThroughPermHandler.deviceId)) {
  139. // return true
  140. // }
  141. return true;
  142. })
  143. try {
  144. if (callback) {
  145. if (deviceToReturn) {
  146. callback(deviceToReturn.deviceId)
  147. } else {
  148. callback()
  149. }
  150. }
  151. }
  152. catch (e) {
  153. console.log(e)
  154. }
  155. return deviceToReturn;
  156. }
  157. })
  158. window.webContents.session.setPermissionCheckHandler((webContents, permission, requestingOrigin, details) => {
  159. if (permission === 'usb' && details.securityOrigin === 'file:///') {
  160. return true
  161. }
  162. })
  163. window.webContents.session.setDevicePermissionHandler((details) => {
  164. if (details.deviceType === 'usb' && details.origin === 'file://') {
  165. return true;
  166. // if (!grantedDeviceThroughPermHandler) {
  167. // grantedDeviceThroughPermHandler = details.device
  168. // return true
  169. // } else {
  170. // return false
  171. // }
  172. }
  173. })
  174. if (isDevelopment) {
  175. window.loadURL(`http://localhost:${process.env.ELECTRON_WEBPACK_WDS_PORT}`);
  176. } else {
  177. window.loadURL(formatUrl({
  178. pathname: path.join(__dirname, 'cocoblockly-x/index.html'),
  179. protocol: 'file',
  180. slashes: true,
  181. search: "lang=zh-hans",
  182. }));
  183. }
  184. window.once('ready-to-show', () => window.show());
  185. window.on('closed', () => { app.quit() });
  186. // window.webContents.on('devtools-opened', () => window.focus());
  187. // window.webContents.openDevTools({ mode: 'undocked' });
  188. return window;
  189. }
  190. function createSubWindow1() {
  191. let window = new BrowserWindow({ autoHideMenuBar: true, webPreferences: { nodeIntegration: true }, width: 0, height: 0, show: false });
  192. // window.webContents.openDevTools({ mode: "undocked" });
  193. if (isDevelopment) {
  194. window.loadURL(`http://localhost:${process.env.ELECTRON_WEBPACK_WDS_PORT}/index.html`);
  195. }
  196. else {
  197. window.loadURL(formatUrl({
  198. pathname: path.join(__dirname, 'index.html'),
  199. protocol: 'file',
  200. slashes: true
  201. }));
  202. }
  203. window.on('closed', () => window = null);
  204. // window.webContents.on('devtools-opened', () => setImmediate(() => window.focus()));
  205. // window.webContents.openDevTools({ mode: 'undocked' });
  206. return window;
  207. }
  208. function createSubWindow2() {
  209. let window = new BrowserWindow({ autoHideMenuBar: true, webPreferences: { nodeIntegration: true }, width: 100, height: 100, show: true });
  210. // window.webContents.openDevTools({ mode: "undocked" });
  211. // if (isDevelopment) {
  212. // window.loadURL(`http://localhost:${process.env.ELECTRON_WEBPACK_WDS_PORT}/serialport.html`);
  213. // } else
  214. // window.loadURL(formatUrl({
  215. // pathname: path.join(__dirname, 'serialport.html'),
  216. // protocol: 'file',
  217. // slashes: true
  218. // }));
  219. let grantedDeviceThroughPermHandler;
  220. window.webContents.session.on('select-usb-device', (event, details, callback) => {
  221. //Add events to handle devices being added or removed before the callback on
  222. //`select-usb-device` is called.
  223. window.webContents.session.on('usb-device-added', (event, device) => {
  224. console.log('usb-device-added FIRED WITH', device)
  225. //Optionally update details.deviceList
  226. })
  227. window.webContents.session.on('usb-device-removed', (event, device) => {
  228. console.log('usb-device-removed FIRED WITH', device)
  229. //Optionally update details.deviceList
  230. })
  231. event.preventDefault()
  232. if (details.deviceList && details.deviceList.length > 0) {
  233. const deviceToReturn = details.deviceList.find((device) => {
  234. // if (!grantedDeviceThroughPermHandler || (device.deviceId != grantedDeviceThroughPermHandler.deviceId)) {
  235. // return true
  236. // }
  237. return true;
  238. })
  239. if (deviceToReturn) {
  240. callback(deviceToReturn.deviceId)
  241. } else {
  242. callback()
  243. }
  244. }
  245. })
  246. window.webContents.session.setPermissionCheckHandler((webContents, permission, requestingOrigin, details) => {
  247. if (permission === 'usb' && details.securityOrigin === 'file:///') {
  248. return true
  249. }
  250. })
  251. window.webContents.session.setDevicePermissionHandler((details) => {
  252. if (details.deviceType === 'usb' && details.origin === 'file://') {
  253. return true;
  254. // if (!grantedDeviceThroughPermHandler) {
  255. // grantedDeviceThroughPermHandler = details.device
  256. // return true
  257. // } else {
  258. // return false
  259. // }
  260. }
  261. })
  262. if (isDevelopment) {
  263. window.loadURL(`http://localhost:${process.env.ELECTRON_WEBPACK_WDS_PORT}`);
  264. } else {
  265. window.loadURL(formatUrl({
  266. pathname: path.join(__dirname, 'cocoblockly-x/index.html'),
  267. protocol: 'file',
  268. slashes: true,
  269. search: "lang=zh-hans",
  270. }));
  271. }
  272. window.on('closed', () => window = null);
  273. window.webContents.on('devtools-opened', () => setImmediate(() => window.focus()));
  274. window.webContents.openDevTools({ mode: 'undocked' });
  275. return window;
  276. }
  277. // ------------ When quit the app ------------ //
  278. app.once('quit', () => {
  279. if (tmp) {
  280. let cmd = platform === 'win32' ? `rd /s /q "${tmp}"` : `rm -rf "${tmp}"`;
  281. exec(cmd);
  282. }
  283. });