Compare commits
21 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cfb3a01239 | ||
|
|
736d575ab1 | ||
|
|
34a9508483 | ||
|
|
21ed8797c2 | ||
|
|
454491572a | ||
|
|
6518a1535c | ||
|
|
0095687bf5 | ||
|
|
d466ed085a | ||
|
|
eeda7c17c5 | ||
|
|
b89deb46db | ||
|
|
951a89c6c9 | ||
|
|
d4e6c84279 | ||
|
|
9731d85ca3 | ||
|
|
98f7ba8931 | ||
|
|
24c775608e | ||
|
|
f4eab12c3f | ||
|
|
8eeddeb4bc | ||
|
|
58f1594d9e | ||
|
|
c126ac0a84 | ||
|
|
6768be710e | ||
|
|
b63aa090dc |
28
CHANGELOG.md
28
CHANGELOG.md
@@ -1,5 +1,24 @@
|
||||
# WebTorrent Desktop Version History
|
||||
|
||||
## v0.5.0 - 2016-05-17
|
||||
|
||||
### Added
|
||||
|
||||
- Select/deselect individual files to torrent.
|
||||
- Automatically include subtitle files (.srt, .vtt) from torrent in the subtitles menu.
|
||||
- "Add Subtitle File..." menu item.
|
||||
|
||||
### Changed
|
||||
|
||||
- When manually adding subtitle track(s), always switch to the new track.
|
||||
|
||||
### Fixed
|
||||
|
||||
- Magnet links throw exception on app launch. (OS X)
|
||||
- Multi-file torrents would not seed in-place, were copied to Downloads folder.
|
||||
- Missing 'About WebTorrent' menu item. (Windows)
|
||||
- Rare exception. ("Cannot create BrowserWindow before app is ready")
|
||||
|
||||
## v0.4.0 - 2016-05-13
|
||||
|
||||
### Added
|
||||
@@ -38,7 +57,8 @@
|
||||
|
||||
### Fixed
|
||||
|
||||
- Disable WebRTC to fix 100% CPU usage/crashes caused by Chromium issue. This is temporary. (OS X)
|
||||
- Disable WebRTC to fix 100% CPU usage/crashes caused by Chromium issue. This is
|
||||
temporary. (OS X)
|
||||
- When fullscreen, make controls use the full window. (OS X)
|
||||
- Support creating torrents that contain .torrent files.
|
||||
- Block power save while casting to a remote device.
|
||||
@@ -50,10 +70,14 @@
|
||||
- Do not stop music when tabbing to another program (OS X)
|
||||
- Properly size the Windows volume mixer icon.
|
||||
- Default to the user's OS-defined, localized "Downloads" folder.
|
||||
- Enforce minimimum window size when resizing player, to prevent window disappearing.
|
||||
- Enforce minimimum window size when resizing player to prevent window disappearing.
|
||||
- Fix rare race condition error on app quit.
|
||||
- Don't use zero-byte torrent "poster" images.
|
||||
|
||||
Thanks to @grunjol, @rguedes, @furstenheim, @karloluis, @DiegoRBaquero, @alxhotel,
|
||||
@AgentEpsilon, @remijouannet, Rolando Guedes, @dcposch, and @feross for contributing
|
||||
to this release!
|
||||
|
||||
## v0.3.3 - 2016-04-07
|
||||
|
||||
### Fixed
|
||||
|
||||
@@ -6,4 +6,4 @@ npm run package -- --sign
|
||||
git push
|
||||
git push --tags
|
||||
npm publish
|
||||
gh-release
|
||||
./node_modules/.bin/gh-release
|
||||
|
||||
@@ -43,6 +43,7 @@ function init () {
|
||||
app.setPath('userData', config.CONFIG_PATH)
|
||||
}
|
||||
|
||||
var isReady = false // app ready, windows can be created
|
||||
app.ipcReady = false // main window has finished loading and IPC is ready
|
||||
app.isQuitting = false
|
||||
|
||||
@@ -57,6 +58,8 @@ function init () {
|
||||
})
|
||||
|
||||
app.on('ready', function () {
|
||||
isReady = true
|
||||
|
||||
windows.createMainWindow()
|
||||
windows.createWebTorrentHiddenWindow()
|
||||
menu.init()
|
||||
@@ -83,7 +86,7 @@ function init () {
|
||||
})
|
||||
|
||||
app.on('activate', function () {
|
||||
windows.createMainWindow()
|
||||
if (isReady) windows.createMainWindow()
|
||||
})
|
||||
}
|
||||
|
||||
|
||||
20
main/menu.js
20
main/menu.js
@@ -86,6 +86,12 @@ function decreaseVolume () {
|
||||
}
|
||||
}
|
||||
|
||||
function openSubtitles () {
|
||||
if (windows.main) {
|
||||
windows.main.send('dispatch', 'openSubtitles')
|
||||
}
|
||||
}
|
||||
|
||||
function onWindowShow () {
|
||||
log('onWindowShow')
|
||||
getMenuItem('Full Screen').enabled = true
|
||||
@@ -103,6 +109,7 @@ function onPlayerOpen () {
|
||||
getMenuItem('Play/Pause').enabled = true
|
||||
getMenuItem('Increase Volume').enabled = true
|
||||
getMenuItem('Decrease Volume').enabled = true
|
||||
getMenuItem('Add Subtitles File...').enabled = true
|
||||
}
|
||||
|
||||
function onPlayerClose () {
|
||||
@@ -110,6 +117,7 @@ function onPlayerClose () {
|
||||
getMenuItem('Play/Pause').enabled = false
|
||||
getMenuItem('Increase Volume').enabled = false
|
||||
getMenuItem('Decrease Volume').enabled = false
|
||||
getMenuItem('Add Subtitles File...').enabled = false
|
||||
}
|
||||
|
||||
function onToggleFullScreen (isFullScreen) {
|
||||
@@ -199,7 +207,7 @@ function getAppMenuTemplate () {
|
||||
type: 'separator'
|
||||
},
|
||||
{
|
||||
label: process.platform === 'windows'
|
||||
label: process.platform === 'win32'
|
||||
? 'Close'
|
||||
: 'Close Window',
|
||||
accelerator: 'CmdOrCtrl+W',
|
||||
@@ -295,6 +303,14 @@ function getAppMenuTemplate () {
|
||||
accelerator: 'CmdOrCtrl+Down',
|
||||
click: decreaseVolume,
|
||||
enabled: false
|
||||
},
|
||||
{
|
||||
type: 'separator'
|
||||
},
|
||||
{
|
||||
label: 'Add Subtitles File...',
|
||||
click: openSubtitles,
|
||||
enabled: false
|
||||
}
|
||||
]
|
||||
},
|
||||
@@ -388,7 +404,7 @@ function getAppMenuTemplate () {
|
||||
}
|
||||
|
||||
// In Linux and Windows it is not possible to open both folders and files
|
||||
if (process.platform === 'linux' || process.platform === 'windows') {
|
||||
if (process.platform === 'linux' || process.platform === 'win32') {
|
||||
// File menu (Windows, Linux)
|
||||
template[0].submenu.unshift({
|
||||
label: 'Create New Torrent from File...',
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "webtorrent-desktop",
|
||||
"description": "WebTorrent, the streaming torrent client. For OS X, Windows, and Linux.",
|
||||
"version": "0.4.0",
|
||||
"version": "0.5.0",
|
||||
"author": {
|
||||
"name": "Feross Aboukhadijeh",
|
||||
"email": "feross@feross.org",
|
||||
@@ -16,6 +16,7 @@
|
||||
"dependencies": {
|
||||
"airplay-js": "guerrerocarlos/node-airplay-js",
|
||||
"application-config": "^0.2.1",
|
||||
"async": "^2.0.0-rc.5",
|
||||
"bitfield": "^1.0.2",
|
||||
"chromecasts": "^1.8.0",
|
||||
"concat-stream": "^1.5.1",
|
||||
@@ -27,6 +28,7 @@
|
||||
"electron-prebuilt": "1.0.2",
|
||||
"fs-extra": "^0.27.0",
|
||||
"hyperx": "^2.0.2",
|
||||
"iso-639-1": "^1.2.1",
|
||||
"languagedetect": "^1.1.1",
|
||||
"main-loop": "^3.2.0",
|
||||
"musicmetadata": "^2.0.2",
|
||||
@@ -80,7 +82,7 @@
|
||||
"open-config": "node ./bin/open-config.js",
|
||||
"package": "node ./bin/package.js",
|
||||
"start": "electron .",
|
||||
"test": "standard && ./bin/check-deps.js",
|
||||
"test": "standard && node ./bin/check-deps.js",
|
||||
"update-authors": "./bin/update-authors.sh"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -117,10 +117,6 @@ table {
|
||||
float: right;
|
||||
}
|
||||
|
||||
.expand-collapse {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.expand-collapse.expanded::before {
|
||||
content: '▲'
|
||||
}
|
||||
@@ -366,7 +362,6 @@ button { /* Rectangular text buttons */
|
||||
border-radius: 3px;
|
||||
font-size: 14px;
|
||||
font-weight: bold;
|
||||
cursor: pointer;
|
||||
color: #aaa;
|
||||
outline: none;
|
||||
}
|
||||
@@ -597,7 +592,7 @@ body.drag .app::after {
|
||||
}
|
||||
|
||||
.torrent-details {
|
||||
padding: 8em 20px 20px 20px;
|
||||
padding: 8em 12px 20px 20px;
|
||||
}
|
||||
|
||||
.torrent-details table {
|
||||
@@ -611,6 +606,10 @@ body.drag .app::after {
|
||||
height: 28px;
|
||||
}
|
||||
|
||||
.torrent-details td {
|
||||
vertical-align: center;
|
||||
}
|
||||
|
||||
.torrent-details tr:hover {
|
||||
background-color: rgba(200, 200, 200, 0.3);
|
||||
}
|
||||
@@ -621,16 +620,16 @@ body.drag .app::after {
|
||||
vertical-align: bottom;
|
||||
}
|
||||
|
||||
.torrent-details td.col-icon {
|
||||
width: 2em;
|
||||
}
|
||||
|
||||
.torrent-details td.col-icon .icon {
|
||||
.torrent-details td .icon {
|
||||
font-size: 18px;
|
||||
position: relative;
|
||||
top: 3px;
|
||||
}
|
||||
|
||||
.torrent-details td.col-icon {
|
||||
width: 2em;
|
||||
}
|
||||
|
||||
.torrent-details td.col-name {
|
||||
width: auto;
|
||||
text-overflow: ellipsis;
|
||||
@@ -646,6 +645,11 @@ body.drag .app::after {
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
.torrent-details td.col-select {
|
||||
width: 2em;
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
/*
|
||||
* PLAYER
|
||||
*/
|
||||
|
||||
@@ -14,9 +14,11 @@ var ipcRenderer = electron.ipcRenderer
|
||||
setupIpc()
|
||||
|
||||
var appConfig = require('application-config')('WebTorrent')
|
||||
var Async = require('async')
|
||||
var concat = require('concat-stream')
|
||||
var dragDrop = require('drag-drop')
|
||||
var fs = require('fs-extra')
|
||||
var iso639 = require('iso-639-1')
|
||||
var mainLoop = require('main-loop')
|
||||
var path = require('path')
|
||||
|
||||
@@ -43,6 +45,9 @@ var Cast = null
|
||||
// For easy debugging in Developer Tools
|
||||
var state = global.state = State.getInitialState()
|
||||
|
||||
// Push the first page into the location history
|
||||
state.location.go({ url: 'home' })
|
||||
|
||||
var vdomLoop
|
||||
|
||||
// All state lives in state.js. `state.saved` is read from and written to a file.
|
||||
@@ -58,9 +63,6 @@ function init () {
|
||||
// Clean up the freshly-loaded config file, which may be from an older version
|
||||
cleanUpConfig()
|
||||
|
||||
// Push the first page into the location history
|
||||
state.location.go({ url: 'home' })
|
||||
|
||||
// Restart everything we were torrenting last time the app ran
|
||||
resumeTorrents()
|
||||
|
||||
@@ -151,6 +153,11 @@ function cleanUpConfig () {
|
||||
delete ts.posterURL
|
||||
ts.posterFileName = infoHash + extension
|
||||
}
|
||||
|
||||
// Migration: add per-file selections
|
||||
if (!ts.selections) {
|
||||
ts.selections = ts.files.map((x) => true)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
@@ -229,6 +236,9 @@ function dispatch (action, ...args) {
|
||||
if (action === 'toggleSelectTorrent') {
|
||||
toggleSelectTorrent(args[0] /* infoHash */)
|
||||
}
|
||||
if (action === 'toggleTorrentFile') {
|
||||
toggleTorrentFile(args[0] /* infoHash */, args[1] /* index */)
|
||||
}
|
||||
if (action === 'openTorrentContextMenu') {
|
||||
openTorrentContextMenu(args[0] /* infoHash */)
|
||||
}
|
||||
@@ -295,10 +305,10 @@ function dispatch (action, ...args) {
|
||||
openSubtitles()
|
||||
}
|
||||
if (action === 'selectSubtitle') {
|
||||
selectSubtitle(args[0] /* label */)
|
||||
selectSubtitle(args[0] /* index */)
|
||||
}
|
||||
if (action === 'showSubtitles') {
|
||||
showSubtitles()
|
||||
if (action === 'toggleSubtitlesMenu') {
|
||||
toggleSubtitlesMenu()
|
||||
}
|
||||
if (action === 'mediaStalled') {
|
||||
state.playing.isStalled = true
|
||||
@@ -422,7 +432,7 @@ function openSubtitles () {
|
||||
properties: [ 'openFile' ]
|
||||
}, function (filenames) {
|
||||
if (!Array.isArray(filenames)) return
|
||||
addSubtitle({path: filenames[0]})
|
||||
addSubtitles(filenames, true)
|
||||
})
|
||||
}
|
||||
|
||||
@@ -542,7 +552,7 @@ function onOpen (files) {
|
||||
// In the player, the only drag-drop function is adding subtitles
|
||||
var isInPlayer = state.location.current().url === 'player'
|
||||
if (isInPlayer) {
|
||||
return files.filter(isSubtitle).forEach(addSubtitle)
|
||||
return addSubtitles(files.filter(isSubtitle), true)
|
||||
}
|
||||
|
||||
// Otherwise, you can only drag-drop onto the home screen
|
||||
@@ -591,50 +601,102 @@ function addTorrent (torrentId) {
|
||||
ipcRenderer.send('wt-start-torrenting', torrentKey, torrentId, path)
|
||||
}
|
||||
|
||||
function addSubtitle (file) {
|
||||
function addSubtitles (files, autoSelect) {
|
||||
// Subtitles are only supported while playing video
|
||||
if (state.playing.type !== 'video') return
|
||||
|
||||
// Read the files concurrently, then add all resulting subtitle tracks
|
||||
console.log(files)
|
||||
var subs = state.playing.subtitles
|
||||
Async.map(files, loadSubtitle, function (err, tracks) {
|
||||
if (err) return onError(err)
|
||||
|
||||
for (var i = 0; i < tracks.length; i++) {
|
||||
// No dupes allowed
|
||||
var track = tracks[i]
|
||||
if (subs.tracks.some((t) => track.filePath === t.filePath)) continue
|
||||
|
||||
// Add the track
|
||||
subs.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 = subs.tracks.length - 1
|
||||
}
|
||||
}
|
||||
|
||||
// Finally, make sure no two tracks have the same label
|
||||
relabelSubtitles()
|
||||
})
|
||||
}
|
||||
|
||||
function loadSubtitle (file, cb) {
|
||||
var srtToVtt = require('srt-to-vtt')
|
||||
var LanguageDetect = require('languagedetect')
|
||||
|
||||
if (state.playing.type !== 'video') return
|
||||
fs.createReadStream(file.path || file).pipe(srtToVtt()).pipe(concat(function (buf) {
|
||||
// Read the .SRT or .VTT file, parse it, add subtitle track
|
||||
var filePath = file.path || file
|
||||
fs.createReadStream(filePath).pipe(srtToVtt()).pipe(concat(function (buf) {
|
||||
// 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)
|
||||
|
||||
// Set the cue text position so it appears above the player controls.
|
||||
// The only way to change cue text position is by modifying the VTT. It is not
|
||||
// possible via CSS.
|
||||
var langDetected = (new LanguageDetect()).detect(buf.toString().replace(/(.*-->.*)/g, ''), 2)
|
||||
langDetected = langDetected.length ? langDetected[0][0] : 'subtitle'
|
||||
langDetected = langDetected.slice(0, 1).toUpperCase() + langDetected.slice(1)
|
||||
var subtitles = Buffer(buf.toString().replace(/(-->.*)/g, '$1 line:88%'))
|
||||
var track = {
|
||||
buffer: 'data:text/vtt;base64,' + subtitles.toString('base64'),
|
||||
language: langDetected,
|
||||
label: langDetected,
|
||||
selected: true
|
||||
filePath: filePath
|
||||
}
|
||||
state.playing.subtitles.tracks.forEach(function (trackItem) {
|
||||
trackItem.selected = false
|
||||
if (trackItem.label === track.label) {
|
||||
var labelParts = /([^\d]+)(\d+)$/.exec(track.label)
|
||||
track.label = labelParts
|
||||
? labelParts[1] + (parseInt(labelParts[2]) + 1)
|
||||
: track.label + ' 2'
|
||||
}
|
||||
})
|
||||
state.playing.subtitles.change = track.label
|
||||
state.playing.subtitles.tracks.push(track)
|
||||
state.playing.subtitles.enabled = true
|
||||
|
||||
cb(null, track)
|
||||
}))
|
||||
}
|
||||
|
||||
function selectSubtitle (label) {
|
||||
state.playing.subtitles.tracks.forEach(function (track) {
|
||||
track.selected = (track.label === label)
|
||||
})
|
||||
state.playing.subtitles.enabled = !!label
|
||||
state.playing.subtitles.change = label
|
||||
state.playing.subtitles.show = false
|
||||
function selectSubtitle (ix) {
|
||||
state.playing.subtitles.selectedIndex = ix
|
||||
}
|
||||
|
||||
function showSubtitles () {
|
||||
state.playing.subtitles.show = !state.playing.subtitles.show
|
||||
// Checks whether a language name like "English" or "German" matches the system
|
||||
// language, aka the current locale
|
||||
function isSystemLanguage (language) {
|
||||
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
|
||||
@@ -655,7 +717,7 @@ function startTorrentingSummary (torrentSummary) {
|
||||
}
|
||||
|
||||
console.log('start torrenting %s %s', s.torrentKey, torrentID)
|
||||
ipcRenderer.send('wt-start-torrenting', s.torrentKey, torrentID, path, s.fileModtimes)
|
||||
ipcRenderer.send('wt-start-torrenting', s.torrentKey, torrentID, path, s.fileModtimes, s.selections)
|
||||
}
|
||||
|
||||
//
|
||||
@@ -764,6 +826,9 @@ function torrentMetadata (torrentKey, torrentInfo) {
|
||||
torrentSummary.path = torrentInfo.path
|
||||
torrentSummary.files = torrentInfo.files
|
||||
torrentSummary.magnetURI = torrentInfo.magnetURI
|
||||
if (!torrentSummary.selections) {
|
||||
torrentSummary.selections = torrentSummary.files.map((x) => true)
|
||||
}
|
||||
update()
|
||||
|
||||
// Save the .torrent file, if it hasn't been saved already
|
||||
@@ -815,6 +880,8 @@ function torrentProgress (progressInfo) {
|
||||
torrentSummary.progress = p
|
||||
})
|
||||
|
||||
checkForSubtitles()
|
||||
|
||||
update()
|
||||
}
|
||||
|
||||
@@ -914,6 +981,9 @@ function openPlayerFromActiveTorrent (torrentSummary, index, timeout, cb) {
|
||||
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)
|
||||
@@ -999,6 +1069,14 @@ function toggleSelectTorrent (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()
|
||||
|
||||
@@ -9,7 +9,8 @@ var LocationHistory = require('./lib/location-history')
|
||||
module.exports = {
|
||||
getInitialState,
|
||||
getDefaultPlayState,
|
||||
getDefaultSavedState
|
||||
getDefaultSavedState,
|
||||
getPlayingTorrentSummary
|
||||
}
|
||||
|
||||
function getInitialState () {
|
||||
@@ -57,7 +58,12 @@ function getInitialState () {
|
||||
*
|
||||
* Also accessible via `require('application-config')('WebTorrent').filePath`
|
||||
*/
|
||||
saved: {}
|
||||
saved: {},
|
||||
|
||||
/*
|
||||
* Getters, for convenience
|
||||
*/
|
||||
getPlayingTorrentSummary
|
||||
}
|
||||
}
|
||||
|
||||
@@ -75,8 +81,9 @@ function getDefaultPlayState () {
|
||||
lastTimeUpdate: 0, /* Unix time in ms */
|
||||
mouseStationarySince: 0, /* Unix time in ms */
|
||||
subtitles: {
|
||||
tracks: [], /* subtitles file (Buffer) */
|
||||
enabled: false
|
||||
tracks: [], /* subtitle tracks, each {label, language, ...} */
|
||||
selectedIndex: -1, /* current subtitle track */
|
||||
showMenu: false /* popover menu, above the video */
|
||||
},
|
||||
aspectRatio: 0 /* aspect ratio of the video */
|
||||
}
|
||||
@@ -263,3 +270,8 @@ function getDefaultSavedState () {
|
||||
: remote.app.getPath('downloads')
|
||||
}
|
||||
}
|
||||
|
||||
function getPlayingTorrentSummary () {
|
||||
var infoHash = this.playing.infoHash
|
||||
return this.saved.torrents.find((x) => x.infoHash === infoHash)
|
||||
}
|
||||
|
||||
@@ -47,8 +47,8 @@ function CreateTorrentPage (state) {
|
||||
basePath = pathPrefix
|
||||
} else {
|
||||
// Multi file torrent: /a/b/{foo, bar}.jpg -> torrent name "b", path "/a"
|
||||
defaultName = files[0].name
|
||||
basePath = path.basename(pathPrefix)
|
||||
defaultName = path.basename(pathPrefix)
|
||||
basePath = path.dirname(pathPrefix)
|
||||
}
|
||||
var maxFileElems = 100
|
||||
var fileElems = files.slice(0, maxFileElems).map(function (file) {
|
||||
|
||||
@@ -54,14 +54,10 @@ function renderMedia (state) {
|
||||
state.playing.setVolume = null
|
||||
}
|
||||
|
||||
// fix textTrack cues not been removed <track> rerender
|
||||
if (state.playing.subtitles.change) {
|
||||
var tracks = mediaElement.textTracks
|
||||
for (var j = 0; j < tracks.length; j++) {
|
||||
// mode is not an <track> attribute, only available on DOM
|
||||
tracks[j].mode = (tracks[j].label === state.playing.subtitles.change) ? 'showing' : 'hidden'
|
||||
}
|
||||
state.playing.subtitles.change = null
|
||||
// Switch to the newly added subtitle track, if available
|
||||
var tracks = mediaElement.textTracks
|
||||
for (var j = 0; j < tracks.length; j++) {
|
||||
tracks[j].mode = (j === state.playing.subtitles.selectedIndex) ? 'showing' : 'hidden'
|
||||
}
|
||||
|
||||
state.playing.currentTime = mediaElement.currentTime
|
||||
@@ -71,13 +67,13 @@ function renderMedia (state) {
|
||||
|
||||
// Add subtitles to the <video> tag
|
||||
var trackTags = []
|
||||
|
||||
if (state.playing.subtitles.enabled && state.playing.subtitles.tracks.length > 0) {
|
||||
if (state.playing.subtitles.selectedIndex >= 0) {
|
||||
for (var i = 0; i < state.playing.subtitles.tracks.length; i++) {
|
||||
var track = state.playing.subtitles.tracks[i]
|
||||
var isSelected = state.playing.subtitles.selectedIndex === i
|
||||
trackTags.push(hx`
|
||||
<track
|
||||
${track.selected ? 'default' : ''}
|
||||
${isSelected ? 'default' : ''}
|
||||
label=${track.label}
|
||||
type='subtitles'
|
||||
src=${track.buffer}>
|
||||
@@ -165,7 +161,7 @@ function renderOverlay (state) {
|
||||
}
|
||||
|
||||
function renderAudioMetadata (state) {
|
||||
var torrentSummary = getPlayingTorrentSummary(state)
|
||||
var torrentSummary = state.getPlayingTorrentSummary()
|
||||
var fileSummary = torrentSummary.files[state.playing.fileIndex]
|
||||
if (!fileSummary.audioInfo) return
|
||||
var info = fileSummary.audioInfo
|
||||
@@ -204,7 +200,7 @@ function renderLoadingSpinner (state) {
|
||||
(new Date().getTime() - state.playing.lastTimeUpdate > 2000)
|
||||
if (!isProbablyStalled) return
|
||||
|
||||
var prog = getPlayingTorrentSummary(state).progress || {}
|
||||
var prog = state.getPlayingTorrentSummary().progress || {}
|
||||
var fileProgress = 0
|
||||
if (prog.files) {
|
||||
var file = prog.files[state.playing.fileIndex]
|
||||
@@ -270,22 +266,24 @@ function renderCastScreen (state) {
|
||||
|
||||
function renderSubtitlesOptions (state) {
|
||||
var subtitles = state.playing.subtitles
|
||||
if (!subtitles.tracks.length || !subtitles.show) return
|
||||
if (!subtitles.tracks.length || !subtitles.showMenu) return
|
||||
|
||||
var items = subtitles.tracks.map(function (track) {
|
||||
var items = subtitles.tracks.map(function (track, ix) {
|
||||
var isSelected = state.playing.subtitles.selectedIndex === ix
|
||||
return hx`
|
||||
<li onclick=${dispatcher('selectSubtitle', track.label)}>
|
||||
<i.icon>${track.selected ? 'radio_button_checked' : 'radio_button_unchecked'}</i>
|
||||
<li onclick=${dispatcher('selectSubtitle', ix)}>
|
||||
<i.icon>${isSelected ? 'radio_button_checked' : 'radio_button_unchecked'}</i>
|
||||
${track.label}
|
||||
</li>
|
||||
`
|
||||
})
|
||||
|
||||
var noneSelected = state.playing.subtitles.selectedIndex === -1
|
||||
return hx`
|
||||
<ul.subtitles-list>
|
||||
${items}
|
||||
<li onclick=${dispatcher('selectSubtitle', '')}>
|
||||
<i.icon>${!subtitles.enabled ? 'radio_button_checked' : 'radio_button_unchecked'}</i>
|
||||
<li onclick=${dispatcher('selectSubtitle', -1)}>
|
||||
<i.icon>${noneSelected ? 'radio_button_checked' : 'radio_button_unchecked'}</i>
|
||||
None
|
||||
</li>
|
||||
</ul>
|
||||
@@ -297,7 +295,7 @@ function renderPlayerControls (state) {
|
||||
var playbackCursorStyle = { left: 'calc(' + positionPercent + '% - 8px)' }
|
||||
var captionsClass = state.playing.subtitles.tracks.length === 0
|
||||
? 'disabled'
|
||||
: state.playing.subtitles.enabled
|
||||
: state.playing.subtitles.selectedIndex >= 0
|
||||
? 'active'
|
||||
: ''
|
||||
|
||||
@@ -484,7 +482,7 @@ function renderPlayerControls (state) {
|
||||
// if no subtitles available select it
|
||||
dispatch('openSubtitles')
|
||||
} else {
|
||||
dispatch('showSubtitles')
|
||||
dispatch('toggleSubtitlesMenu')
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -495,7 +493,7 @@ var volumeChanging = false
|
||||
// Renders the loading bar. Shows which parts of the torrent are loaded, which
|
||||
// can be "spongey" / non-contiguous
|
||||
function renderLoadingBar (state) {
|
||||
var torrentSummary = getPlayingTorrentSummary(state)
|
||||
var torrentSummary = state.getPlayingTorrentSummary()
|
||||
if (!torrentSummary.progress) {
|
||||
return []
|
||||
}
|
||||
@@ -532,7 +530,7 @@ function renderLoadingBar (state) {
|
||||
|
||||
// Returns the CSS background-image string for a poster image + dark vignette
|
||||
function cssBackgroundImagePoster (state) {
|
||||
var torrentSummary = getPlayingTorrentSummary(state)
|
||||
var torrentSummary = state.getPlayingTorrentSummary()
|
||||
var posterPath = TorrentSummary.getPosterPath(torrentSummary)
|
||||
if (!posterPath) return ''
|
||||
return cssBackgroundImageDarkGradient() + `, url(${posterPath})`
|
||||
@@ -542,8 +540,3 @@ function cssBackgroundImageDarkGradient () {
|
||||
return 'radial-gradient(circle at center, ' +
|
||||
'rgba(0,0,0,0.4) 0%, rgba(0,0,0,1) 100%)'
|
||||
}
|
||||
|
||||
function getPlayingTorrentSummary (state) {
|
||||
var infoHash = state.playing.infoHash
|
||||
return state.saved.torrents.find((x) => x.infoHash === infoHash)
|
||||
}
|
||||
|
||||
@@ -208,7 +208,8 @@ function TorrentList (state) {
|
||||
// Show a single torrentSummary file in the details view for a single torrent
|
||||
function renderFileRow (torrentSummary, file, index) {
|
||||
// First, find out how much of the file we've downloaded
|
||||
var isDone = false
|
||||
var isSelected = torrentSummary.selections[index] // Are we even torrenting it?
|
||||
var isDone = false // Are we finished torrenting it?
|
||||
var progress = ''
|
||||
if (torrentSummary.progress && torrentSummary.progress.files) {
|
||||
var fileProg = torrentSummary.progress.files[index]
|
||||
@@ -217,26 +218,38 @@ function TorrentList (state) {
|
||||
}
|
||||
|
||||
// Second, render the file as a table row
|
||||
var isPlayable = TorrentPlayer.isPlayable(file)
|
||||
var infoHash = torrentSummary.infoHash
|
||||
var icon
|
||||
var rowClass = ''
|
||||
var handleClick
|
||||
if (TorrentPlayer.isPlayable(file)) {
|
||||
if (isPlayable) {
|
||||
icon = 'play_arrow' /* playable? add option to play */
|
||||
handleClick = dispatcher('play', infoHash, index)
|
||||
} else {
|
||||
icon = 'description' /* file icon, opens in OS default app */
|
||||
rowClass = isDone ? '' : 'disabled'
|
||||
handleClick = dispatcher('openFile', infoHash, index)
|
||||
}
|
||||
var rowClass = ''
|
||||
if (!isSelected) rowClass = 'disabled' // File deselected, not being torrented
|
||||
if (!isDone && !isPlayable) rowClass = 'disabled' // Can't open yet, can't stream
|
||||
return hx`
|
||||
<tr onclick=${handleClick} class='${rowClass}'>
|
||||
<td class='col-icon'>
|
||||
<tr>
|
||||
<td class='col-icon ${rowClass}' onclick=${handleClick}>
|
||||
<i class='icon'>${icon}</i>
|
||||
</td>
|
||||
<td class='col-name'>${file.name}</td>
|
||||
<td class='col-progress'>${progress}</td>
|
||||
<td class='col-size'>${prettyBytes(file.length)}</td>
|
||||
<td class='col-name ${rowClass}' onclick=${handleClick}>
|
||||
${file.name}
|
||||
</td>
|
||||
<td class='col-progress ${rowClass}' onclick=${handleClick}>
|
||||
${isSelected ? progress : ''}
|
||||
</td>
|
||||
<td class='col-size ${rowClass}' onclick=${handleClick}>
|
||||
${prettyBytes(file.length)}
|
||||
</td>
|
||||
<td class='col-select'
|
||||
onclick=${dispatcher('toggleTorrentFile', infoHash, index)}>
|
||||
<i class='icon'>${isSelected ? 'close' : 'add'}</i>
|
||||
</td>
|
||||
</tr>
|
||||
`
|
||||
}
|
||||
|
||||
@@ -49,8 +49,8 @@ function init () {
|
||||
client.on('warning', (err) => ipc.send('wt-warning', null, err.message))
|
||||
client.on('error', (err) => ipc.send('wt-error', null, err.message))
|
||||
|
||||
ipc.on('wt-start-torrenting', (e, torrentKey, torrentID, path, fileModtimes) =>
|
||||
startTorrenting(torrentKey, torrentID, path, fileModtimes))
|
||||
ipc.on('wt-start-torrenting', (e, torrentKey, torrentID, path, fileModtimes, selections) =>
|
||||
startTorrenting(torrentKey, torrentID, path, fileModtimes, selections))
|
||||
ipc.on('wt-stop-torrenting', (e, infoHash) =>
|
||||
stopTorrenting(infoHash))
|
||||
ipc.on('wt-create-torrent', (e, torrentKey, options) =>
|
||||
@@ -65,6 +65,8 @@ function init () {
|
||||
startServer(infoHash, index))
|
||||
ipc.on('wt-stop-server', (e) =>
|
||||
stopServer())
|
||||
ipc.on('wt-select-files', (e, infoHash, selections) =>
|
||||
selectFiles(infoHash, selections))
|
||||
|
||||
ipc.send('ipcReadyWebTorrent')
|
||||
|
||||
@@ -73,7 +75,7 @@ function init () {
|
||||
|
||||
// Starts a given TorrentID, which can be an infohash, magnet URI, etc. Returns WebTorrent object
|
||||
// See https://github.com/feross/webtorrent/blob/master/docs/api.md#clientaddtorrentid-opts-function-ontorrent-torrent-
|
||||
function startTorrenting (torrentKey, torrentID, path, fileModtimes) {
|
||||
function startTorrenting (torrentKey, torrentID, path, fileModtimes, selections) {
|
||||
console.log('starting torrent %s: %s', torrentKey, torrentID)
|
||||
|
||||
var torrent = client.add(torrentID, {
|
||||
@@ -81,8 +83,13 @@ function startTorrenting (torrentKey, torrentID, path, fileModtimes) {
|
||||
fileModtimes: fileModtimes
|
||||
})
|
||||
torrent.key = torrentKey
|
||||
|
||||
// Listen for ready event, progress notifications, etc
|
||||
addTorrentEvents(torrent)
|
||||
|
||||
// Only download the files the user wants, not necessarily all files
|
||||
torrent.once('ready', () => selectFiles(torrent, selections))
|
||||
|
||||
return torrent
|
||||
}
|
||||
|
||||
@@ -157,9 +164,7 @@ function getTorrentFileInfo (file) {
|
||||
return {
|
||||
name: file.name,
|
||||
length: file.length,
|
||||
path: file.path,
|
||||
numPiecesPresent: 0,
|
||||
numPieces: null
|
||||
path: file.path
|
||||
}
|
||||
}
|
||||
|
||||
@@ -310,6 +315,48 @@ function getAudioMetadata (infoHash, index) {
|
||||
})
|
||||
}
|
||||
|
||||
function selectFiles (torrentOrInfoHash, selections) {
|
||||
// Get the torrent object
|
||||
var torrent
|
||||
if (typeof torrentOrInfoHash === 'string') {
|
||||
torrent = client.get(torrentOrInfoHash)
|
||||
} else {
|
||||
torrent = torrentOrInfoHash
|
||||
}
|
||||
|
||||
// Selections not specified?
|
||||
// Load all files. We still need to replace the default whole-torrent
|
||||
// selection with individual selections for each file, so we can
|
||||
// select/deselect files later on
|
||||
if (!selections) {
|
||||
selections = torrent.files.map((x) => true)
|
||||
}
|
||||
|
||||
// Selections specified incorrectly?
|
||||
if (selections.length !== torrent.files.length) {
|
||||
throw new Error('got ' + selections.length + ' file selections, ' +
|
||||
'but the torrent contains ' + torrent.files.length + ' files')
|
||||
}
|
||||
|
||||
// Remove default selection (whole torrent)
|
||||
torrent.deselect(0, torrent.pieces.length - 1, false)
|
||||
|
||||
// Add selections (individual files)
|
||||
for (var i = 0; i < selections.length; i++) {
|
||||
var file = torrent.files[i]
|
||||
if (selections[i]) {
|
||||
file.select()
|
||||
} else {
|
||||
console.log('deselecting file ' + i + ' of torrent ' + torrent.name)
|
||||
file.deselect()
|
||||
|
||||
// If we deselected a file, try to nuke it to save disk space
|
||||
var filePath = path.join(torrent.path, file.path)
|
||||
fs.unlink(filePath) // Ignore errors for now
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Gets a WebTorrent handle by torrentKey
|
||||
// Throws an Error if we're not currently torrenting anything w/ that key
|
||||
function getTorrent (torrentKey) {
|
||||
|
||||
Reference in New Issue
Block a user