- Fix bug where new install was relying on the migration to run on startup to fix up the default config - Moved save/load functions into state.js - Removed exported getInitialState, getDefaultSavedState since that's leaky. The state module should take care of that.
1299 lines
39 KiB
JavaScript
1299 lines
39 KiB
JavaScript
console.time('init')
|
|
|
|
var crashReporter = require('../crash-reporter')
|
|
crashReporter.init()
|
|
|
|
var dragDrop = require('drag-drop')
|
|
var electron = require('electron')
|
|
var fs = require('fs-extra')
|
|
var mainLoop = require('main-loop')
|
|
var parallel = require('run-parallel')
|
|
var path = require('path')
|
|
|
|
var createElement = require('virtual-dom/create-element')
|
|
var diff = require('virtual-dom/diff')
|
|
var patch = require('virtual-dom/patch')
|
|
|
|
var App = require('./views/app')
|
|
var config = require('../config')
|
|
var errors = require('./lib/errors')
|
|
var sound = require('./lib/sound')
|
|
var State = require('./lib/state')
|
|
var TorrentPlayer = require('./lib/torrent-player')
|
|
var TorrentSummary = require('./lib/torrent-summary')
|
|
var {setDispatch} = require('./lib/dispatcher')
|
|
|
|
// Electron apps have two processes: a main process (node) runs first and starts
|
|
// a renderer process (essentially a Chrome window). We're in the renderer process,
|
|
// and this IPC channel receives from and sends messages to the main process
|
|
var ipcRenderer = electron.ipcRenderer
|
|
|
|
var state, vdomLoop
|
|
|
|
// This dependency is the slowest-loading, so we lazy load it
|
|
var Cast = null
|
|
|
|
init()
|
|
|
|
function init () {
|
|
// All state lives in state.js. `state.saved` is read from and written to a file.
|
|
// All other state is ephemeral. First we load state.saved then initialize the app.
|
|
State.load(onState)
|
|
|
|
setDispatch(dispatch)
|
|
}
|
|
|
|
/**
|
|
* Called once when the application loads. (Not once per window.)
|
|
* Connects to the torrent networks, sets up the UI and OS integrations like
|
|
* the dock icon and drag+drop.
|
|
*/
|
|
function onState (err, _state) {
|
|
if (err) return onError(err)
|
|
state = _state
|
|
|
|
// Add first page to location history
|
|
state.location.go({ url: 'home' })
|
|
|
|
// Restart everything we were torrenting last time the app ran
|
|
resumeTorrents()
|
|
|
|
// Lazy-load other stuff, like the AppleTV module, later to keep startup fast
|
|
window.setTimeout(delayedInit, config.DELAYED_INIT)
|
|
|
|
// The UI is built with virtual-dom, a minimalist library extracted from React
|
|
// The concepts--one way data flow, a pure function that renders state to a
|
|
// virtual DOM tree, and a diff that applies changes in the vdom to the real
|
|
// DOM, are all the same. Learn more: https://facebook.github.io/react/
|
|
vdomLoop = mainLoop(state, render, {
|
|
create: createElement,
|
|
diff: diff,
|
|
patch: patch
|
|
})
|
|
document.body.appendChild(vdomLoop.target)
|
|
|
|
// Listen for messages from the main process
|
|
setupIpc()
|
|
|
|
// Calling update() updates the UI given the current state
|
|
// Do this at least once a second to give every file in every torrentSummary
|
|
// a progress bar and to keep the cursor in sync when playing a video
|
|
setInterval(update, 1000)
|
|
|
|
// OS integrations:
|
|
// ...drag and drop a torrent or video file to play or seed
|
|
dragDrop('body', onOpen)
|
|
|
|
// ...same thing if you paste a torrent
|
|
document.addEventListener('paste', onPaste)
|
|
|
|
// ...focus and blur. Needed to show correct dock icon text ("badge") in OSX
|
|
window.addEventListener('focus', onFocus)
|
|
window.addEventListener('blur', onBlur)
|
|
|
|
sound.play('STARTUP')
|
|
|
|
// Done! Ideally we want to get here < 500ms after the user clicks the app
|
|
console.timeEnd('init')
|
|
}
|
|
|
|
function delayedInit () {
|
|
lazyLoadCast()
|
|
sound.preload()
|
|
}
|
|
|
|
// Lazily loads Chromecast and Airplay support
|
|
function lazyLoadCast () {
|
|
if (!Cast) {
|
|
Cast = require('./lib/cast')
|
|
Cast.init(state, update) // Search the local network for Chromecast and Airplays
|
|
}
|
|
return Cast
|
|
}
|
|
|
|
// This is the (mostly) pure function from state -> UI. Returns a virtual DOM
|
|
// tree. Any events, such as button clicks, will turn into calls to dispatch()
|
|
function render (state) {
|
|
try {
|
|
return App(state)
|
|
} catch (e) {
|
|
console.log('rendering error: %s\n\t%s', e.message, e.stack)
|
|
}
|
|
}
|
|
|
|
// Calls render() to go from state -> UI, then applies to vdom to the real DOM.
|
|
function update () {
|
|
showOrHidePlayerControls()
|
|
vdomLoop.update(state)
|
|
updateElectron()
|
|
}
|
|
|
|
function updateElectron () {
|
|
if (state.window.title !== state.prev.title) {
|
|
state.prev.title = state.window.title
|
|
ipcRenderer.send('setTitle', state.window.title)
|
|
}
|
|
if (state.dock.progress !== state.prev.progress) {
|
|
state.prev.progress = state.dock.progress
|
|
ipcRenderer.send('setProgress', state.dock.progress)
|
|
}
|
|
if (state.dock.badge !== state.prev.badge) {
|
|
state.prev.badge = state.dock.badge
|
|
ipcRenderer.send('setBadge', state.dock.badge || '')
|
|
}
|
|
}
|
|
|
|
// Events from the UI never modify state directly. Instead they call dispatch()
|
|
function dispatch (action, ...args) {
|
|
// Log dispatch calls, for debugging
|
|
if (!['mediaMouseMoved', 'mediaTimeUpdate'].includes(action)) {
|
|
console.log('dispatch: %s %o', action, args)
|
|
}
|
|
|
|
if (action === 'onOpen') {
|
|
onOpen(args[0] /* files */)
|
|
}
|
|
if (action === 'addTorrent') {
|
|
addTorrent(args[0] /* torrent */)
|
|
}
|
|
if (action === 'openTorrentFile') {
|
|
ipcRenderer.send('openTorrentFile') /* open torrent file */
|
|
}
|
|
if (action === 'openFiles') {
|
|
ipcRenderer.send('openFiles') /* add files with dialog */
|
|
}
|
|
if (action === 'showCreateTorrent') {
|
|
showCreateTorrent(args[0] /* paths */)
|
|
}
|
|
if (action === 'openTorrentAddress') {
|
|
state.modal = { id: 'open-torrent-address-modal' }
|
|
update()
|
|
}
|
|
if (action === 'createTorrent') {
|
|
createTorrent(args[0] /* options */)
|
|
}
|
|
if (action === 'openItem') {
|
|
openItem(args[0] /* infoHash */, args[1] /* index */)
|
|
}
|
|
if (action === 'toggleTorrent') {
|
|
toggleTorrent(args[0] /* infoHash */)
|
|
}
|
|
if (action === 'deleteTorrent') {
|
|
deleteTorrent(args[0] /* infoHash */)
|
|
}
|
|
if (action === 'toggleSelectTorrent') {
|
|
toggleSelectTorrent(args[0] /* infoHash */)
|
|
}
|
|
if (action === 'toggleTorrentFile') {
|
|
toggleTorrentFile(args[0] /* infoHash */, args[1] /* index */)
|
|
}
|
|
if (action === 'openTorrentContextMenu') {
|
|
openTorrentContextMenu(args[0] /* infoHash */)
|
|
}
|
|
if (action === 'openDevice') {
|
|
lazyLoadCast().open(args[0] /* deviceType */)
|
|
}
|
|
if (action === 'closeDevice') {
|
|
lazyLoadCast().close()
|
|
}
|
|
if (action === 'setDimensions') {
|
|
setDimensions(args[0] /* dimensions */)
|
|
}
|
|
if (action === 'backToList') {
|
|
backToList()
|
|
}
|
|
if (action === 'escapeBack') {
|
|
if (state.modal) {
|
|
dispatch('exitModal')
|
|
} else if (state.window.isFullScreen) {
|
|
dispatch('toggleFullScreen')
|
|
} else {
|
|
dispatch('back')
|
|
}
|
|
}
|
|
if (action === 'back') {
|
|
state.location.back()
|
|
}
|
|
if (action === 'forward') {
|
|
state.location.forward()
|
|
}
|
|
if (action === 'playPause') {
|
|
playPause()
|
|
}
|
|
if (action === 'play') {
|
|
playFile(args[0] /* infoHash */, args[1] /* index */)
|
|
}
|
|
if (action === 'playbackJump') {
|
|
jumpToTime(args[0] /* seconds */)
|
|
}
|
|
if (action === 'skip') {
|
|
jumpToTime(state.playing.currentTime + (args[0] /* direction */ * 10))
|
|
}
|
|
if (action === 'changePlaybackRate') {
|
|
changePlaybackRate(args[0] /* direction */)
|
|
}
|
|
if (action === 'changeVolume') {
|
|
changeVolume(args[0] /* increase */)
|
|
}
|
|
if (action === 'setVolume') {
|
|
setVolume(args[0] /* increase */)
|
|
}
|
|
if (action === 'openSubtitles') {
|
|
openSubtitles()
|
|
}
|
|
if (action === 'selectSubtitle') {
|
|
selectSubtitle(args[0] /* index */)
|
|
}
|
|
if (action === 'toggleSubtitlesMenu') {
|
|
toggleSubtitlesMenu()
|
|
}
|
|
if (action === 'mediaStalled') {
|
|
state.playing.isStalled = true
|
|
}
|
|
if (action === 'mediaError') {
|
|
if (state.location.url() === 'player') {
|
|
state.playing.location = 'error'
|
|
ipcRenderer.send('checkForVLC')
|
|
ipcRenderer.once('checkForVLC', function (e, isInstalled) {
|
|
state.modal = {
|
|
id: 'unsupported-media-modal',
|
|
error: args[0],
|
|
vlcInstalled: isInstalled
|
|
}
|
|
})
|
|
}
|
|
}
|
|
if (action === 'mediaTimeUpdate') {
|
|
state.playing.lastTimeUpdate = new Date().getTime()
|
|
state.playing.isStalled = false
|
|
}
|
|
if (action === 'mediaMouseMoved') {
|
|
state.playing.mouseStationarySince = new Date().getTime()
|
|
}
|
|
if (action === 'vlcPlay') {
|
|
ipcRenderer.send('vlcPlay', state.server.localURL)
|
|
state.playing.location = 'vlc'
|
|
}
|
|
if (action === 'vlcNotFound') {
|
|
if (state.modal && state.modal.id === 'unsupported-media-modal') {
|
|
state.modal.vlcNotFound = true
|
|
}
|
|
}
|
|
if (action === 'toggleFullScreen') {
|
|
ipcRenderer.send('toggleFullScreen', args[0] /* optional bool */)
|
|
}
|
|
if (action === 'exitModal') {
|
|
state.modal = null
|
|
}
|
|
if (action === 'preferences') {
|
|
state.location.go({
|
|
url: 'preferences',
|
|
onbeforeload: function (cb) {
|
|
// initialize preferences
|
|
state.window.title = 'Preferences'
|
|
state.unsaved = Object.assign(state.unsaved || {}, {prefs: state.saved.prefs || {}})
|
|
cb()
|
|
},
|
|
onbeforeunload: function (cb) {
|
|
// save state after preferences
|
|
savePreferences()
|
|
state.window.title = config.APP_WINDOW_TITLE
|
|
cb()
|
|
}
|
|
})
|
|
}
|
|
if (action === 'updatePreferences') {
|
|
updatePreferences(args[0], args[1] /* property, value */)
|
|
}
|
|
if (action === 'updateAvailable') {
|
|
updateAvailable(args[0] /* version */)
|
|
}
|
|
if (action === 'skipVersion') {
|
|
if (!state.saved.skippedVersions) state.saved.skippedVersions = []
|
|
state.saved.skippedVersions.push(args[0] /* version */)
|
|
saveStateThrottled()
|
|
}
|
|
if (action === 'saveState') {
|
|
State.save(state)
|
|
}
|
|
if (action === 'setTitle') {
|
|
state.window.title = args[0] /* title */
|
|
}
|
|
|
|
// Update the virtual-dom, unless it's just a mouse move event
|
|
if (action !== 'mediaMouseMoved' || showOrHidePlayerControls()) {
|
|
update()
|
|
}
|
|
}
|
|
|
|
// Shows a modal saying that we have an update
|
|
function updateAvailable (version) {
|
|
if (state.saved.skippedVersions && state.saved.skippedVersions.includes(version)) {
|
|
console.log('new version skipped by user: v' + version)
|
|
return
|
|
}
|
|
state.modal = { id: 'update-available-modal', version: version }
|
|
}
|
|
|
|
function play () {
|
|
if (!state.playing.isPaused) return
|
|
state.playing.isPaused = false
|
|
if (isCasting()) {
|
|
Cast.play()
|
|
}
|
|
ipcRenderer.send('blockPowerSave')
|
|
}
|
|
|
|
function pause () {
|
|
if (state.playing.isPaused) return
|
|
state.playing.isPaused = true
|
|
if (isCasting()) {
|
|
Cast.pause()
|
|
}
|
|
ipcRenderer.send('unblockPowerSave')
|
|
}
|
|
|
|
function playPause () {
|
|
if (state.location.url() !== 'player') return
|
|
if (state.playing.isPaused) {
|
|
play()
|
|
} else {
|
|
pause()
|
|
}
|
|
}
|
|
|
|
function jumpToTime (time) {
|
|
if (isCasting()) {
|
|
Cast.seek(time)
|
|
} else {
|
|
state.playing.jumpToTime = time
|
|
}
|
|
}
|
|
function changePlaybackRate (direction) {
|
|
var rate = state.playing.playbackRate
|
|
if (direction > 0 && rate >= 0.25 && rate < 2) {
|
|
rate += 0.25
|
|
} else if (direction < 0 && rate > 0.25 && rate <= 2) {
|
|
rate -= 0.25
|
|
} else if (direction < 0 && rate === 0.25) { /* when we set playback rate at 0 in html 5, playback hangs ;( */
|
|
rate = -1
|
|
} else if (direction > 0 && rate === -1) {
|
|
rate = 0.25
|
|
} else if ((direction > 0 && rate >= 1 && rate < 16) || (direction < 0 && rate > -16 && rate <= -1)) {
|
|
rate *= 2
|
|
} else if ((direction < 0 && rate > 1 && rate <= 16) || (direction > 0 && rate >= -16 && rate < -1)) {
|
|
rate /= 2
|
|
}
|
|
state.playing.playbackRate = rate
|
|
if (lazyLoadCast().isCasting() && !Cast.setRate(rate)) {
|
|
state.playing.playbackRate = 1
|
|
}
|
|
}
|
|
function changeVolume (delta) {
|
|
// change volume with delta value
|
|
setVolume(state.playing.volume + delta)
|
|
}
|
|
|
|
function setVolume (volume) {
|
|
// check if its in [0.0 - 1.0] range
|
|
volume = Math.max(0, Math.min(1, volume))
|
|
if (isCasting()) {
|
|
Cast.setVolume(volume)
|
|
} else {
|
|
state.playing.setVolume = volume
|
|
}
|
|
}
|
|
|
|
function openSubtitles () {
|
|
electron.remote.dialog.showOpenDialog({
|
|
title: 'Select a subtitles file.',
|
|
filters: [ { name: 'Subtitles', extensions: ['vtt', 'srt'] } ],
|
|
properties: [ 'openFile' ]
|
|
}, function (filenames) {
|
|
if (!Array.isArray(filenames)) return
|
|
addSubtitles(filenames, true)
|
|
})
|
|
}
|
|
|
|
// Quits any modal popovers and returns to the torrent list screen
|
|
function backToList () {
|
|
// Exit any modals and screens with a back button
|
|
state.modal = null
|
|
state.location.backToFirst(function () {
|
|
// If we were already on the torrent list, scroll to the top
|
|
var contentTag = document.querySelector('.content')
|
|
if (contentTag) contentTag.scrollTop = 0
|
|
|
|
// Work around virtual-dom issue: it doesn't expose its redraw function,
|
|
// and only redraws on requestAnimationFrame(). That means when the user
|
|
// closes the window (hide window / minimize to tray) and we want to pause
|
|
// the video, we update the vdom but it keeps playing until you reopen!
|
|
var mediaTag = document.querySelector('video,audio')
|
|
if (mediaTag) mediaTag.pause()
|
|
})
|
|
}
|
|
|
|
// Checks whether we are connected and already casting
|
|
// Returns false if we not casting (state.playing.location === 'local')
|
|
// or if we're trying to connect but haven't yet ('chromecast-pending', etc)
|
|
function isCasting () {
|
|
return state.playing.location === 'chromecast' ||
|
|
state.playing.location === 'airplay' ||
|
|
state.playing.location === 'dlna'
|
|
}
|
|
|
|
function setupIpc () {
|
|
ipcRenderer.on('log', (e, ...args) => console.log(...args))
|
|
ipcRenderer.on('error', (e, ...args) => console.error(...args))
|
|
|
|
ipcRenderer.on('dispatch', (e, ...args) => dispatch(...args))
|
|
|
|
ipcRenderer.on('fullscreenChanged', function (e, isFullScreen) {
|
|
state.window.isFullScreen = isFullScreen
|
|
if (!isFullScreen) {
|
|
// Aspect ratio gets reset in fullscreen mode, so restore it (OS X)
|
|
ipcRenderer.send('setAspectRatio', state.playing.aspectRatio)
|
|
}
|
|
update()
|
|
})
|
|
|
|
ipcRenderer.on('wt-infohash', (e, ...args) => torrentInfoHash(...args))
|
|
ipcRenderer.on('wt-metadata', (e, ...args) => torrentMetadata(...args))
|
|
ipcRenderer.on('wt-done', (e, ...args) => torrentDone(...args))
|
|
ipcRenderer.on('wt-warning', (e, ...args) => torrentWarning(...args))
|
|
ipcRenderer.on('wt-error', (e, ...args) => torrentError(...args))
|
|
|
|
ipcRenderer.on('wt-progress', (e, ...args) => torrentProgress(...args))
|
|
ipcRenderer.on('wt-file-modtimes', (e, ...args) => torrentFileModtimes(...args))
|
|
ipcRenderer.on('wt-file-saved', (e, ...args) => torrentFileSaved(...args))
|
|
ipcRenderer.on('wt-poster', (e, ...args) => torrentPosterSaved(...args))
|
|
ipcRenderer.on('wt-audio-metadata', (e, ...args) => torrentAudioMetadata(...args))
|
|
ipcRenderer.on('wt-server-running', (e, ...args) => torrentServerRunning(...args))
|
|
|
|
ipcRenderer.send('ipcReady')
|
|
}
|
|
|
|
// Starts all torrents that aren't paused on program startup
|
|
function resumeTorrents () {
|
|
state.saved.torrents
|
|
.filter((torrentSummary) => torrentSummary.status !== 'paused')
|
|
.forEach((torrentSummary) => startTorrentingSummary(torrentSummary))
|
|
}
|
|
|
|
// Updates a single property in the UNSAVED prefs
|
|
// For example: updatePreferences("foo.bar", "baz")
|
|
// Call savePreferences to save to config.json
|
|
function updatePreferences (property, value) {
|
|
var path = property.split('.')
|
|
var key = state.unsaved.prefs
|
|
for (var i = 0; i < path.length - 1; i++) {
|
|
if (typeof key[path[i]] === 'undefined') {
|
|
key[path[i]] = {}
|
|
}
|
|
key = key[path[i]]
|
|
}
|
|
key[path[i]] = value
|
|
}
|
|
|
|
// All unsaved prefs take effect atomically, and are saved to config.json
|
|
function savePreferences () {
|
|
state.saved.prefs = Object.assign(state.saved.prefs || {}, state.unsaved.prefs)
|
|
State.save(state)
|
|
update()
|
|
}
|
|
|
|
// Don't write state.saved to file more than once a second
|
|
function saveStateThrottled () {
|
|
if (state.saveStateTimeout) return
|
|
state.saveStateTimeout = setTimeout(function () {
|
|
delete state.saveStateTimeout
|
|
State.save(state)
|
|
update()
|
|
}, 1000)
|
|
}
|
|
|
|
// Called when the user adds files (.torrent, files to seed, subtitles) to the app
|
|
// via any method (drag-drop, drag to app icon, command line)
|
|
function onOpen (files) {
|
|
if (!Array.isArray(files)) files = [ files ]
|
|
|
|
if (state.modal) {
|
|
state.modal = null
|
|
}
|
|
|
|
var subtitles = files.filter(isSubtitle)
|
|
|
|
if (state.location.url() === 'home' || subtitles.length === 0) {
|
|
if (files.every(isTorrent)) {
|
|
if (state.location.url() !== 'home') {
|
|
backToList()
|
|
}
|
|
// All .torrent files? Add them.
|
|
files.forEach(addTorrent)
|
|
} else {
|
|
// Show the Create Torrent screen. Let's seed those files.
|
|
showCreateTorrent(files)
|
|
}
|
|
} else if (state.location.url() === 'player') {
|
|
addSubtitles(subtitles, true)
|
|
}
|
|
|
|
update()
|
|
}
|
|
|
|
function isTorrent (file) {
|
|
var name = typeof file === 'string' ? file : file.name
|
|
var isTorrentFile = path.extname(name).toLowerCase() === '.torrent'
|
|
var isMagnet = typeof file === 'string' && /^magnet:/.test(file)
|
|
return isTorrentFile || isMagnet
|
|
}
|
|
|
|
function isSubtitle (file) {
|
|
var name = typeof file === 'string' ? file : file.name
|
|
var ext = path.extname(name).toLowerCase()
|
|
return ext === '.srt' || ext === '.vtt'
|
|
}
|
|
|
|
// Gets a torrent summary {name, infoHash, status} from state.saved.torrents
|
|
// Returns undefined if we don't know that infoHash
|
|
function getTorrentSummary (torrentKey) {
|
|
if (!torrentKey) return undefined
|
|
return state.saved.torrents.find((x) =>
|
|
x.torrentKey === torrentKey || x.infoHash === torrentKey)
|
|
}
|
|
|
|
// Adds a torrent to the list, starts downloading/seeding. TorrentID can be a
|
|
// magnet URI, infohash, or torrent file: https://github.com/feross/webtorrent#clientaddtorrentid-opts-function-ontorrent-torrent-
|
|
var instantIoRegex = /^(https:\/\/)?instant\.io\/#/
|
|
function addTorrent (torrentId) {
|
|
backToList()
|
|
var torrentKey = state.nextTorrentKey++
|
|
var path = state.saved.prefs.downloadPath
|
|
if (torrentId.path) {
|
|
// Use path string instead of W3C File object
|
|
torrentId = torrentId.path
|
|
}
|
|
// Allow a instant.io link to be pasted
|
|
// TODO: remove this once support is added to webtorrent core
|
|
if (typeof torrentId === 'string' && instantIoRegex.test(torrentId)) {
|
|
torrentId = torrentId.slice(torrentId.indexOf('#') + 1)
|
|
}
|
|
ipcRenderer.send('wt-start-torrenting', torrentKey, torrentId, path)
|
|
}
|
|
|
|
function addSubtitles (files, autoSelect) {
|
|
// Subtitles are only supported when playing video files
|
|
if (state.playing.type !== 'video') return
|
|
if (files.length === 0) return
|
|
|
|
// Read the files concurrently, then add all resulting subtitle tracks
|
|
var tasks = files.map((file) => (cb) => loadSubtitle(file, cb))
|
|
parallel(tasks, function (err, tracks) {
|
|
if (err) return onError(err)
|
|
|
|
for (var i = 0; i < tracks.length; i++) {
|
|
// No dupes allowed
|
|
var track = tracks[i]
|
|
if (state.playing.subtitles.tracks.some(
|
|
(t) => track.filePath === t.filePath)) continue
|
|
|
|
// Add the track
|
|
state.playing.subtitles.tracks.push(track)
|
|
|
|
// If we're auto-selecting a track, try to find one in the user's language
|
|
if (autoSelect && (i === 0 || isSystemLanguage(track.language))) {
|
|
state.playing.subtitles.selectedIndex =
|
|
state.playing.subtitles.tracks.length - 1
|
|
}
|
|
}
|
|
|
|
// Finally, make sure no two tracks have the same label
|
|
relabelSubtitles()
|
|
})
|
|
}
|
|
|
|
function loadSubtitle (file, cb) {
|
|
var concat = require('simple-concat')
|
|
var LanguageDetect = require('languagedetect')
|
|
var srtToVtt = require('srt-to-vtt')
|
|
|
|
// Read the .SRT or .VTT file, parse it, add subtitle track
|
|
var filePath = file.path || file
|
|
|
|
var vttStream = fs.createReadStream(filePath).pipe(srtToVtt())
|
|
|
|
concat(vttStream, function (err, buf) {
|
|
if (err) return onError(new Error('Error parsing subtitles file.'))
|
|
|
|
// Detect what language the subtitles are in
|
|
var vttContents = buf.toString().replace(/(.*-->.*)/g, '')
|
|
var langDetected = (new LanguageDetect()).detect(vttContents, 2)
|
|
langDetected = langDetected.length ? langDetected[0][0] : 'subtitle'
|
|
langDetected = langDetected.slice(0, 1).toUpperCase() + langDetected.slice(1)
|
|
|
|
var track = {
|
|
buffer: 'data:text/vtt;base64,' + buf.toString('base64'),
|
|
language: langDetected,
|
|
label: langDetected,
|
|
filePath: filePath
|
|
}
|
|
|
|
cb(null, track)
|
|
})
|
|
}
|
|
|
|
function selectSubtitle (ix) {
|
|
state.playing.subtitles.selectedIndex = ix
|
|
}
|
|
|
|
// Checks whether a language name like "English" or "German" matches the system
|
|
// language, aka the current locale
|
|
function isSystemLanguage (language) {
|
|
var iso639 = require('iso-639-1')
|
|
var osLangISO = window.navigator.language.split('-')[0] // eg "en"
|
|
var langIso = iso639.getCode(language) // eg "de" if language is "German"
|
|
return langIso === osLangISO
|
|
}
|
|
|
|
// Make sure we don't have two subtitle tracks with the same label
|
|
// Labels each track by language, eg "German", "English", "English 2", ...
|
|
function relabelSubtitles () {
|
|
var counts = {}
|
|
state.playing.subtitles.tracks.forEach(function (track) {
|
|
var lang = track.language
|
|
counts[lang] = (counts[lang] || 0) + 1
|
|
track.label = counts[lang] > 1 ? (lang + ' ' + counts[lang]) : lang
|
|
})
|
|
}
|
|
|
|
function checkForSubtitles () {
|
|
if (state.playing.type !== 'video') return
|
|
var torrentSummary = state.getPlayingTorrentSummary()
|
|
if (!torrentSummary || !torrentSummary.progress) return
|
|
|
|
torrentSummary.progress.files.forEach(function (fp, ix) {
|
|
if (fp.numPieces !== fp.numPiecesPresent) return // ignore incomplete files
|
|
var file = torrentSummary.files[ix]
|
|
if (!isSubtitle(file.name)) return
|
|
var filePath = path.join(torrentSummary.path, file.path)
|
|
addSubtitles([filePath], false)
|
|
})
|
|
}
|
|
|
|
function toggleSubtitlesMenu () {
|
|
state.playing.subtitles.showMenu = !state.playing.subtitles.showMenu
|
|
}
|
|
|
|
// Starts downloading and/or seeding a given torrentSummary. Returns WebTorrent object
|
|
function startTorrentingSummary (torrentSummary) {
|
|
var s = torrentSummary
|
|
|
|
// Backward compatibility for config files save before we had torrentKey
|
|
if (!s.torrentKey) s.torrentKey = state.nextTorrentKey++
|
|
|
|
// Use Downloads folder by default
|
|
var path = s.path || state.saved.prefs.downloadPath
|
|
|
|
var torrentID
|
|
if (s.torrentFileName) { // Load torrent file from disk
|
|
torrentID = TorrentSummary.getTorrentPath(torrentSummary)
|
|
} else { // Load torrent from DHT
|
|
torrentID = s.magnetURI || s.infoHash
|
|
}
|
|
|
|
console.log('start torrenting %s %s', s.torrentKey, torrentID)
|
|
ipcRenderer.send('wt-start-torrenting', s.torrentKey, torrentID, path, s.fileModtimes, s.selections)
|
|
}
|
|
|
|
//
|
|
// TORRENT MANAGEMENT
|
|
// Send commands to the WebTorrent process, handle events
|
|
//
|
|
|
|
// Shows the Create Torrent page with options to seed a given file or folder
|
|
function showCreateTorrent (files) {
|
|
// Files will either be an array of file objects, which we can send directly
|
|
// to the create-torrent screen
|
|
if (files.length === 0 || typeof files[0] !== 'string') {
|
|
state.location.go({
|
|
url: 'create-torrent',
|
|
files: files
|
|
})
|
|
return
|
|
}
|
|
|
|
// ... or it will be an array of mixed file and folder paths. We have to walk
|
|
// through all the folders and find the files
|
|
findFilesRecursive(files, showCreateTorrent)
|
|
}
|
|
|
|
// Recursively finds {name, path, size} for all files in a folder
|
|
// Calls `cb` on success, calls `onError` on failure
|
|
function findFilesRecursive (paths, cb) {
|
|
if (paths.length > 1) {
|
|
var numComplete = 0
|
|
var ret = []
|
|
paths.forEach(function (path) {
|
|
findFilesRecursive([path], function (fileObjs) {
|
|
ret = ret.concat(fileObjs)
|
|
if (++numComplete === paths.length) {
|
|
cb(ret)
|
|
}
|
|
})
|
|
})
|
|
return
|
|
}
|
|
|
|
var fileOrFolder = paths[0]
|
|
fs.stat(fileOrFolder, function (err, stat) {
|
|
if (err) return onError(err)
|
|
|
|
// Files: return name, path, and size
|
|
if (!stat.isDirectory()) {
|
|
var filePath = fileOrFolder
|
|
return cb([{
|
|
name: path.basename(filePath),
|
|
path: filePath,
|
|
size: stat.size
|
|
}])
|
|
}
|
|
|
|
// Folders: recurse, make a list of all the files
|
|
var folderPath = fileOrFolder
|
|
fs.readdir(folderPath, function (err, fileNames) {
|
|
if (err) return onError(err)
|
|
var paths = fileNames.map((fileName) => path.join(folderPath, fileName))
|
|
findFilesRecursive(paths, cb)
|
|
})
|
|
})
|
|
}
|
|
|
|
// Creates a new torrent and start seeeding
|
|
function createTorrent (options) {
|
|
var torrentKey = state.nextTorrentKey++
|
|
ipcRenderer.send('wt-create-torrent', torrentKey, options)
|
|
state.location.backToFirst(function () {
|
|
state.location.clearForward('create-torrent')
|
|
})
|
|
}
|
|
|
|
function torrentInfoHash (torrentKey, infoHash) {
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
console.log('got infohash for %s torrent %s',
|
|
torrentSummary ? 'existing' : 'new', torrentKey)
|
|
|
|
if (!torrentSummary) {
|
|
// Check if an existing (non-active) torrent has the same info hash
|
|
if (state.saved.torrents.find((t) => t.infoHash === infoHash)) {
|
|
ipcRenderer.send('wt-stop-torrenting', infoHash)
|
|
return onError(new Error('Cannot add duplicate torrent'))
|
|
}
|
|
|
|
torrentSummary = {
|
|
torrentKey: torrentKey,
|
|
status: 'new'
|
|
}
|
|
state.saved.torrents.unshift(torrentSummary)
|
|
sound.play('ADD')
|
|
}
|
|
|
|
torrentSummary.infoHash = infoHash
|
|
update()
|
|
}
|
|
|
|
function torrentWarning (torrentKey, message) {
|
|
onWarning(message)
|
|
}
|
|
|
|
function torrentError (torrentKey, message) {
|
|
// TODO: WebTorrent needs semantic errors
|
|
if (message.startsWith('Cannot add duplicate torrent')) {
|
|
// Remove infohash from the message
|
|
message = 'Cannot add duplicate torrent'
|
|
}
|
|
onError(message)
|
|
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
if (torrentSummary) {
|
|
console.log('Pausing torrent %s due to error: %s', torrentSummary.infoHash, message)
|
|
torrentSummary.status = 'paused'
|
|
update()
|
|
}
|
|
}
|
|
|
|
function torrentMetadata (torrentKey, torrentInfo) {
|
|
// Summarize torrent
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
torrentSummary.status = 'downloading'
|
|
torrentSummary.name = torrentSummary.displayName || torrentInfo.name
|
|
torrentSummary.path = torrentInfo.path
|
|
torrentSummary.magnetURI = torrentInfo.magnetURI
|
|
// TODO: make torrentInfo immutable, save separately as torrentSummary.info
|
|
// For now, check whether torrentSummary.files has already been set:
|
|
var hasDetailedFileInfo = torrentSummary.files && torrentSummary.files[0].path
|
|
if (!hasDetailedFileInfo) {
|
|
torrentSummary.files = torrentInfo.files
|
|
}
|
|
if (!torrentSummary.selections) {
|
|
torrentSummary.selections = torrentSummary.files.map((x) => true)
|
|
}
|
|
torrentSummary.defaultPlayFileIndex = pickFileToPlay(torrentInfo.files)
|
|
update()
|
|
|
|
// Save the .torrent file, if it hasn't been saved already
|
|
if (!torrentSummary.torrentFileName) ipcRenderer.send('wt-save-torrent-file', torrentKey)
|
|
|
|
// Auto-generate a poster image, if it hasn't been generated already
|
|
if (!torrentSummary.posterFileName) ipcRenderer.send('wt-generate-torrent-poster', torrentKey)
|
|
}
|
|
|
|
function torrentDone (torrentKey, torrentInfo) {
|
|
// Update the torrent summary
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
torrentSummary.status = 'seeding'
|
|
|
|
// Notify the user that a torrent finished, but only if we actually DL'd at least part of it.
|
|
// Don't notify if we merely finished verifying data files that were already on disk.
|
|
if (torrentInfo.bytesReceived > 0) {
|
|
if (!state.window.isFocused) {
|
|
state.dock.badge += 1
|
|
}
|
|
showDoneNotification(torrentSummary)
|
|
ipcRenderer.send('downloadFinished', getTorrentPath(torrentSummary))
|
|
}
|
|
|
|
update()
|
|
}
|
|
|
|
function torrentProgress (progressInfo) {
|
|
// Overall progress across all active torrents, 0 to 1
|
|
var progress = progressInfo.progress
|
|
var hasActiveTorrents = progressInfo.hasActiveTorrents
|
|
|
|
// Hide progress bar when client has no torrents, or progress is 100%
|
|
// TODO: isn't this equivalent to: if (progress === 1) ?
|
|
if (!hasActiveTorrents || progress === 1) {
|
|
progress = -1
|
|
}
|
|
|
|
// Show progress bar under the WebTorrent taskbar icon, on OSX
|
|
state.dock.progress = progress
|
|
|
|
// Update progress for each individual torrent
|
|
progressInfo.torrents.forEach(function (p) {
|
|
var torrentSummary = getTorrentSummary(p.torrentKey)
|
|
if (!torrentSummary) {
|
|
console.log('warning: got progress for missing torrent %s', p.torrentKey)
|
|
return
|
|
}
|
|
torrentSummary.progress = p
|
|
})
|
|
|
|
// TODO: Find an efficient way to re-enable this line, which allows subtitle
|
|
// files which are completed after a video starts to play to be added
|
|
// dynamically to the list of subtitles.
|
|
// checkForSubtitles()
|
|
|
|
update()
|
|
}
|
|
|
|
function torrentFileModtimes (torrentKey, fileModtimes) {
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
torrentSummary.fileModtimes = fileModtimes
|
|
saveStateThrottled()
|
|
}
|
|
|
|
function torrentFileSaved (torrentKey, torrentFileName) {
|
|
console.log('torrent file saved %s: %s', torrentKey, torrentFileName)
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
torrentSummary.torrentFileName = torrentFileName
|
|
saveStateThrottled()
|
|
}
|
|
|
|
function torrentPosterSaved (torrentKey, posterFileName) {
|
|
var torrentSummary = getTorrentSummary(torrentKey)
|
|
torrentSummary.posterFileName = posterFileName
|
|
saveStateThrottled()
|
|
}
|
|
|
|
function torrentAudioMetadata (infoHash, index, info) {
|
|
var torrentSummary = getTorrentSummary(infoHash)
|
|
var fileSummary = torrentSummary.files[index]
|
|
fileSummary.audioInfo = info
|
|
update()
|
|
}
|
|
|
|
function torrentServerRunning (serverInfo) {
|
|
state.server = serverInfo
|
|
}
|
|
|
|
// Picks the default file to play from a list of torrent or torrentSummary files
|
|
// Returns an index or undefined, if no files are playable
|
|
function pickFileToPlay (files) {
|
|
// first, try to find the biggest video file
|
|
var videoFiles = files.filter(TorrentPlayer.isVideo)
|
|
if (videoFiles.length > 0) {
|
|
var largestVideoFile = videoFiles.reduce(function (a, b) {
|
|
return a.length > b.length ? a : b
|
|
})
|
|
return files.indexOf(largestVideoFile)
|
|
}
|
|
|
|
// if there are no videos, play the first audio file
|
|
var audioFiles = files.filter(TorrentPlayer.isAudio)
|
|
if (audioFiles.length > 0) {
|
|
return files.indexOf(audioFiles[0])
|
|
}
|
|
|
|
// no video or audio means nothing is playable
|
|
return undefined
|
|
}
|
|
|
|
function playFile (infoHash, index) {
|
|
state.location.go({
|
|
url: 'player',
|
|
onbeforeload: function (cb) {
|
|
play()
|
|
openPlayer(infoHash, index, cb)
|
|
},
|
|
onbeforeunload: closePlayer
|
|
}, function (err) {
|
|
if (err) onError(err)
|
|
})
|
|
}
|
|
|
|
// Opens the video player to a specific torrent
|
|
function openPlayer (infoHash, index, cb) {
|
|
var torrentSummary = getTorrentSummary(infoHash)
|
|
|
|
// automatically choose which file in the torrent to play, if necessary
|
|
if (index === undefined) index = torrentSummary.defaultPlayFileIndex
|
|
if (index === undefined) index = pickFileToPlay(torrentSummary.files)
|
|
if (index === undefined) return cb(new errors.UnplayableError())
|
|
|
|
// update UI to show pending playback
|
|
if (torrentSummary.progress !== 1) sound.play('PLAY')
|
|
torrentSummary.playStatus = 'requested'
|
|
update()
|
|
|
|
var timeout = setTimeout(function () {
|
|
torrentSummary.playStatus = 'timeout' /* no seeders available? */
|
|
sound.play('ERROR')
|
|
cb(new Error('Playback timed out. Try again.'))
|
|
update()
|
|
}, 10000) /* give it a few seconds */
|
|
|
|
if (torrentSummary.status === 'paused') {
|
|
startTorrentingSummary(torrentSummary)
|
|
ipcRenderer.once('wt-ready-' + torrentSummary.infoHash,
|
|
() => openPlayerFromActiveTorrent(torrentSummary, index, timeout, cb))
|
|
} else {
|
|
openPlayerFromActiveTorrent(torrentSummary, index, timeout, cb)
|
|
}
|
|
}
|
|
|
|
function openPlayerFromActiveTorrent (torrentSummary, index, timeout, cb) {
|
|
var fileSummary = torrentSummary.files[index]
|
|
|
|
// update state
|
|
state.playing.infoHash = torrentSummary.infoHash
|
|
state.playing.fileIndex = index
|
|
state.playing.type = TorrentPlayer.isVideo(fileSummary) ? 'video'
|
|
: TorrentPlayer.isAudio(fileSummary) ? 'audio'
|
|
: 'other'
|
|
|
|
// pick up where we left off
|
|
if (fileSummary.currentTime) {
|
|
var fraction = fileSummary.currentTime / fileSummary.duration
|
|
var secondsLeft = fileSummary.duration - fileSummary.currentTime
|
|
if (fraction < 0.9 && secondsLeft > 10) {
|
|
state.playing.jumpToTime = fileSummary.currentTime
|
|
}
|
|
}
|
|
|
|
// if it's audio, parse out the metadata (artist, title, etc)
|
|
if (state.playing.type === 'audio' && !fileSummary.audioInfo) {
|
|
ipcRenderer.send('wt-get-audio-metadata', torrentSummary.infoHash, index)
|
|
}
|
|
|
|
// if it's video, check for subtitles files that are done downloading
|
|
checkForSubtitles()
|
|
|
|
ipcRenderer.send('wt-start-server', torrentSummary.infoHash, index)
|
|
ipcRenderer.once('wt-server-' + torrentSummary.infoHash, function (e, info) {
|
|
clearTimeout(timeout)
|
|
|
|
// if we timed out (user clicked play a long time ago), don't autoplay
|
|
var timedOut = torrentSummary.playStatus === 'timeout'
|
|
delete torrentSummary.playStatus
|
|
if (timedOut) {
|
|
ipcRenderer.send('wt-stop-server')
|
|
return update()
|
|
}
|
|
|
|
// otherwise, play the video
|
|
state.window.title = torrentSummary.files[state.playing.fileIndex].name
|
|
update()
|
|
|
|
ipcRenderer.send('onPlayerOpen')
|
|
|
|
cb()
|
|
})
|
|
}
|
|
|
|
function closePlayer (cb) {
|
|
if (isCasting()) {
|
|
Cast.close()
|
|
}
|
|
if (state.playing.location === 'vlc') {
|
|
ipcRenderer.send('vlcQuit')
|
|
}
|
|
state.window.title = config.APP_WINDOW_TITLE
|
|
// Lets save volume for later
|
|
state.previousVolume = state.playing.volume
|
|
|
|
state.playing = State.getDefaultPlayState()
|
|
state.server = null
|
|
|
|
if (state.window.isFullScreen) {
|
|
dispatch('toggleFullScreen', false)
|
|
}
|
|
restoreBounds()
|
|
|
|
ipcRenderer.send('wt-stop-server')
|
|
ipcRenderer.send('unblockPowerSave')
|
|
ipcRenderer.send('onPlayerClose')
|
|
|
|
update()
|
|
cb()
|
|
}
|
|
|
|
function openItem (infoHash, index) {
|
|
var torrentSummary = getTorrentSummary(infoHash)
|
|
var filePath = path.join(
|
|
torrentSummary.path,
|
|
torrentSummary.files[index].path)
|
|
ipcRenderer.send('openItem', filePath)
|
|
}
|
|
|
|
// TODO: use torrentKey, not infoHash
|
|
function toggleTorrent (infoHash) {
|
|
var torrentSummary = getTorrentSummary(infoHash)
|
|
if (torrentSummary.status === 'paused') {
|
|
torrentSummary.status = 'new'
|
|
startTorrentingSummary(torrentSummary)
|
|
sound.play('ENABLE')
|
|
} else {
|
|
torrentSummary.status = 'paused'
|
|
ipcRenderer.send('wt-stop-torrenting', torrentSummary.infoHash)
|
|
sound.play('DISABLE')
|
|
}
|
|
}
|
|
|
|
// TODO: use torrentKey, not infoHash
|
|
function deleteTorrent (infoHash) {
|
|
ipcRenderer.send('wt-stop-torrenting', infoHash)
|
|
|
|
var index = state.saved.torrents.findIndex((x) => x.infoHash === infoHash)
|
|
if (index > -1) state.saved.torrents.splice(index, 1)
|
|
saveStateThrottled()
|
|
state.location.clearForward('player') // prevent user from going forward to a deleted torrent
|
|
sound.play('DELETE')
|
|
}
|
|
|
|
function toggleSelectTorrent (infoHash) {
|
|
// toggle selection
|
|
state.selectedInfoHash = state.selectedInfoHash === infoHash ? null : infoHash
|
|
update()
|
|
}
|
|
|
|
function toggleTorrentFile (infoHash, index) {
|
|
var torrentSummary = getTorrentSummary(infoHash)
|
|
torrentSummary.selections[index] = !torrentSummary.selections[index]
|
|
|
|
// Let the WebTorrent process know to start or stop fetching that file
|
|
ipcRenderer.send('wt-select-files', infoHash, torrentSummary.selections)
|
|
}
|
|
|
|
function openTorrentContextMenu (infoHash) {
|
|
var torrentSummary = getTorrentSummary(infoHash)
|
|
var menu = new electron.remote.Menu()
|
|
|
|
if (torrentSummary.files) {
|
|
menu.append(new electron.remote.MenuItem({
|
|
label: process.platform === 'darwin' ? 'Show in Finder' : 'Show in Folder',
|
|
click: () => showItemInFolder(torrentSummary)
|
|
}))
|
|
menu.append(new electron.remote.MenuItem({
|
|
type: 'separator'
|
|
}))
|
|
}
|
|
|
|
menu.append(new electron.remote.MenuItem({
|
|
label: 'Copy Magnet Link to Clipboard',
|
|
click: () => electron.clipboard.writeText(torrentSummary.magnetURI)
|
|
}))
|
|
|
|
menu.append(new electron.remote.MenuItem({
|
|
label: 'Copy Instant.io Link to Clipboard',
|
|
click: () => electron.clipboard.writeText(`https://instant.io/#${torrentSummary.infoHash}`)
|
|
}))
|
|
|
|
menu.append(new electron.remote.MenuItem({
|
|
label: 'Save Torrent File As...',
|
|
click: () => saveTorrentFileAs(torrentSummary)
|
|
}))
|
|
|
|
menu.popup(electron.remote.getCurrentWindow())
|
|
}
|
|
|
|
function getTorrentPath (torrentSummary) {
|
|
var itemPath = path.join(torrentSummary.path, torrentSummary.files[0].path)
|
|
if (torrentSummary.files.length > 1) {
|
|
itemPath = path.dirname(itemPath)
|
|
}
|
|
return itemPath
|
|
}
|
|
|
|
function showItemInFolder (torrentSummary) {
|
|
ipcRenderer.send('showItemInFolder', getTorrentPath(torrentSummary))
|
|
}
|
|
|
|
function saveTorrentFileAs (torrentSummary) {
|
|
var newFileName = `${path.parse(torrentSummary.name).name}.torrent`
|
|
var opts = {
|
|
title: 'Save Torrent File',
|
|
defaultPath: path.join(state.saved.prefs.downloadPath, newFileName),
|
|
filters: [
|
|
{ name: 'Torrent Files', extensions: ['torrent'] },
|
|
{ name: 'All Files', extensions: ['*'] }
|
|
]
|
|
}
|
|
electron.remote.dialog.showSaveDialog(electron.remote.getCurrentWindow(), opts, function (savePath) {
|
|
var torrentPath = TorrentSummary.getTorrentPath(torrentSummary)
|
|
fs.readFile(torrentPath, function (err, torrentFile) {
|
|
if (err) return onError(err)
|
|
fs.writeFile(savePath, torrentFile, function (err) {
|
|
if (err) return onError(err)
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
// Set window dimensions to match video dimensions or fill the screen
|
|
function setDimensions (dimensions) {
|
|
// Don't modify the window size if it's already maximized
|
|
if (electron.remote.getCurrentWindow().isMaximized()) {
|
|
state.window.bounds = null
|
|
return
|
|
}
|
|
|
|
// Save the bounds of the window for later. See restoreBounds()
|
|
state.window.bounds = {
|
|
x: window.screenX,
|
|
y: window.screenY,
|
|
width: window.outerWidth,
|
|
height: window.outerHeight
|
|
}
|
|
state.window.wasMaximized = electron.remote.getCurrentWindow().isMaximized
|
|
|
|
// Limit window size to screen size
|
|
var screenWidth = window.screen.width
|
|
var screenHeight = window.screen.height
|
|
var aspectRatio = dimensions.width / dimensions.height
|
|
var scaleFactor = Math.min(
|
|
Math.min(screenWidth / dimensions.width, 1),
|
|
Math.min(screenHeight / dimensions.height, 1)
|
|
)
|
|
var width = Math.max(
|
|
Math.floor(dimensions.width * scaleFactor),
|
|
config.WINDOW_MIN_WIDTH
|
|
)
|
|
var height = Math.max(
|
|
Math.floor(dimensions.height * scaleFactor),
|
|
config.WINDOW_MIN_HEIGHT
|
|
)
|
|
|
|
ipcRenderer.send('setAspectRatio', aspectRatio)
|
|
ipcRenderer.send('setBounds', {x: null, y: null, width, height})
|
|
state.playing.aspectRatio = aspectRatio
|
|
}
|
|
|
|
function restoreBounds () {
|
|
ipcRenderer.send('setAspectRatio', 0)
|
|
if (state.window.bounds) {
|
|
ipcRenderer.send('setBounds', state.window.bounds, false)
|
|
}
|
|
}
|
|
|
|
function showDoneNotification (torrent) {
|
|
var notif = new window.Notification('Download Complete', {
|
|
body: torrent.name,
|
|
silent: true
|
|
})
|
|
|
|
notif.onclick = function () {
|
|
ipcRenderer.send('show')
|
|
}
|
|
|
|
sound.play('DONE')
|
|
}
|
|
|
|
// Hide player controls while playing video, if the mouse stays still for a while
|
|
// Never hide the controls when:
|
|
// * The mouse is over the controls or we're scrubbing (see CSS)
|
|
// * The video is paused
|
|
// * The video is playing remotely on Chromecast or Airplay
|
|
function showOrHidePlayerControls () {
|
|
var hideControls = state.location.url() === 'player' &&
|
|
state.playing.mouseStationarySince !== 0 &&
|
|
new Date().getTime() - state.playing.mouseStationarySince > 2000 &&
|
|
!state.playing.isPaused &&
|
|
state.playing.location === 'local'
|
|
|
|
if (hideControls !== state.playing.hideControls) {
|
|
state.playing.hideControls = hideControls
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Event handlers
|
|
function onError (err) {
|
|
console.error(err.stack || err)
|
|
sound.play('ERROR')
|
|
state.errors.push({
|
|
time: new Date().getTime(),
|
|
message: err.message || err
|
|
})
|
|
update()
|
|
}
|
|
|
|
function onWarning (err) {
|
|
console.log('warning: %s', err.message || err)
|
|
}
|
|
|
|
function onPaste (e) {
|
|
if (e.target.tagName.toLowerCase() === 'input') return
|
|
|
|
var torrentIds = electron.clipboard.readText().split('\n')
|
|
torrentIds.forEach(function (torrentId) {
|
|
torrentId = torrentId.trim()
|
|
if (torrentId.length === 0) return
|
|
addTorrent(torrentId)
|
|
})
|
|
|
|
update()
|
|
}
|
|
|
|
function onFocus (e) {
|
|
state.window.isFocused = true
|
|
state.dock.badge = 0
|
|
update()
|
|
}
|
|
|
|
function onBlur () {
|
|
state.window.isFocused = false
|
|
update()
|
|
}
|