app/js/renderer.js

  1. /**
  2. * @file Contains all renderer code
  3. * @author yafp
  4. * @namespace renderer
  5. */
  6. 'use strict'
  7. // ----------------------------------------------------------------------------
  8. // IMPORT MODULES
  9. // ----------------------------------------------------------------------------
  10. require('v8-compile-cache')
  11. // ----------------------------------------------------------------------------
  12. // IMPORT MEDIA-DUPES MODULES
  13. // ----------------------------------------------------------------------------
  14. const utils = require('./js/modules/utils.js')
  15. const ffmpeg = require('./js/modules/ffmpeg.js')
  16. const ui = require('./js/modules/ui.js')
  17. const settings = require('./js/modules/settings.js')
  18. const youtubeDl = require('./js/modules/youtubeDl.js')
  19. const sentry = require('./js/modules/sentry.js')
  20. const crash = require('./js/modules/crashReporter.js') // crashReporter
  21. const unhandled = require('./js/modules/unhandled.js') // electron-unhandled
  22. // ----------------------------------------------------------------------------
  23. // ERROR HANDLING
  24. // ----------------------------------------------------------------------------
  25. // crash.initCrashReporter() // since electron9: crashReporter.start is deprecated in the renderer process. Call it from the main process instead.
  26. unhandled.initUnhandled()
  27. // ----------------------------------------------------------------------------
  28. // VARIABLES
  29. // ----------------------------------------------------------------------------
  30. // Settings variables
  31. //
  32. var ytdlBinaryVersion = '0.0.0'
  33. var youtubeDlBinaryDetailsVersion
  34. var youtubeDlBinaryDetailsPath
  35. var youtubeDLBinaryDetailsExec
  36. // ----------------------------------------------------------------------------
  37. // FUNCTIONS - MAIN WINDOW CLICKS
  38. // ----------------------------------------------------------------------------
  39. /**
  40. * @function windowMainClickDistract
  41. * @summary Handles the click on the app icon
  42. * @description Triggered from the mainWindow. Starts the easteregg / distract function
  43. * @memberof renderer
  44. */
  45. function windowMainClickDistract () {
  46. ui.windowMainDistract()
  47. }
  48. /**
  49. * @function windowMainClickButtonAddUrl
  50. * @summary Handles the click on the AddUrl button
  51. * @description Triggered from the mainWindow. Starts the add url function from the module ui
  52. * @memberof renderer
  53. */
  54. function windowMainClickButtonAddUrl () {
  55. ui.windowMainAddUrl()
  56. }
  57. /**
  58. * @function windowMainClickButtonVideo
  59. * @summary Handles the click on the video button
  60. * @description Triggered from the mainWindow. Starts the video download function from the module ui
  61. * @memberof renderer
  62. */
  63. function windowMainClickButtonVideo () {
  64. ui.windowMainDownloadContent('video')
  65. sentry.countEvent('usageButtonVideoExec')
  66. }
  67. /**
  68. * @function windowMainClickButtonVideoV2
  69. * @summary Handles the click on the video button
  70. * @description Triggered from the mainWindow. Starts the video download function from the module ui
  71. * @memberof renderer
  72. */
  73. function windowMainClickButtonVideoV2 () {
  74. ui.windowMainDownloadVideo()
  75. }
  76. /**
  77. * @function windowMainClickButtonAudio
  78. * @summary Handles the click on the audio button
  79. * @description Triggered from the mainWindow. Starts the audio download function from the module ui
  80. * @memberof renderer
  81. */
  82. function windowMainClickButtonAudio () {
  83. ui.windowMainDownloadContent('audio')
  84. sentry.countEvent('usageButtonAudioExec')
  85. }
  86. /**
  87. * @function windowMainClickButtonSettings
  88. * @summary Handles the click on the settings button
  89. * @description Triggered from the mainWindow. Starts the settings UI from the module ui
  90. * @memberof renderer
  91. */
  92. function windowMainClickButtonSettings () {
  93. ui.windowMainSettingsUiLoad()
  94. }
  95. /**
  96. * @function windowMainClickButtonIntro
  97. * @summary Handles the click on the intro button
  98. * @description Triggered from the mainWindow. Starts the application intro from the module ui
  99. * @memberof renderer
  100. */
  101. function windowMainClickButtonIntro () {
  102. ui.windowMainIntroShow()
  103. }
  104. /**
  105. * @function windowMainClickButtonDownloads
  106. * @summary Handles the click on the Downloads button
  107. * @description Triggered from the mainWindow. Starts the open-download-folder function from the module ui
  108. * @memberof renderer
  109. */
  110. function windowMainClickButtonDownloads () {
  111. ui.windowMainOpenDownloadFolder()
  112. }
  113. /**
  114. * @function windowMainClickButtonLogReset
  115. * @summary Handles the click on the log-reset button
  116. * @description Triggered from the mainWindow. Starts the reset-log function from the module ui
  117. * @memberof renderer
  118. */
  119. function windowMainClickButtonLogReset () {
  120. ui.windowMainLogReset()
  121. }
  122. /**
  123. * @function windowMainClickButtonUIReset
  124. * @summary Handles the click on the reset-UI button
  125. * @description Triggered from the mainWindow. Starts the reset-UI function from the module ui
  126. * @memberof renderer
  127. */
  128. function windowMainClickButtonUIReset () {
  129. ui.windowMainResetAskUser()
  130. }
  131. // ----------------------------------------------------------------------------
  132. // FUNCTIONS - SETTINGS WINDOW CLICKS
  133. // ----------------------------------------------------------------------------
  134. /**
  135. * @function windowSettingsClickIconUserSettingsDir
  136. * @summary Handles the click on the settings icon
  137. * @description Triggered from the settingsWindow. Starts the open-settings-folder function from the module settings
  138. * @memberof renderer
  139. */
  140. function windowSettingsClickIconUserSettingsDir () {
  141. settings.settingsFolderOpen()
  142. }
  143. /**
  144. * @function windowSettingsClickButtonChooseDownloadDir
  145. * @summary Handles the click on the choose download dir button. Starts the select-download-dir function from the module settings
  146. * @description Triggered from the settingsWindow.
  147. * @memberof renderer
  148. */
  149. function windowSettingsClickButtonChooseDownloadDir () {
  150. settings.settingsSelectDownloadDir()
  151. }
  152. /**
  153. * @function windowSettingsClickCheckboxVerboseMode
  154. * @summary Handles the click on the checkbox verbose mode
  155. * @description Triggered from the settingsWindow.
  156. * @memberof renderer
  157. */
  158. function windowSettingsClickCheckboxVerboseMode () {
  159. settings.settingsToggleVerboseMode()
  160. }
  161. /**
  162. * @function windowSettingsClickCheckboxAdditionalParameter
  163. * @summary Handles the click on the checkbox vadditional parameter
  164. * @description Triggered from the settingsWindow.
  165. * @memberof renderer
  166. */
  167. function windowSettingsClickCheckboxAdditionalParameter () {
  168. settings.settingsToggleAdditionalParameter()
  169. }
  170. /**
  171. * @function windowSettingsClickButtonAdditionalParameterSave
  172. * @summary Handles the click on the button additional parameter save
  173. * @description Triggered from the settingsWindow.
  174. * @memberof renderer
  175. */
  176. function windowSettingsClickButtonAdditionalParameterSave () {
  177. settings.settingsSaveAdditionalParameter()
  178. }
  179. /**
  180. * @function windowSettingsClickCheckboxUpdatePolicy
  181. * @summary Handles the click on the checkbox verbose mode
  182. * @description Triggered from the settingsWindow.
  183. * @memberof renderer
  184. */
  185. function windowSettingsClickCheckboxUpdatePolicy () {
  186. settings.settingsTogglePrereleases()
  187. }
  188. /**
  189. * @function windowSettingsClickIconBug
  190. * @summary Handles the click on the bug icon
  191. * @description Triggered from the settingsWindow.
  192. * @memberof renderer
  193. */
  194. function windowSettingsClickIconBug () {
  195. settings.settingsOpenDevTools()
  196. }
  197. /**
  198. * @function windowSettingsClickCheckboxErrorReporting
  199. * @summary Handles the click on the checkbox error reporting
  200. * @description Triggered from the settingsWindow.
  201. * @memberof renderer
  202. */
  203. function windowSettingsClickCheckboxErrorReporting () {
  204. settings.settingsToggleErrorReporting()
  205. }
  206. /**
  207. * @function windowSettingsClickCheckboxErrorReportingMoreInfo
  208. * @summary Handles the click on the question icon in the error reporting section
  209. * @description Triggered from the settingsWindow.
  210. * @memberof renderer
  211. */
  212. function windowSettingsClickCheckboxErrorReportingMoreInfo () {
  213. const { urlGithubSentryUsage } = require('./js/modules/urls.js') // get url
  214. utils.openURL(urlGithubSentryUsage)
  215. }
  216. /**
  217. * @function windowSettingsClickDropdownAudioFormats
  218. * @summary Handles the click on the dropdown audio formats
  219. * @description Triggered from the settingsWindow.
  220. * @memberof renderer
  221. */
  222. function windowSettingsClickDropdownAudioFormats () {
  223. settings.settingsAudioFormatSave()
  224. }
  225. /**
  226. * @function windowSettingsClickOpenUrl
  227. * @summary Handles the click on the dropdown audio formats
  228. * @description Triggered from the settingsWindow.
  229. * @param {string} url - the url
  230. * @memberof renderer
  231. */
  232. function windowSettingsClickOpenUrl (url) {
  233. settings.settingsOpenExternal(url)
  234. }
  235. /**
  236. * @function windowSettingsClickYoutubeDlUpdate
  237. * @summary Starts the check for youtube-dl updates routine with feedback to the user
  238. * @description Starts the check for youtube-dl updates routine with feedback to the user
  239. * @memberof renderer
  240. */
  241. function windowSettingsClickYoutubeDlUpdate () {
  242. youtubeDl.binaryUpdateCheck(false, false) // If silent = false -> Forces result feedback, even if no update is available
  243. }
  244. // ----------------------------------------------------------------------------
  245. // FUNCTIONS - OTHERS
  246. // ----------------------------------------------------------------------------
  247. /**
  248. * @function titlebarInit
  249. * @summary Init the titlebar for the frameless mainWindow
  250. * @description Creates a custom titlebar for the mainWindow using custom-electron-titlebar (https://github.com/AlexTorresSk/custom-electron-titlebar).
  251. * @memberof renderer
  252. */
  253. function titlebarInit () {
  254. const customTitlebar = require('custom-electron-titlebar')
  255. const myTitlebar = new customTitlebar.Titlebar({
  256. // new customTitlebar.Titlebar({
  257. titleHorizontalAlignment: 'center', // position of window title
  258. icon: 'img/icon/icon.png',
  259. drag: true, // whether or not you can drag the window by holding the click on the title bar.
  260. backgroundColor: customTitlebar.Color.fromHex('#343a40'),
  261. minimizable: true,
  262. maximizable: true,
  263. closeable: true,
  264. unfocusEffect: false, // added in 0.9.0
  265. itemBackgroundColor: customTitlebar.Color.fromHex('#ffe5e5') // menu item -> hover color
  266. })
  267. // Be aware: the font-size of .window-title (aka application name) is set by app/css/core.css
  268. utils.writeConsoleMsg('info', 'titlebarInit ::: Initialized custom titlebar')
  269. }
  270. /**
  271. * @function checkApplicationDependencies
  272. * @function checkApplicationDependencies
  273. * @summary Checks for missing dependencies
  274. * @description Checks on startup for missing dependencies (youtube-dl and ffmpeg). Both are bundles and should be find
  275. * @memberof renderer
  276. */
  277. function checkApplicationDependencies () {
  278. var countErrors = 0
  279. // youtube-dl
  280. //
  281. var youtubeDlBinaryPath = youtubeDl.binaryPathGet()
  282. if (utils.pathExists(youtubeDlBinaryPath) === true) {
  283. utils.writeConsoleMsg('info', 'checkApplicationDependencies ::: Found youtube-dl in: _' + youtubeDlBinaryPath + '_.')
  284. } else {
  285. countErrors = countErrors + 1
  286. utils.writeConsoleMsg('error', 'checkApplicationDependencies ::: Unable to find youtube-dl in: _' + youtubeDlBinaryPath + '_.')
  287. utils.showNoty('error', 'Unable to find dependency <b>youtube-dl</b>. Please report this.', 0)
  288. }
  289. // ffmpeg
  290. //
  291. var ffmpegBinaryPath = ffmpeg.ffmpegGetBinaryPath()
  292. if (utils.pathExists(ffmpegBinaryPath) === true) {
  293. utils.writeConsoleMsg('info', 'checkApplicationDependencies ::: Found ffmpeg in: _' + ffmpegBinaryPath + '_.')
  294. } else {
  295. countErrors = countErrors + 1
  296. utils.writeConsoleMsg('error', 'checkApplicationDependencies ::: Unable to find ffmpeg in: _' + ffmpegBinaryPath + '_.')
  297. utils.showNoty('error', 'Unable to find dependency <b>ffmpeg</b>. Please report this', 0)
  298. }
  299. // if errors occured - disable / hide the action buttons
  300. //
  301. if (countErrors !== 0) {
  302. $('#buttonStartVideoExec').hide() // hide video button
  303. $('#buttonStartVideo').hide() // hide video button
  304. $('#buttonStartAudioExec').hide() // hide audio button
  305. utils.showNoty('error', 'Download buttons are now hidden. Please contact the developers via github.', 0)
  306. }
  307. utils.writeConsoleMsg('info', 'checkApplicationDependencies ::: Finished checking dependencies. Found overall _' + countErrors + '_ problems.')
  308. }
  309. /**
  310. * @function settingsLoadAllOnAppStart
  311. * @summary Reads all user-setting-files and fills some global variables
  312. * @description Reads all user-setting-files and fills some global variables
  313. * @memberof renderer
  314. */
  315. function settingsLoadAllOnAppStart () {
  316. utils.writeConsoleMsg('info', 'settingsLoadAllOnAppStart ::: Gonna read several user config files now ...')
  317. utils.userSettingRead('enableVerboseMode') // verbose mode
  318. utils.userSettingRead('enableUrlInformations') // url informations
  319. utils.userSettingRead('enableAdditionalParameter') // additional parameter
  320. utils.userSettingRead('additionalYoutubeDlParameter') // additional parameter
  321. utils.userSettingRead('enablePrereleases') // pre-releases
  322. utils.userSettingRead('enableErrorReporting') // get setting for error-reporting
  323. utils.userSettingRead('downloadDir') // download dir
  324. utils.userSettingRead('audioFormat') // get setting for configured audio format
  325. }
  326. /**
  327. * @function settingsLoadAllOnSettingsUiLoad
  328. * @summary Reads all user-setting-files and fills some global variables and adjusts the settings UI
  329. * @description Reads all user-setting-files and fills some global variables and adjusts the settings UI
  330. * @memberof renderer
  331. */
  332. function settingsLoadAllOnSettingsUiLoad () {
  333. utils.writeConsoleMsg('info', 'settingsLoadAllOnAppStart ::: Gonna read several user config files now and adjust the settings UI')
  334. utils.userSettingRead('enableVerboseMode', true) // verbose mode
  335. utils.userSettingRead('enableUrlInformations', true) // url informations
  336. utils.userSettingRead('enableAdditionalParameter', true) // enable or not: additional parameter
  337. utils.userSettingRead('additionalYoutubeDlParameter', true) // the actual additional youtube-dl parameter
  338. utils.userSettingRead('enablePrereleases', true) // pre-releases
  339. utils.userSettingRead('enableErrorReporting', true) // get setting for error-reporting
  340. utils.userSettingRead('downloadDir', true) // download dir
  341. utils.userSettingRead('audioFormat', true) // load configured audio format and update the settings UI
  342. settings.settingsEnableOrDisableYoutubeDLUpdateButton()
  343. }
  344. /**
  345. * @function urlInputFieldOnKeyUp
  346. * @summary On Key up event, checks if the field is empty or not
  347. * @description On Key up event, checks if the field is empty or not
  348. * @memberof renderer
  349. */
  350. function urlInputFieldOnKeyUp () {
  351. var currentContentOfUrlInputField = $('#inputNewUrl').val() // get current content of field
  352. if (currentContentOfUrlInputField === '') {
  353. utils.writeConsoleMsg('info', 'urlInputFieldOnKeyUp ::: Is now empty, gonna reset the background color')
  354. ui.inputUrlFieldSetState() // empty = white
  355. } else {
  356. ui.inputUrlFieldSetState('unchecked') // unchecked = light red
  357. var isUrlValid = utils.validURL(currentContentOfUrlInputField)
  358. if (isUrlValid) {
  359. utils.writeConsoleMsg('info', 'urlInputFieldOnKeyUp ::: User input is a valid URL (' + currentContentOfUrlInputField + ').')
  360. ui.inputUrlFieldSetState('valid') // valid = green
  361. } else {
  362. // utils.writeConsoleMsg('info', 'urlInputFieldOnKeyUp ::: User input is not a valid URL (' + currentContentOfUrlInputField + ').')
  363. }
  364. }
  365. }
  366. /**
  367. * @function urlInputFieldOnKeyPress
  368. * @summary Executed on keypress inside url-input-field
  369. * @description Checks if the key-press was the ENTER-key - if so simulates a press of the button ADD URL
  370. * @memberof renderer
  371. * @event keyCode - The key press event
  372. */
  373. function urlInputFieldOnKeyPress (event) {
  374. var code = 0
  375. code = event.keyCode
  376. if (code === 13) {
  377. windowMainClickButtonAddUrl() // simulare click on ADD URL buttom
  378. }
  379. }
  380. /**
  381. * @function urlInputFieldOnFocus
  382. * @summary Handles auto-pasting urls to url input field
  383. * @description Executed on focus - checks if the clipboard contains a valid URL - if so - its auto-pasted into the field
  384. * @memberof renderer
  385. */
  386. function urlInputFieldOnFocus () {
  387. utils.writeConsoleMsg('info', 'urlInputFieldOnFocus ::: url input field got focus')
  388. var currentContentOfUrlInputField = $('#inputNewUrl').val() // get current content of field
  389. // if the field is empty - continue
  390. if (currentContentOfUrlInputField === '') {
  391. const { clipboard } = require('electron')
  392. var currentClipboardContent = clipboard.readText() // get content of clipboard
  393. currentClipboardContent = currentClipboardContent.trim() // remove leading and trailing blanks
  394. var isUrlValid = utils.validURL(currentClipboardContent)
  395. if (isUrlValid) {
  396. utils.writeConsoleMsg('info', 'urlInputFieldOnFocus ::: Clipboard contains a valid URL: _' + currentClipboardContent + '_.')
  397. $('#inputNewUrl').val(currentClipboardContent) // paste it
  398. $('#inputNewUrl').select() // select it entirely
  399. ui.inputUrlFieldSetState('valid') // valid = green
  400. } else {
  401. utils.writeConsoleMsg('warn', 'urlInputFieldOnFocus ::: Clipboard contains a non valid URL: _' + currentClipboardContent + '_.')
  402. }
  403. } else {
  404. // input field is not empty
  405. // doing nothing at this point
  406. }
  407. }
  408. /**
  409. * @function searchUpdate
  410. * @summary Checks if there is a new media-dupes release available
  411. * @description Compares the local app version number with the tag of the latest github release. Displays a notification in the settings window if an update is available. Is executed on app launch NOT on reload.
  412. * @memberof renderer
  413. * @param {booean} [silent] - Boolean with default value. Shows a feedback in case of no available updates If 'silent' = false. Special handling for manually triggered update search
  414. */
  415. function searchUpdate (silent = true) {
  416. var semver = require('semver')
  417. ui.windowMainApplicationStateSet('Searching media-dupes updates')
  418. // check if pre-releases should be included or not
  419. var curEnablePrereleasesSetting = utils.globalObjectGet('enablePrereleases')
  420. // get url for github releases / api
  421. const { urlGithubApiReleases } = require('./js/modules/urls.js') // get API url
  422. var remoteAppVersionLatest = '0.0.0'
  423. var remoteAppVersionLatestPrerelease = false
  424. var localAppVersion = '0.0.0'
  425. var versions
  426. // get local version
  427. //
  428. localAppVersion = require('electron').remote.app.getVersion()
  429. // var updateStatus = $.get(urlGithubApiReleases, function (data, status) {
  430. $.get(urlGithubApiReleases, function (data, status) {
  431. // 3000 // in milliseconds
  432. utils.writeConsoleMsg('info', 'searchUpdate ::: Accessing _' + urlGithubApiReleases + '_ ended with: _' + status + '_')
  433. // success
  434. versions = data.sort(function (v1, v2) {
  435. // return semver.compare(v2.tag_name, v1.tag_name);
  436. // console.error(v1.tag_name)
  437. // console.error(v2.tag_name)
  438. })
  439. if (curEnablePrereleasesSetting === true) {
  440. // user wants the latest release - ignoring if it is a prerelease or an official one
  441. utils.writeConsoleMsg('info', 'searchUpdate ::: Including pre-releases in update search')
  442. remoteAppVersionLatest = versions[0].tag_name // Example: 0.4.2
  443. remoteAppVersionLatestPrerelease = versions[0].prerelease // boolean
  444. } else {
  445. // user wants official releases only
  446. utils.writeConsoleMsg('info', 'searchUpdate ::: Ignoring pre-releases in update search')
  447. // find the latest non pre-release build
  448. // loop over the versions array to find the latest non-pre-release
  449. // var latestOfficialRelease
  450. for (var i = 0; i < versions.length; i++) {
  451. if (versions[i].prerelease === false) {
  452. // latestOfficialRelease = i
  453. break
  454. }
  455. }
  456. remoteAppVersionLatest = versions[i].tag_name // Example: 0.4.2
  457. remoteAppVersionLatestPrerelease = versions[i].prerelease // boolean
  458. }
  459. // simulate different update scenarios:
  460. //
  461. // localAppVersion = '0.0.1'; // overwrite variable to simulate
  462. // remoteAppVersionLatest = 'v0.6.0' // overwrite variable to simulate
  463. // strip the v away
  464. // - up to 0.5.0 the tag used on github did not start with v.
  465. // - comapring versions without leading chars is much easier.
  466. localAppVersion = localAppVersion.replace('v', '')
  467. remoteAppVersionLatest = remoteAppVersionLatest.replace('v', '')
  468. utils.writeConsoleMsg('info', 'searchUpdate ::: Local media-dupes version: ' + localAppVersion)
  469. utils.writeConsoleMsg('info', 'searchUpdate ::: Latest media-dupes version: ' + remoteAppVersionLatest)
  470. // If a stable (not a prelease) update is available - see #73
  471. // if (localAppVersion < remoteAppVersionLatest) {
  472. if (semver.lt(localAppVersion, remoteAppVersionLatest)) {
  473. utils.writeConsoleMsg('info', 'searchUpdate ::: Found update, notify user')
  474. // prepare the message for the user - depending on the fact if it is a pre-release or not
  475. var updateText
  476. if (remoteAppVersionLatestPrerelease === false) {
  477. updateText = 'A media-dupes update from <b>' + localAppVersion + '</b> to version <b>' + remoteAppVersionLatest + '</b> is available. Do you want to visit the release page?'
  478. } else {
  479. updateText = 'A media-dupes <b>pre-release</b> update from <b>' + localAppVersion + '</b> to version <b>' + remoteAppVersionLatest + '</b> is available. Do you want to visit the release page?'
  480. }
  481. // ask user using a noty confirm dialog
  482. const Noty = require('noty')
  483. var n = new Noty(
  484. {
  485. theme: 'bootstrap-v4',
  486. layout: 'bottom',
  487. type: 'info',
  488. closeWith: [''], // to prevent closing the confirm-dialog by clicking something other then a confirm-dialog-button
  489. text: updateText,
  490. buttons: [
  491. Noty.button('Yes', 'btn btn-success mediaDupes_btnDefaultWidth', function () {
  492. n.close()
  493. openReleasesOverview()
  494. },
  495. {
  496. id: 'button1', 'data-status': 'ok'
  497. }),
  498. Noty.button('No', 'btn btn-secondary mediaDupes_btnDefaultWidth float-right', function () {
  499. n.close()
  500. if (remoteAppVersionLatestPrerelease === false) {
  501. utils.showNoty('warning', 'Please be aware that not updating <b>media-dupes</b> will result in an <b>outdated youtube-dl version</b> which again will result in download errors.', 0)
  502. }
  503. })
  504. ]
  505. })
  506. // show the noty dialog
  507. n.show()
  508. } else {
  509. utils.writeConsoleMsg('info', 'searchUpdate ::: No newer version of media-dupes found.')
  510. // when executed manually via menu -> user should see result of this search
  511. if (silent === false) {
  512. utils.showNoty('info', 'No updates for <b>media-dupes (' + localAppVersion + ')</b> available.')
  513. }
  514. }
  515. utils.writeConsoleMsg('info', 'searchUpdate ::: Successfully checked ' + urlGithubApiReleases + ' for available releases')
  516. })
  517. .done(function () {
  518. // utils.writeConsoleMsg('info', 'searchUpdate ::: Successfully checked ' + urlGithubApiReleases + ' for available releases');
  519. })
  520. .fail(function () {
  521. utils.writeConsoleMsg('info', 'searchUpdate ::: Checking ' + urlGithubApiReleases + ' for available releases failed.')
  522. utils.showNoty('error', 'Checking <b>' + urlGithubApiReleases + '</b> for available media-dupes releases failed. Please troubleshoot your network connection.', 0)
  523. })
  524. .always(function () {
  525. utils.writeConsoleMsg('info', 'searchUpdate ::: Finished checking ' + urlGithubApiReleases + ' for available releases')
  526. ui.windowMainButtonsOthersEnable()
  527. ui.windowMainApplicationStateSet()
  528. })
  529. }
  530. /**
  531. * @function openReleasesOverview
  532. * @summary Opens the media-dupes release page
  533. * @description Opens the url https://github.com/yafp/media-dupes/releases in the default browser. Used in searchUpdate().
  534. * @memberof renderer
  535. */
  536. function openReleasesOverview () {
  537. const { urlGitHubReleases } = require('./js/modules/urls.js')
  538. utils.writeConsoleMsg('info', 'openReleasesOverview ::: Opening _' + urlGitHubReleases + '_ to show available releases.')
  539. utils.openURL(urlGitHubReleases)
  540. }
  541. /**
  542. * @function settingsShowYoutubeDLInfo
  543. * @summary Searches the youtube-binary and shows it in the settings UI
  544. * @description Searches the youtube-binary and shows it in the settings UI
  545. * @memberof renderer
  546. */
  547. function settingsShowYoutubeDLInfo () {
  548. const youtubedl = require('youtube-dl')
  549. settingsGetYoutubeDLBinaryVersion(function () {
  550. utils.writeConsoleMsg('info', 'settingsShowYoutubeDLInfo ::: Searching youtube-dl ...')
  551. var youtubeDl = youtubedl.getYtdlBinary()
  552. if (youtubeDl === '') {
  553. utils.writeConsoleMsg('error', 'settingsShowYoutubeDLInfo ::: Unable to find youtube-dl')
  554. utils.showNoty('error', 'Unable to find dependency <b>youtube-dl</b>.', 0)
  555. } else {
  556. utils.writeConsoleMsg('info', 'settingsShowYoutubeDLInfo ::: Found youtube-dl in: _' + youtubeDl + '_.')
  557. $('#userSettingsYouTubeDLPathInfo').val(youtubeDl) // show in UI
  558. $('#headerYoutubeDL').html('Installation <small>Version: ' + ytdlBinaryVersion + '</small>')
  559. }
  560. })
  561. }
  562. /**
  563. * @function settingsShowFfmpegInfo
  564. * @summary Searches the ffmpeg-binary and shows it in the settings UI
  565. * @description Searches the ffmpeg-binary and shows it in the settings UI
  566. * @memberof renderer
  567. */
  568. function settingsShowFfmpegInfo () {
  569. var ffmpeg = require('ffmpeg-static-electron')
  570. // utils.writeConsoleMsg('info', 'settingsShowFfmpegInfo ::: Searching ffmpeg ...')
  571. if (ffmpeg === '') {
  572. utils.writeConsoleMsg('error', 'settingsShowFfmpegInfo ::: Unable to find ffmpeg')
  573. utils.showNoty('error', 'Unable to find dependency <b>ffmpeg</b>.', 0)
  574. } else {
  575. utils.writeConsoleMsg('info', 'settingsShowFfmpegInfo ::: Found ffmpeg in: _' + ffmpeg.path + '_.')
  576. $('#userSettingsFfmpegPathInfo').val(ffmpeg.path) // show in UI
  577. }
  578. }
  579. /**
  580. * @function settingsGetYoutubeDLBinaryVersion
  581. * @summary Gets the youtube-dl binary version and displays it in settings ui
  582. * @description Reads the youtube-dl binary version from 'node_modules/youtube-dl/bin/details'
  583. * @memberof renderer
  584. * @return ytdlBinaryVersion - The youtube-dl binary version string
  585. */
  586. function settingsGetYoutubeDLBinaryVersion (_callback) {
  587. const fs = require('fs')
  588. var youtubeDlBinaryDetailsPath = youtubeDl.binaryDetailsPathGet() // get path to youtube-dl binary details
  589. fs.readFile(youtubeDlBinaryDetailsPath, 'utf8', function (error, contents) {
  590. if (error) {
  591. utils.writeConsoleMsg('error', 'settingsGetYoutubeDLBinaryVersion ::: Unable to detect youtube-dl binary version. Error: ' + error + '.')
  592. utils.showNoty('error', 'Unable to detect local youtube-dl binary version number. Error: ' + error, 0) // see sentry issue: MEDIA-DUPES-5A
  593. throw error
  594. } else {
  595. const data = JSON.parse(contents)
  596. ytdlBinaryVersion = data.version // extract and store the version number
  597. utils.writeConsoleMsg('info', 'settingsGetYoutubeDLBinaryVersion ::: youtube-dl binary is version: _' + ytdlBinaryVersion + '_.')
  598. _callback()
  599. }
  600. })
  601. }
  602. /**
  603. * @function validateUrlBeforeAdd
  604. * @summary Gets the content of the url field, checks if it is a valid url, if so checks if it is reachable or not
  605. * @description Gets the content of the url field, checks if it is a valid url, if so checks if it is reachable or not
  606. * @memberof renderer
  607. */
  608. function validateUrlBeforeAdd () {
  609. var currentContentOfUrlInputField = $('#inputNewUrl').val() // get current content of field
  610. // if the field is empty - continue
  611. if (currentContentOfUrlInputField === '') {
  612. utils.writeConsoleMsg('info', 'validateUrlBeforeAdd ::: Empty field')
  613. } else {
  614. var isUrlValid = utils.validURL(currentContentOfUrlInputField)
  615. if (isUrlValid) {
  616. utils.writeConsoleMsg('info', 'validateUrlBeforeAdd ::: URL seems valid URL (' + currentContentOfUrlInputField + ').')
  617. } else {
  618. utils.writeConsoleMsg('info', 'urlInputFieldOnFocus ::: Clipboard contains a non valid URL (' + currentContentOfUrlInputField + ').')
  619. }
  620. }
  621. }
  622. // ----------------------------------------------------------------------------
  623. // IPC
  624. // ----------------------------------------------------------------------------
  625. // ----------------------------------------------------------------------------
  626. // IPC - on ready-to-show
  627. // ----------------------------------------------------------------------------
  628. /**
  629. * @name startSearchUpdatesSilent
  630. * @summary Triggers the check for media-dupes updates in silent mode
  631. * @description Called via ipc from main.js on-ready to start the search for media-dupes updates
  632. * @memberof renderer
  633. */
  634. /*
  635. require('electron').ipcRenderer.on('startSearchUpdatesSilent', function () {
  636. searchUpdate(true) // If silent = false -> Forces result feedback, even if no update is available
  637. })
  638. */
  639. /**
  640. * @name youtubeDlSearchUpdatesSilent
  641. * @summary Triggers the check for youtube-dl updates in silent mode
  642. * @description Called via ipc from main.js on-ready to check the search for youtube-dl updates
  643. * @memberof renderer
  644. */
  645. /*
  646. require('electron').ipcRenderer.on('youtubeDlSearchUpdatesSilent', function () {
  647. youtubeDl.binaryUpdateCheck(true, false) // If silent = false -> Forces result feedback, even if no update is available
  648. })
  649. */
  650. /**
  651. * @name initSettings
  652. * @summary Triggers the check for the application dependencies
  653. * @description Called via ipc from main.js on-ready to check the application dependencies
  654. * @memberof renderer
  655. */
  656. /*
  657. require('electron').ipcRenderer.on('initSettings', function () {
  658. settingsLoadAllOnAppStart()
  659. })
  660. */
  661. /**
  662. * @name startCheckingDependencies
  663. * @summary Triggers the check for the application dependencies
  664. * @description Called via ipc from main.js on-ready to check the application dependencies
  665. * @memberof renderer
  666. */
  667. require('electron').ipcRenderer.on('startCheckingDependencies', function () {
  668. checkApplicationDependencies()
  669. })
  670. /**
  671. * @name scheduleUpdateCheckMediaDupes
  672. * @summary Starts the silent search for media-dupes updates
  673. * @description Starts the silent search for media-dupes updates after several seconds (to speed up the application startup)
  674. * @memberof renderer
  675. */
  676. require('electron').ipcRenderer.on('scheduleUpdateCheckMediaDupes', function () {
  677. setTimeout(
  678. function () {
  679. utils.writeConsoleMsg('info', 'scheduleUpdateCheckMediaDupes ::: Starting scheduled search for new media-dupes updates.')
  680. searchUpdate(true) // silent
  681. }, 3000) // after 3 seconds
  682. })
  683. /**
  684. * @name scheduleUpdateCheckYoutubeDl
  685. * @summary Starts the silent search for youtube-dl updates
  686. * @description Starts the silent search for youtube-dl updates after several seconds (to speed up the application startup)
  687. * @memberof renderer
  688. */
  689. require('electron').ipcRenderer.on('scheduleUpdateCheckYoutubeDl', function () {
  690. setTimeout(
  691. function () {
  692. utils.writeConsoleMsg('info', 'scheduleUpdateCheckYoutubeDl ::: Starting scheduled search for new youtube-dl updates.')
  693. youtubeDl.binaryUpdateCheck(true, false) // If silent = false -> Forces result feedback, even if no update is available
  694. }, 3000) // after 3 seconds
  695. })
  696. /**
  697. * @name startDisclaimerCheck
  698. * @summary Triggers the check for disclaimer need
  699. * @description Called via ipc from main.js on-ready to check for the disclaimer need
  700. * @memberof renderer
  701. */
  702. require('electron').ipcRenderer.on('startDisclaimerCheck', function () {
  703. utils.disclaimerCheck()
  704. })
  705. /**
  706. * @name todoListCheck
  707. * @summary Triggers the check restoring previosly stored urls
  708. * @description Called via ipc from main.js on-ready-to-show and starts the restore function
  709. * @memberof renderer
  710. */
  711. require('electron').ipcRenderer.on('todoListCheck', function () {
  712. // utils.writeConsoleMsg('info', 'todoListCheck ::: main.js forces renderer to check for urls to restore')
  713. ui.windowMainToDoListRestore()
  714. })
  715. // ----------------------------------------------------------------------------
  716. // IPC - by menu
  717. // ----------------------------------------------------------------------------
  718. /**
  719. * @name startSearchUpdatesVerbose
  720. * @summary Start searching for updates in non-silent mode
  721. * @description Called via ipc from main.js / menu to search for applicatipn updates
  722. * @memberof renderer
  723. */
  724. require('electron').ipcRenderer.on('startSearchUpdatesVerbose', function () {
  725. searchUpdate(false) // silent = false. Forces result feedback, even if no update is available
  726. })
  727. /**
  728. * @name openSettings
  729. * @summary Triggers loading the settings UI
  730. * @description Called via ipc from main.js / menu to open the Settings UI
  731. * @memberof renderer
  732. */
  733. require('electron').ipcRenderer.on('openSettings', function () {
  734. ui.windowMainSettingsUiLoad()
  735. })
  736. /**
  737. * @name openYoutubeSuggestDialog
  738. * @summary Triggers a input dialog to search for youtube suggest based on an input string
  739. * @description Called via ipc from main.js / menu to ....
  740. * @memberof renderer
  741. */
  742. require('electron').ipcRenderer.on('openYoutubeSuggestDialog', function () {
  743. ui.youtubeSuggest()
  744. })
  745. /**
  746. * @name youtubeDlBinaryUpdate
  747. * @summary Triggers updating the youtube-dl binary
  748. * @description Called via ipc from main.js / menu to update the youtube-dl binary
  749. * @memberof renderer
  750. */
  751. require('electron').ipcRenderer.on('youtubeDlBinaryUpdate', function () {
  752. youtubeDl.binaryUpdateCheck(false, true) // silent = false && force = true
  753. })
  754. /**
  755. * @name youtubeDlBinaryPathReset
  756. * @summary Triggers resetting the path to the youtube-dl binary back to default
  757. * @description Called via ipc from main.js / menu to reset the path to the youtube-dl binary
  758. * @memberof renderer
  759. */
  760. require('electron').ipcRenderer.on('youtubeDlBinaryPathReset', function () {
  761. var youtubeDlBinaryDetailsPath = youtubeDl.binaryDetailsPathGet() // get path to youtube-dl binary details file
  762. utils.canWriteFileOrFolder(youtubeDlBinaryDetailsPath, function (error, isWritable) {
  763. if (error) {
  764. utils.writeConsoleMsg('error', 'youtubeDlBinaryPathReset ::: Error while trying to check if the youtube-dl details file is writeable or not. Error: ' + error)
  765. throw error
  766. }
  767. if (isWritable === true) {
  768. utils.writeConsoleMsg('info', 'youtubeDlBinaryPathReset ::: : Found the youtube-dl details file and it is writeable. Gonna ask the user now if he wants to reset the path now')
  769. // ask the user if he wants to update using a confirm dialog
  770. const Noty = require('noty')
  771. var n = new Noty(
  772. {
  773. theme: 'bootstrap-v4',
  774. layout: 'bottom',
  775. type: 'info',
  776. closeWith: [''], // to prevent closing the confirm-dialog by clicking something other then a confirm-dialog-button
  777. text: 'Do you really want to reset the youtube-dl binary path back to its default value?',
  778. buttons: [
  779. Noty.button('Yes', 'btn btn-success mediaDupes_btnDownloadActionWidth', function () {
  780. n.close()
  781. youtubeDl.binaryPathReset(youtubeDlBinaryDetailsPath)
  782. },
  783. {
  784. id: 'button1', 'data-status': 'ok'
  785. }),
  786. Noty.button('No', 'btn btn-secondary mediaDupes_btnDownloadActionWidth float-right', function () {
  787. n.close()
  788. })
  789. ]
  790. })
  791. n.show() // show the noty dialog
  792. } else {
  793. // details file cant be resetted due to permission issues
  794. utils.writeConsoleMsg('warn', 'youtubeDlBinaryPathReset ::: Found youtube-dl binary update, but unable to execute update due to permissions')
  795. utils.showNoty('error', 'Unable to reset the <b>youtube-dl</b> setup due to permissions issues. The file: ' + youtubeDlBinaryDetailsPath + ' is not writeable.')
  796. }
  797. })
  798. utils.writeConsoleMsg('info', 'youtubeDlBinaryPathReset ::: Finished re-setting the binary path for youtube-dl')
  799. })
  800. // ----------------------------------------------------------------------------
  801. // IPC - by power-state
  802. // ----------------------------------------------------------------------------
  803. /**
  804. * @name powerMonitorNotification
  805. * @summary Triggers power specific notifications to the UI
  806. * @description Called via ipc from main.js to trigger a notification about the powerState
  807. * @memberof renderer
  808. */
  809. require('electron').ipcRenderer.on('powerMonitorNotification', function (event, messageType, messageText, messageDuration) {
  810. utils.writeConsoleMsg('warn', 'powerMonitorNotification ::: Main wants to show a notification of the type: _' + messageType + '_ and the message: _' + messageText + '_ with the duration: _' + messageDuration + '_.')
  811. utils.showNoty(messageType, messageText, messageDuration)
  812. })
  813. // ----------------------------------------------------------------------------
  814. // IPC - by settings window closed
  815. // ----------------------------------------------------------------------------
  816. /**
  817. * @name unblurMainUI
  818. * @summary Triggers unbluring the UI
  819. * @description Called via ipc from main.js when the settings UI got closed to trigger unblur'ing the main UI
  820. * @memberof renderer
  821. */
  822. require('electron').ipcRenderer.on('unblurMainUI', function () {
  823. ui.windowMainBlurSet(false)
  824. })
  825. /**
  826. * @name blurMainUI
  827. * @summary Triggers bluring the UI
  828. * @description Called via ipc from main.js when the main window is ready-to-show
  829. * @memberof renderer
  830. */
  831. require('electron').ipcRenderer.on('blurMainUI', function () {
  832. ui.windowMainBlurSet(true)
  833. })
  834. /**
  835. * @name countAppStarts
  836. * @summary ..
  837. * @description Called via ipc from main.js when the main window is ready-to-show
  838. * @memberof renderer
  839. */
  840. require('electron').ipcRenderer.on('countAppStarts', function () {
  841. sentry.countEvent('usageApplicationStarted')
  842. })
  843. // ----------------------------------------------------------------------------
  844. // IPC - by mainwindow close event
  845. // ----------------------------------------------------------------------------
  846. /**
  847. * @name todoListTryToSave
  848. * @summary Triggers saving of the current todoList
  849. * @description Called via ipc from main.js when the application gets closed Should save existing todoList entries
  850. * @memberof renderer
  851. */
  852. require('electron').ipcRenderer.on('todoListTryToSave', function () {
  853. ui.windowMainToDoListSave()
  854. })