main.js

  1. /**
  2. * @file Contains the main.js code of media-dupes
  3. * @author yafp
  4. * @namespace main
  5. */
  6. // console.time('init') // start measuring startup time
  7. // -----------------------------------------------------------------------------
  8. // REQUIRE: 3rd PARTY
  9. // -----------------------------------------------------------------------------
  10. const { app, BrowserWindow, electron, ipcMain, Menu } = require('electron')
  11. const shell = require('electron').shell
  12. const path = require('path')
  13. const fs = require('fs')
  14. const openAboutWindow = require('about-window').default // for: about-window
  15. // -----------------------------------------------------------------------------
  16. // Shared object
  17. // -----------------------------------------------------------------------------
  18. //
  19. var consoleOutput = false // can be changed using --verbose
  20. // power save blocker
  21. var powerSaveBlockerEnabled = false
  22. var powerSaveBlockerId = -1
  23. // Settings UI
  24. var enableVerboseMode = false
  25. var enableAdditionalParameter = false
  26. var additionalYoutubeDlParameter = ''
  27. var enableErrorReporting = true
  28. var downloadDir = app.getPath('downloads') // Detect the default-download-folder of the user from the OS
  29. var audioFormat = 'mp3' // mp3 is the default
  30. var confirmedDisclaimer = false
  31. // Main UI
  32. var applicationState = 'idle' // default is idle
  33. var todoListStateEmpty = true // is empty by default
  34. global.sharedObj = {
  35. // console Output
  36. consoleOutput: consoleOutput,
  37. // power management
  38. powerSaveBlockerEnabled: powerSaveBlockerEnabled,
  39. powerSaveBlockerId: powerSaveBlockerId,
  40. // settings UI
  41. enableErrorReporting: enableErrorReporting,
  42. enableVerboseMode: enableVerboseMode,
  43. enableAdditionalParameter: enableAdditionalParameter,
  44. additionalYoutubeDlParameter: additionalYoutubeDlParameter,
  45. downloadDir: downloadDir,
  46. audioFormat: audioFormat,
  47. confirmedDisclaimer: confirmedDisclaimer,
  48. // main UI
  49. applicationState: applicationState,
  50. todoListStateEmpty: todoListStateEmpty
  51. }
  52. // ----------------------------------------------------------------------------
  53. // REQUIRE: MEDIA-DUPES MODULES
  54. // ----------------------------------------------------------------------------
  55. const crash = require('./app/js/modules/crashReporter.js') // crashReporter
  56. const sentry = require('./app/js/modules/sentry.js') // sentry
  57. const unhandled = require('./app/js/modules/unhandled.js') // electron-unhandled
  58. const utils = require('./app/js/modules/utils.js')
  59. // ----------------------------------------------------------------------------
  60. // COMMAND-LINE-ARGS
  61. // ----------------------------------------------------------------------------
  62. // image to unicode: https://drewish.com/projects/unicoder/
  63. // const appLogo = '\t ▖▄▖▌▌▌▌▄▖▖ \n\t ▗▐▐▐▗▚▚▚▚▚▚▚▀▌▌▖ \n\t ▗▐▐▐▐▐▐▐▐▐▐▐▐▐▐▐▐▐▚ \n\t ▗▐▐▐▚▚▚▀▝▐▐▐▞▝▝▞▞▞▞▞▌▖ \n\t ▌▌▌▌▌▌▘ ▗▚▜▖ ▝▐▐▞▞▞▞ \n\t▗▚▚▚▌▚ ▐▐▐▖▖ ▌▌▌▌▛▖\n\t▚▚▚▘ ▝▜▐▐▚▚▚▘ ▘▚▚▚▚▘\n\t▚▚▙▘ ▚▚▚▚ ▝▞▞▞\n\t▝▞▄ ▘▘ ▌▌▘\n\t ▌▌▙▖ ▗▐▐▐ \n\t ▝▐▗▚▜▐▚▜▐▚▚▜▐▐▐▐▞▌▌▌▌▘ \n\t ▝▐▐▐▐▐▐▐▐▐▐▐▐▚▚▚▚▚▘▘ \n\t ▝▐▐▐▐▐▐▐▐▐▐▐▐▐▝ ▘ \n\t ▘▘▘▘▚▌▘▘▘▘ '
  64. const appLogo = ' ▖▄▖▌▌▌▌▄▖▖ \n ▗▐▐▐▗▚▚▚▚▚▚▚▀▌▌▖ \n ▗▐▐▐▐▐▐▐▐▐▐▐▐▐▐▐▐▐▚ \n ▗▐▐▐▚▚▚▀▝▐▐▐▞▝▝▞▞▞▞▞▌▖ \n ▌▌▌▌▌▌▘ ▗▚▜▖ ▝▐▐▞▞▞▞ \n▗▚▚▚▌▚ ▐▐▐▖▖ ▌▌▌▌▛▖\n▚▚▚▘ ▝▜▐▐▚▚▚▘ ▘▚▚▚▚▘\n▚▚▙▘ ▚▚▚▚ ▝▞▞▞\n▝▞▄ ▘▘ ▌▌▘\n ▌▌▙▖ ▗▐▐▐ \n ▝▐▗▚▜▐▚▜▐▚▚▜▐▐▐▐▞▌▌▌▌▘ \n ▝▐▐▐▐▐▐▐▐▐▐▐▐▚▚▚▚▚▘▘ \n ▝▐▐▐▐▐▐▐▐▐▐▐▐▐▝ ▘ \n ▘▘▘▘▚▌▘▘▘▘ '
  65. var yargs = require('yargs')
  66. .strict(true) // arguments must be valid
  67. .usage('Usage: $0 <command> [options]')
  68. // define arguments
  69. .option('verbose', {
  70. alias: 'v',
  71. type: 'boolean',
  72. description: 'Starts media-dupes with verbose output'
  73. })
  74. .option('help', {
  75. alias: 'h',
  76. type: 'boolean',
  77. description: 'Shows the media-dupes help'
  78. })
  79. .option('version', {
  80. type: 'boolean',
  81. description: 'Shows the media-dupes version'
  82. })
  83. // displayed in case of invalid parameters
  84. .showHelpOnFail(false, 'Specify --help to display the available options')
  85. // show the project url
  86. .epilog(appLogo)
  87. .epilog('Project URL: https://github.com/yafp/media-dupes')
  88. // Show an example
  89. //
  90. // .example('$0 --help', 'Shows the media-dupes help')
  91. .argv
  92. if (yargs.verbose === true) {
  93. global.sharedObj.consoleOutput = true // update the global object
  94. }
  95. // ----------------------------------------------------------------------------
  96. // ERROR-HANDLING:
  97. // ----------------------------------------------------------------------------
  98. crash.initCrashReporter()
  99. unhandled.initUnhandled()
  100. sentry.enableSentry() // sentry is enabled by default
  101. // ----------------------------------------------------------------------------
  102. // VARIABLES & CONSTANTS
  103. // ----------------------------------------------------------------------------
  104. // Keep a global reference of the window object, if you don't, the window will
  105. // be closed automatically when the JavaScript object is garbage collected.
  106. let mainWindow
  107. let settingsWindow
  108. let distractionWindow
  109. const gotTheLock = app.requestSingleInstanceLock() // for: single-instance handling
  110. const defaultUserDataPath = app.getPath('userData') // for: storing window position and size
  111. const { urlGitHubGeneral, urlGitHubIssues, urlGitHubChangelog, urlGitHubReleases, urlYoutubeDlSupportedSites } = require('./app/js/modules/urls.js') // project-urls
  112. // Caution: Warning since electron 8
  113. app.allowRendererProcessReuse = true // see: https://github.com/electron/electron/issues/18397. Set to true with media-dupes 0.9.0
  114. // mainWindow: minimal window size
  115. const mainWindowMinimalWindowHeight = 730
  116. const mainWindowMinimalWindowWidth = 600
  117. // settingsWundow: minimal window size
  118. const settingsWindowMinimalWindowHeight = 400
  119. const settingsWindowMinimalWindowWidth = 800
  120. // ----------------------------------------------------------------------------
  121. // FUNCTIONS
  122. // ----------------------------------------------------------------------------
  123. /**
  124. * @function doLog
  125. * @summary Writes console output for the main process
  126. * @description Writes console output for the main process
  127. * @memberof main
  128. * @param {string} type - The log type
  129. * @param {string} message - The log message
  130. */
  131. function doLog (type, message) {
  132. const logM = require('electron-log')
  133. const prefix = '[ Main ] '
  134. if (global.sharedObj.consoleOutput === false) {
  135. logM.transports.console.level = false // disable Terminal output. Still logs to DevTools and LogFile
  136. }
  137. // important: https://github.com/megahertz/electron-log/issues/189
  138. // electron-log can: error, warn, info, verbose, debug, silly
  139. switch (type) {
  140. case 'info':
  141. logM.info(prefix + message)
  142. break
  143. case 'warn':
  144. logM.warn(prefix + message)
  145. break
  146. case 'error':
  147. logM.error(prefix + message)
  148. break
  149. default:
  150. logM.silly(prefix + message)
  151. // code block
  152. }
  153. }
  154. /**
  155. * @function createWindowSettings
  156. * @summary Manages the BrowserWindow for the Settings UI
  157. * @description Manages the BrowserWindow for the Settings UI
  158. * @memberof main
  159. */
  160. function createWindowSettings () {
  161. doLog('info', 'createWindowSettings ::: Creating the settings window')
  162. // Create the browser window.for Settings
  163. settingsWindow = new BrowserWindow({
  164. modal: true,
  165. frame: true, // false results in a borderless window. Needed for custom titlebar
  166. titleBarStyle: 'default', // needed for custom-electron-titlebar. See: https://electronjs.org/docs/api/frameless-window
  167. backgroundColor: '#ffffff', // since 0.3.0
  168. show: true, // hide until: ready-to-show
  169. center: true, // Show window in the center of the screen. (since 0.3.0)
  170. width: settingsWindowMinimalWindowWidth,
  171. minWidth: settingsWindowMinimalWindowWidth,
  172. minimizable: false, // not implemented on linux
  173. maximizable: false, // not implemented on linux
  174. height: settingsWindowMinimalWindowHeight,
  175. minHeight: settingsWindowMinimalWindowHeight,
  176. icon: path.join(__dirname, 'app/img/icon/icon.png'),
  177. webPreferences: {
  178. nodeIntegration: true,
  179. webSecurity: true // introduced in 0.3.0
  180. }
  181. })
  182. settingsWindow.loadFile('app/settings.html') // load the setting.html to the settings-window
  183. settingsWindow.removeMenu() // the settings window needs no menu
  184. // Call from renderer: Settings UI - toggle dev tools
  185. ipcMain.on('settingsToggleDevTools', function () {
  186. settingsWindow.webContents.toggleDevTools()
  187. })
  188. // Emitted before the window is closed.
  189. settingsWindow.on('close', function () {
  190. doLog('info', 'createWindowSettings ::: settingsWindow will close (event: close)')
  191. })
  192. // Emitted when the window is closed.
  193. settingsWindow.on('closed', function (event) {
  194. doLog('info', 'createWindowSettings ::: settingsWindow is closed (event: closed)')
  195. settingsWindow = null // Dereference the window object
  196. mainWindow.webContents.send('unblurMainUI') // unblur the main UI
  197. })
  198. }
  199. /**
  200. * @function createWindowDistraction
  201. * @summary creates the window for distraction mode
  202. * @description creates the window for distraction mode
  203. * @memberof main
  204. */
  205. function createWindowDistraction () {
  206. doLog('info', 'createWindowDistraction ::: Creating the distraction window')
  207. // Create the distraction browser window.
  208. distractionWindow = new BrowserWindow({
  209. frame: true, // false results in a borderless window. Needed for custom titlebar
  210. backgroundColor: '#ffffff',
  211. center: true,
  212. width: 550,
  213. minWidth: 550,
  214. resizable: false,
  215. height: 720,
  216. minHeight: 720,
  217. icon: path.join(__dirname, 'app/img/icon/icon.png'),
  218. webPreferences: {
  219. webSecurity: true
  220. }
  221. })
  222. distractionWindow.loadFile('app/distraction.html') // load the distraction.html to the
  223. distractionWindow.removeMenu() // the distraction-window needs no menu
  224. // Emitted before the window is closed.
  225. distractionWindow.on('close', function () {
  226. doLog('info', 'createWindowDistraction ::: distractionWindow will close (event: close)')
  227. })
  228. // Emitted when the window is closed.
  229. distractionWindow.on('closed', function (event) {
  230. doLog('info', 'createWindowDistraction ::: distractionWindow is closed (event: closed)')
  231. distractionWindow = null // Dereference the window object
  232. })
  233. }
  234. /**
  235. * @function createWindowMain
  236. * @summary Creates the mainWindow
  237. * @description Creates the mainWindow (restores window position and size of possible)
  238. * @memberof main
  239. */
  240. function createWindowMain () {
  241. doLog('info', 'createWindowMain ::: Starting to create the application windows')
  242. // Check last window position and size from user data
  243. var windowWidth
  244. var windowHeight
  245. var windowPositionX
  246. var windowPositionY
  247. // Read a local config file
  248. var customUserDataPath = path.join(defaultUserDataPath, 'MediaDupesWindowPosSize.json')
  249. var data
  250. try {
  251. data = JSON.parse(fs.readFileSync(customUserDataPath, 'utf8'))
  252. windowWidth = data.bounds.width // window size: width
  253. windowHeight = data.bounds.height // window size: height
  254. windowPositionX = data.bounds.x // window position: x
  255. windowPositionY = data.bounds.y // window position: y
  256. doLog('info', 'createWindowMain ::: Got last window position and size information from _' + customUserDataPath + '_.')
  257. } catch (e) {
  258. doLog('warn', 'createWindowMain ::: No last window position and size information found in _' + customUserDataPath + '_. Using fallback values')
  259. // set some default values for window size
  260. windowWidth = mainWindowMinimalWindowWidth
  261. windowHeight = mainWindowMinimalWindowHeight
  262. }
  263. // Create the browser window.
  264. mainWindow = new BrowserWindow({
  265. frame: false, // false results in a borderless window. Needed for custom titlebar
  266. titleBarStyle: 'hidden', // needed for custom-electron-titlebar. See: https://electronjs.org/docs/api/frameless-window
  267. backgroundColor: '#ffffff', // since 0.3.0
  268. show: false, // hide until: ready-to-show event is fired
  269. center: true, // Show window in the center of the screen. (since 0.3.0)
  270. width: windowWidth,
  271. minWidth: mainWindowMinimalWindowWidth,
  272. height: windowHeight,
  273. minHeight: mainWindowMinimalWindowHeight,
  274. icon: path.join(__dirname, 'app/img/icon/icon.png'),
  275. webPreferences: {
  276. enableRemoteModule: true, // since electron 9.0.0
  277. nodeIntegration: true,
  278. webSecurity: true // introduced in 0.3.0
  279. }
  280. })
  281. // Restore window position if possible
  282. //
  283. // requirements: found values in MediaDupesWindowPosSize.json from the previous session
  284. if ((typeof windowPositionX !== 'undefined') && (typeof windowPositionY !== 'undefined')) {
  285. mainWindow.setPosition(windowPositionX, windowPositionY)
  286. }
  287. // Call from renderer: reload the application
  288. ipcMain.on('reloadMainWindow', (event) => {
  289. doLog('info', 'createWindowMain ::: Trying to reload the main window.')
  290. mainWindow.reload()
  291. })
  292. // Call from renderer: Open download folder
  293. ipcMain.on('openUserDownloadFolder', (event, userSettingValue) => {
  294. doLog('info', 'ipc.openUserDownloadFolder ::: Trying to open the user download folder _' + userSettingValue + '_.')
  295. // try to open it
  296. shell.openPath(userSettingValue) // FIXME: check result
  297. /*
  298. var openUserDownloadFolder = shell.openPath(userSettingValue)
  299. if (Object.keys(openUserDownloadFolder).length === 0) {
  300. doLog('info', 'ipc.openUserDownloadFolder ::: Opened the media-dupes download folder (ipcMain)')
  301. } else {
  302. doLog('error', 'ipc.openUserDownloadFolder ::: Failed to open the user download folder (ipcMain). Error: ' + openUserDownloadFolder)
  303. }
  304. */
  305. })
  306. // Call from renderer: Open settings folder
  307. ipcMain.on('settingsFolderOpen', (event) => {
  308. doLog('info', 'ipc.settingsFolderOpen ::: Trying to open the users settings folder (ipcMain)')
  309. const userSettingsPath = path.join(app.getPath('userData'), 'UserSettings') // change path for userSettings
  310. // try to open it
  311. shell.openPath(userSettingsPath) // FIXME: check result
  312. /*
  313. if (shell.openPath(userSettingsPath) === true) {
  314. doLog('info', 'ipc.settingsFolderOpen ::: Opened the media-dupes settings folder (ipcMain)')
  315. } else {
  316. doLog('error', 'ipc.settingsFolderOpen ::: Failed to open the user settings folder (ipcMain)')
  317. }
  318. */
  319. })
  320. // Call from renderer: Urgent window
  321. ipcMain.on('makeWindowUrgent', function () {
  322. mainWindow.flashFrame(true)
  323. })
  324. // Call from renderer: Option: load settings UI
  325. ipcMain.on('settingsUiLoad', function () {
  326. createWindowSettings()
  327. })
  328. // Call from renderer: Option: load distraction UI
  329. ipcMain.on('startDistraction', function () {
  330. createWindowDistraction()
  331. })
  332. // Call from renderer: show mainUI
  333. ipcMain.on('showAndFocusMainUI', function () {
  334. mainWindow.show()
  335. mainWindow.focus()
  336. })
  337. // Call from renderer: Update property from globalObj
  338. ipcMain.on('globalObjectSet', function (event, property, value) {
  339. doLog('info', 'ipc.globalObjectSet ::: Set _' + property + '_ to: _' + value + '_')
  340. global.sharedObj[property] = value // update the property in the global shared object
  341. if (global.sharedObj.consoleOutput === true) { // If console output is enabled- show the entire sharedObject
  342. console.warn(global.sharedObj)
  343. }
  344. })
  345. // Call from renderer - Enable the power save blocker. See #97
  346. ipcMain.on('enablePowerSaveBlocker', function () {
  347. const { powerSaveBlocker } = require('electron')
  348. const id = powerSaveBlocker.start('prevent-display-sleep')
  349. if (powerSaveBlocker.isStarted(id)) {
  350. doLog('info', 'ipc.enablePowerSaveBlocker ::: Successfully enabled the PowerSaveBlocker with the ID _' + id + '_ as app is currently downloading')
  351. global.sharedObj.powerSaveBlockerEnabled = true
  352. global.sharedObj.powerSaveBlockerId = id
  353. } else {
  354. doLog('error', 'ipc.enablePowerSaveBlocker ::: Enabling the Power-Save-Blocker for the current download failed')
  355. global.sharedObj.powerSaveBlockerEnabled = false
  356. global.sharedObj.powerSaveBlockerId = -1
  357. }
  358. })
  359. // Call from renderer: disables the powersaveblocker
  360. ipcMain.on('disablePowerSaveBlocker', function (event, id) {
  361. const { powerSaveBlocker } = require('electron')
  362. powerSaveBlocker.stop(id)
  363. global.sharedObj.powerSaveBlockerEnabled = false
  364. global.sharedObj.powerSaveBlockerId = -1
  365. doLog('info', 'ipc.disablePowerSaveBlocker ::: Disabled the PowerSaveBlocker with the ID: _' + id + '_.')
  366. })
  367. mainWindow.loadFile('app/index.html') // load the index.html to the main window
  368. // Emitted when the web page becomes unresponsive.
  369. mainWindow.on('unresponsive', function () {
  370. doLog('warn', 'createWindowMain (on unresponsive) ::: mainWindow is now unresponsive (event: unresponsive)')
  371. })
  372. // Emitted when the unresponsive web page becomes responsive again.
  373. mainWindow.on('responsive', function () {
  374. doLog('info', 'createWindowMain (on responsive) ::: mainWindow is now responsive again (event: responsive)')
  375. })
  376. // Emitted when the web page has been rendered (while not being shown) and window can be displayed without a visual flash.
  377. mainWindow.on('ready-to-show', function () {
  378. doLog('info', 'createWindowMain (ready to show) ::: mainWindow is now ready, so show it and then focus it (event: ready-to-show)')
  379. mainWindow.show()
  380. mainWindow.focus()
  381. // do some checks & routines once at start of the application
  382. mainWindow.webContents.send('startCheckingDependencies') // check application dependencies
  383. mainWindow.webContents.send('todoListCheck') // search if there are urls to restore
  384. // mainWindow.webContents.send('unblurMainUI') // unblur the main UI
  385. // Start checks for updates scheduled to improve startup time
  386. mainWindow.webContents.send('scheduleUpdateCheckMediaDupes')
  387. mainWindow.webContents.send('scheduleUpdateCheckYoutubeDl')
  388. mainWindow.webContents.send('countAppStarts') // count app starts
  389. })
  390. // end: ready-toshow
  391. // Emitted when the web page has been rendered (while not being shown) and window can be displayed without a visual flash.
  392. mainWindow.on('show', function () {
  393. doLog('info', 'createWindowMain (show) ::: mainWindow is now ready, so show it and then focus it (event: ready-to-show)')
  394. mainWindow.webContents.send('blurMainUI') // blur the main UI
  395. mainWindow.webContents.send('startDisclaimerCheck') // check if disclaimer must be shown
  396. // console.timeEnd('init') // Stop measuring startup time
  397. })
  398. // end: ready-toshow
  399. // Emitted before the window is closed.
  400. mainWindow.on('close', function (event) {
  401. doLog('info', 'createWindowMain (on close) ::: mainWindow will close (event: close)')
  402. var curState = global.sharedObj.applicationState // get applicationState
  403. doLog('info', 'createWindowMain (on close) ::: Current application state is: _' + curState + '_.')
  404. if (curState === 'Download in progress') {
  405. var choiceA = require('electron').dialog.showMessageBoxSync(this, // since electron7 showMessageBox no longer blocks the close. Therefor we are using showMessageBoxSync
  406. {
  407. icon: path.join(__dirname, 'app/img/icon/icon.png'),
  408. type: 'question',
  409. buttons: ['Yes', 'No'],
  410. title: 'Downloads in progress',
  411. message: 'media-dupes is currently downloading.\n\nDo you really want to quit?'
  412. })
  413. if (choiceA === 1) {
  414. event.preventDefault() // user pressed No
  415. return
  416. }
  417. }
  418. // todoList handling - see #66
  419. //
  420. var curTodoListStateEmpty = global.sharedObj.todoListStateEmpty
  421. doLog('info', 'createWindowMain (on close) ::: Is the todo list currently empty?: _' + curTodoListStateEmpty + '_.')
  422. if (curTodoListStateEmpty === false) {
  423. // todo List contains data which should be handled
  424. var choiceB = require('electron').dialog.showMessageBoxSync(this,
  425. {
  426. icon: path.join(__dirname, 'app/img/icon/icon.png'),
  427. type: 'question',
  428. buttons: ['Yes', 'No'],
  429. title: 'Save current todo list?',
  430. message: 'Your todo list contains unprocessed URLs.\n\nDo you want to restore them on next launch?'
  431. })
  432. if (choiceB === 0) {
  433. doLog('info', 'createWindowMain (on close) ::: User wants to save his todo list')
  434. mainWindow.webContents.send('todoListTryToSave')
  435. } else {
  436. doLog('info', 'createWindowMain (on close) ::: User does NOT want to save his todo list')
  437. }
  438. } else {
  439. doLog('info', 'createWindowMain ::: There is nothing in the todo list to save')
  440. }
  441. // get window position and size
  442. var data = {
  443. bounds: mainWindow.getBounds()
  444. }
  445. // define target path (in user data)
  446. var customUserDataPath = path.join(defaultUserDataPath, 'MediaDupesWindowPosSize.json')
  447. // try to write
  448. fs.writeFile(customUserDataPath, JSON.stringify(data), function (error) {
  449. if (error) {
  450. doLog('error', 'createWindowMain ::: storing window-position and -size of mainWindow in _' + customUserDataPath + '_ failed with error: _' + error + '_ (event: close)')
  451. throw error
  452. }
  453. doLog('info', 'createWindowMain ::: mainWindow stored window-position and -size in _' + customUserDataPath + '_ (event: close)')
  454. })
  455. })
  456. // end: close
  457. // Emitted when the window is closed.
  458. mainWindow.on('closed', function (event) {
  459. doLog('info', 'createWindowMain (closed) ::: mainWindow is closed (event: closed)')
  460. mainWindow = null // Dereference the window object,
  461. })
  462. // end: closed
  463. }
  464. /**
  465. * @function createMenuMain
  466. * @summary Creates the menu for the main UI
  467. * @description Creates the menu for the main UI
  468. * @memberof main
  469. */
  470. function createMenuMain () {
  471. var menu = Menu.buildFromTemplate([
  472. // Menu: File
  473. {
  474. label: 'File',
  475. submenu: [
  476. // Settings
  477. {
  478. label: 'Settings',
  479. icon: path.join(__dirname, 'app/img/menu/file/cog_red.png'),
  480. click () {
  481. mainWindow.webContents.send('openSettings')
  482. },
  483. accelerator: 'CmdOrCtrl+,'
  484. },
  485. {
  486. type: 'separator'
  487. },
  488. // Exit
  489. {
  490. role: 'quit',
  491. label: 'Exit',
  492. icon: path.join(__dirname, 'app/img/menu/file/power-off_red.png'),
  493. click () {
  494. app.quit()
  495. },
  496. accelerator: 'CmdOrCtrl+Q'
  497. }
  498. ]
  499. },
  500. // Menu: Edit
  501. {
  502. label: 'Edit',
  503. submenu: [
  504. {
  505. label: 'Undo',
  506. icon: path.join(__dirname, 'app/img/menu/edit/undo_red.png'),
  507. accelerator: 'CmdOrCtrl+Z',
  508. selector: 'undo:'
  509. },
  510. {
  511. label: 'Redo',
  512. icon: path.join(__dirname, 'app/img/menu/edit/redo_red.png'),
  513. accelerator: 'Shift+CmdOrCtrl+Z',
  514. selector: 'redo:'
  515. },
  516. {
  517. type: 'separator'
  518. },
  519. {
  520. label: 'Cut',
  521. icon: path.join(__dirname, 'app/img/menu/edit/cut_red.png'),
  522. accelerator: 'CmdOrCtrl+X',
  523. selector: 'cut:'
  524. },
  525. {
  526. label: 'Copy',
  527. icon: path.join(__dirname, 'app/img/menu/edit/copy_red.png'),
  528. accelerator: 'CmdOrCtrl+C',
  529. selector: 'copy:'
  530. },
  531. {
  532. label: 'Paste',
  533. icon: path.join(__dirname, 'app/img/menu/edit/paste_red.png'),
  534. accelerator: 'CmdOrCtrl+V',
  535. selector: 'paste:'
  536. },
  537. {
  538. label: 'Select All',
  539. accelerator: 'CmdOrCtrl+A',
  540. selector: 'selectAll:'
  541. }
  542. ]
  543. },
  544. // Menu: View
  545. /*
  546. {
  547. label: 'View',
  548. submenu: [
  549. {
  550. role: 'reload',
  551. label: 'Reload',
  552. click (item, mainWindow) {
  553. mainWindow.reload()
  554. },
  555. accelerator: 'CmdOrCtrl+R'
  556. }
  557. ]
  558. },
  559. */
  560. // Menu: Youtube
  561. /*
  562. {
  563. label: 'Search',
  564. submenu: [
  565. {
  566. label: 'Youtube Suggest',
  567. click (item, mainWindow) {
  568. mainWindow.webContents.send('openYoutubeSuggestDialog')
  569. },
  570. accelerator: 'CmdOrCtrl+S'
  571. }
  572. ]
  573. },
  574. */
  575. // Menu: Window
  576. {
  577. label: 'Window',
  578. submenu: [
  579. {
  580. role: 'reload',
  581. label: 'Reload',
  582. click (item, mainWindow) {
  583. mainWindow.reload()
  584. },
  585. accelerator: 'CmdOrCtrl+R'
  586. },
  587. {
  588. role: 'togglefullscreen',
  589. label: 'Toggle Fullscreen',
  590. click (item, mainWindow) {
  591. if (mainWindow.isFullScreen()) {
  592. mainWindow.setFullScreen(false)
  593. } else {
  594. mainWindow.setFullScreen(true)
  595. }
  596. },
  597. accelerator: 'F11' // is most likely predefined on osx - results in: doesnt work on osx
  598. },
  599. {
  600. role: 'minimize',
  601. label: 'Minimize',
  602. icon: path.join(__dirname, 'app/img/menu/window/window-minimize_red.png'),
  603. click (item, mainWindow) {
  604. if (mainWindow.isMinimized()) {
  605. // mainWindow.restore();
  606. } else {
  607. mainWindow.minimize()
  608. }
  609. },
  610. accelerator: 'CmdOrCtrl+M'
  611. },
  612. {
  613. label: 'Maximize',
  614. icon: path.join(__dirname, 'app/img/menu/window/window-maximize_red.png'),
  615. click (item, mainWindow) {
  616. if (mainWindow.isMaximized()) {
  617. mainWindow.unmaximize()
  618. } else {
  619. mainWindow.maximize()
  620. }
  621. },
  622. accelerator: 'CmdOrCtrl+K'
  623. }
  624. ]
  625. },
  626. // Menu: Help
  627. {
  628. role: 'help',
  629. label: 'Help',
  630. submenu: [
  631. // About
  632. {
  633. role: 'about',
  634. label: 'About',
  635. icon: path.join(__dirname, 'app/img/menu/help/info-circle_red.png'),
  636. click () {
  637. openAboutWindow({
  638. icon_path: path.join(__dirname, 'app/img/about/icon_about.png'),
  639. open_devtools: false,
  640. use_version_info: true,
  641. win_options: // https://github.com/electron/electron/blob/master/docs/api/browser-window.md#new-browserwindowoptions
  642. {
  643. autoHideMenuBar: true,
  644. titleBarStyle: 'hidden',
  645. minimizable: false, // not implemented on linux
  646. maximizable: false, // not implemented on linux
  647. movable: false, // not implemented on linux
  648. resizable: false,
  649. alwaysOnTop: true,
  650. fullscreenable: false,
  651. skipTaskbar: false
  652. }
  653. })
  654. }
  655. },
  656. // open homepage
  657. {
  658. label: 'Homepage',
  659. icon: path.join(__dirname, 'app/img/menu/help/github_red.png'),
  660. click () {
  661. shell.openExternal(urlGitHubGeneral)
  662. },
  663. accelerator: 'F1'
  664. },
  665. // report issue
  666. {
  667. label: 'Report issue',
  668. icon: path.join(__dirname, 'app/img/menu/help/github_red.png'),
  669. click () {
  670. shell.openExternal(urlGitHubIssues)
  671. },
  672. accelerator: 'F2'
  673. },
  674. // open changelog
  675. {
  676. label: 'Changelog',
  677. icon: path.join(__dirname, 'app/img/menu/help/github_red.png'),
  678. click () {
  679. shell.openExternal(urlGitHubChangelog)
  680. },
  681. accelerator: 'F3'
  682. },
  683. // open Releases
  684. {
  685. label: 'Releases',
  686. icon: path.join(__dirname, 'app/img/menu/help/github_red.png'),
  687. click () {
  688. shell.openExternal(urlGitHubReleases)
  689. },
  690. accelerator: 'F4'
  691. },
  692. {
  693. type: 'separator'
  694. },
  695. // Youtube-suggest
  696. {
  697. label: 'Youtube Suggest',
  698. click (item, mainWindow) {
  699. mainWindow.webContents.send('openYoutubeSuggestDialog')
  700. },
  701. accelerator: 'CmdOrCtrl+S'
  702. },
  703. {
  704. type: 'separator'
  705. },
  706. // Update
  707. {
  708. label: 'Search media-dupes updates',
  709. icon: path.join(__dirname, 'app/img/menu/help/cloud-download-alt_red.png'),
  710. click (item, mainWindow) {
  711. mainWindow.webContents.send('startSearchUpdatesVerbose')
  712. },
  713. enabled: true,
  714. accelerator: 'F9'
  715. },
  716. {
  717. type: 'separator'
  718. },
  719. // Console
  720. {
  721. id: 'HelpConsole',
  722. label: 'Console',
  723. icon: path.join(__dirname, 'app/img/menu/help/terminal_red.png'),
  724. click (item, mainWindow) {
  725. mainWindow.webContents.toggleDevTools()
  726. },
  727. enabled: true,
  728. accelerator: 'F12'
  729. },
  730. {
  731. type: 'separator'
  732. },
  733. // SubMenu youtube-dl maintenance of help
  734. {
  735. label: 'Youtube-DL',
  736. icon: path.join(__dirname, 'app/img/menu/help/youtube_red.png'),
  737. submenu: [
  738. // Show supported sites
  739. {
  740. id: 'youtubeDlShowSupportedSites',
  741. label: 'Show list of supported sites',
  742. click () {
  743. shell.openExternal(urlYoutubeDlSupportedSites)
  744. },
  745. enabled: true
  746. },
  747. // Clear cache in userData
  748. {
  749. id: 'youtubeDlBinaryPathReset',
  750. label: 'Reset youtube-dl binary path',
  751. click (item, mainWindow) {
  752. mainWindow.webContents.send('youtubeDlBinaryPathReset')
  753. },
  754. enabled: true
  755. },
  756. // Force update (ignoring if there is an update available or not)
  757. {
  758. id: 'youtubeDlBinaryUpdateForce',
  759. label: 'Force updating youtube-dl binary',
  760. click (item, mainWindow) {
  761. mainWindow.webContents.send('youtubeDlBinaryUpdate')
  762. },
  763. enabled: true
  764. }
  765. ]
  766. }
  767. ]
  768. }
  769. ])
  770. // use the menu
  771. Menu.setApplicationMenu(menu)
  772. }
  773. /**
  774. * @function forceSingleAppInstance
  775. * @summary Takes care that there is only 1 instance of this app running
  776. * @description Takes care that there is only 1 instance of this app running
  777. * @memberof main
  778. */
  779. function forceSingleAppInstance () {
  780. if (!gotTheLock) {
  781. doLog('warn', 'forceSingleAppInstance ::: There is already another instance of media-dupes')
  782. app.quit() // quit the second instance
  783. } else {
  784. app.on('second-instance', (event, commandLine, workingDirectory) => {
  785. // Someone tried to run a second instance, we should focus our first instance window.
  786. // if (mainWindow) {
  787. if (mainWindow === null) {
  788. // do nothing - there is no mainwindow - most likely we are on macOS
  789. } else {
  790. // mainWindow exists
  791. if (mainWindow.isMinimized()) {
  792. mainWindow.restore()
  793. }
  794. mainWindow.focus()
  795. }
  796. // }
  797. })
  798. }
  799. }
  800. /**
  801. * @function powerMonitorInit
  802. * @summary Initialized a powermonitor after the app is ready
  803. * @description Initialized a powermonitor after the app is ready. See: https://electronjs.org/docs/api/power-monitor
  804. * @memberof main
  805. */
  806. function powerMonitorInit () {
  807. const { powerMonitor } = require('electron') // This module cannot be used until the ready event of the app module is emitted.
  808. // suspend
  809. powerMonitor.on('suspend', () => {
  810. doLog('warn', 'powerMonitorInit ::: The system is going to sleep (event: suspend)')
  811. mainWindow.webContents.send('todoListTryToSave') // try to save the todolist - see #79
  812. powerMonitorNotify('warning', 'The system is going to sleep (event: suspend)', 0)
  813. })
  814. // resume
  815. powerMonitor.on('resume', () => {
  816. doLog('info', 'powerMonitorInit ::: The system is resuming (event: resume)')
  817. mainWindow.webContents.send('todoListCheck') // search if there are urls to restore
  818. powerMonitorNotify('info', 'The system resumed (event: resume)', 0)
  819. })
  820. // shutdown (Linux, macOS)
  821. powerMonitor.on('shutdown', () => {
  822. doLog('info', 'powerMonitorInit ::: The system is going to shutdown (event: shutdown)')
  823. })
  824. // OTHER SUPPORTED EVENTS:
  825. //
  826. // on-ac (Windows)
  827. // on-battery (Windows)
  828. // lock-screen (macOs, Windows)
  829. // unlock-screen (macOs, Windows)
  830. }
  831. /**
  832. * @function powerMonitorNotify
  833. * @summary Used to tell the renderer to display a notification
  834. * @description Used to tell the renderer to display a notification
  835. * @memberof main
  836. */
  837. function powerMonitorNotify (messageType, messageText, messageDuration) {
  838. doLog('warn', 'powerMonitorNotify ::: Going to tell the renderer to show a powerMonitor notification')
  839. mainWindow.webContents.send('powerMonitorNotification', messageType, messageText, messageDuration)
  840. }
  841. // This method will be called when Electron has finished
  842. // initialization and is ready to create browser windows.
  843. // Some APIs can only be used after this event occurs.
  844. //
  845. // app.on('ready', createWindow)
  846. app.on('ready', function () {
  847. forceSingleAppInstance() // check for single instance
  848. createWindowMain() // create the application UI
  849. createMenuMain() // create the application menu
  850. powerMonitorInit() //
  851. })
  852. // Quit when all windows are closed.
  853. app.on('window-all-closed', function () {
  854. // On macOS it is common for applications and their menu bar
  855. // to stay active until the user quits explicitly with Cmd + Q
  856. if (process.platform !== 'darwin') {
  857. app.quit()
  858. }
  859. })
  860. app.on('activate', function () {
  861. // On macOS it's common to re-create a window in the app when the
  862. // dock icon is clicked and there are no other windows open.
  863. if (mainWindow === null) {
  864. createWindowMain()
  865. }
  866. })