Torrent details
Show file list, open folder containing downloaded files, open individual files, play/pause individual videos
This commit is contained in:
11
main/ipc.js
11
main/ipc.js
@@ -4,13 +4,13 @@ module.exports = {
|
|||||||
|
|
||||||
var debug = require('debug')('webtorrent-app:ipcMain')
|
var debug = require('debug')('webtorrent-app:ipcMain')
|
||||||
var electron = require('electron')
|
var electron = require('electron')
|
||||||
var menu = require('./menu')
|
|
||||||
var windows = require('./windows')
|
|
||||||
|
|
||||||
var app = electron.app
|
var app = electron.app
|
||||||
var ipcMain = electron.ipcMain
|
var ipcMain = electron.ipcMain
|
||||||
var powerSaveBlocker = electron.powerSaveBlocker
|
var powerSaveBlocker = electron.powerSaveBlocker
|
||||||
|
|
||||||
|
var menu = require('./menu')
|
||||||
|
var windows = require('./windows')
|
||||||
|
|
||||||
// has to be a number, not a boolean, and undefined throws an error
|
// has to be a number, not a boolean, and undefined throws an error
|
||||||
var powerSaveBlocked = 0
|
var powerSaveBlocked = 0
|
||||||
|
|
||||||
@@ -56,6 +56,11 @@ function init () {
|
|||||||
powerSaveBlocker.stop(powerSaveBlocked)
|
powerSaveBlocker.stop(powerSaveBlocked)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
ipcMain.on('openItem', function (e, path) {
|
||||||
|
console.log('opening file or folder: ' + path)
|
||||||
|
electron.shell.openItem(path)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function setBounds (bounds) {
|
function setBounds (bounds) {
|
||||||
|
|||||||
@@ -145,12 +145,14 @@ table {
|
|||||||
* BUTTONS
|
* BUTTONS
|
||||||
*/
|
*/
|
||||||
|
|
||||||
a, i {
|
a,
|
||||||
|
i {
|
||||||
cursor: default;
|
cursor: default;
|
||||||
-webkit-app-region: no-drag;
|
-webkit-app-region: no-drag;
|
||||||
}
|
}
|
||||||
|
|
||||||
a:not(.disabled):hover, i:not(.disabled):hover {
|
a:not(.disabled):hover,
|
||||||
|
i:not(.disabled):hover {
|
||||||
-webkit-filter: brightness(1.3);
|
-webkit-filter: brightness(1.3);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -210,7 +212,7 @@ a:not(.disabled):hover, i:not(.disabled):hover {
|
|||||||
|
|
||||||
.header .nav {
|
.header .nav {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
margin-right: 7px;
|
margin-right: 9px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header .nav.left {
|
.header .nav.left {
|
||||||
@@ -237,7 +239,8 @@ a:not(.disabled):hover, i:not(.disabled):hover {
|
|||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header .nav .back, .header .nav .forward {
|
.header .nav .back,
|
||||||
|
.header .nav .forward {
|
||||||
font-size: 30px;
|
font-size: 30px;
|
||||||
margin-top: -3px;
|
margin-top: -3px;
|
||||||
}
|
}
|
||||||
@@ -370,7 +373,8 @@ input {
|
|||||||
animation: fadein .4s;
|
animation: fadein .4s;
|
||||||
}
|
}
|
||||||
|
|
||||||
.torrent, .torrent-placeholder {
|
.torrent,
|
||||||
|
.torrent-placeholder {
|
||||||
height: 120px;
|
height: 120px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -390,11 +394,14 @@ input {
|
|||||||
text-shadow: rgba(0, 0, 0, 0.5) 0 0 4px;
|
text-shadow: rgba(0, 0, 0, 0.5) 0 0 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.torrent .metadata span:not(:last-child)::after {
|
||||||
|
content: ' — ';
|
||||||
|
}
|
||||||
|
|
||||||
.torrent .buttons {
|
.torrent .buttons {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 25px;
|
top: 25px;
|
||||||
right: 0;
|
right: 10px;
|
||||||
height: 100%;
|
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -403,7 +410,7 @@ input {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.torrent .buttons > * {
|
.torrent .buttons > * {
|
||||||
margin-right: 6px; /* space buttons apart, align the Xs under the + */
|
margin-left: 6px; /* space buttons apart */
|
||||||
}
|
}
|
||||||
|
|
||||||
.torrent .buttons .download {
|
.torrent .buttons .download {
|
||||||
@@ -413,7 +420,6 @@ input {
|
|||||||
border-radius: 14px;
|
border-radius: 14px;
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
padding-top: 6px;
|
padding-top: 6px;
|
||||||
margin-right: 10px; /* download and play btns need more space to look good */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.torrent .buttons .download.downloading {
|
.torrent .buttons .download.downloading {
|
||||||
@@ -467,24 +473,27 @@ input {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.torrent .buttons .delete {
|
.torrent .buttons .delete {
|
||||||
|
position: relative;
|
||||||
|
top: 2px;
|
||||||
opacity: 0.5;
|
opacity: 0.5;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.torrent .buttons .delete:hover {
|
||||||
|
opacity: 0.7;
|
||||||
|
}
|
||||||
|
|
||||||
.torrent .name {
|
.torrent .name {
|
||||||
font-size: 1.5em;
|
font-size: 1.5em;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
line-height: 1.5em;
|
line-height: 1.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.torrent .status, .torrent .status2 {
|
.torrent .status,
|
||||||
|
.torrent .status2 {
|
||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
line-height: 1.5em;
|
line-height: 1.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.torrent span:not(:last-child)::after {
|
|
||||||
content: ' — ';
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* TORRENT LIST: DRAG-DROP TARGET
|
* TORRENT LIST: DRAG-DROP TARGET
|
||||||
*/
|
*/
|
||||||
@@ -520,10 +529,20 @@ body.drag .torrent-placeholder span {
|
|||||||
padding: 8em 20px 20px 20px;
|
padding: 8em 20px 20px 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.torrent-details .open-folder {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
|
||||||
.torrent-details table {
|
.torrent-details table {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 1000px;
|
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
border: none;
|
||||||
|
border-spacing: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.torrent-details tr:hover,
|
||||||
|
.torrent-details .open-folder:hover {
|
||||||
|
background-color: rgba(200, 200, 200, 0.3);
|
||||||
}
|
}
|
||||||
|
|
||||||
.torrent-details td {
|
.torrent-details td {
|
||||||
@@ -531,6 +550,16 @@ body.drag .torrent-placeholder span {
|
|||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.torrent-details td.col-icon {
|
||||||
|
width: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.torrent-details td.col-icon .icon {
|
||||||
|
font-size: 18px;
|
||||||
|
position: relative;
|
||||||
|
top: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
.torrent-details td.col-name {
|
.torrent-details td.col-name {
|
||||||
width: auto;
|
width: auto;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
@@ -667,7 +696,8 @@ body.drag .torrent-placeholder span {
|
|||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
line-height: 1.5em;
|
line-height: 1.5em;
|
||||||
}
|
}
|
||||||
.torrent, .torrent-placeholder {
|
.torrent,
|
||||||
|
.torrent-placeholder {
|
||||||
height: 150px;
|
height: 150px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -677,7 +707,8 @@ body.drag .torrent-placeholder span {
|
|||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
line-height: 1.5em;
|
line-height: 1.5em;
|
||||||
}
|
}
|
||||||
.torrent, .torrent-placeholder {
|
.torrent,
|
||||||
|
.torrent-placeholder {
|
||||||
height: 180px;
|
height: 180px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ var patch = require('virtual-dom/patch')
|
|||||||
var App = require('./views/app')
|
var App = require('./views/app')
|
||||||
var config = require('../config')
|
var config = require('../config')
|
||||||
var torrentPoster = require('./lib/torrent-poster')
|
var torrentPoster = require('./lib/torrent-poster')
|
||||||
|
var TorrentPlayer = require('./lib/torrent-player')
|
||||||
|
|
||||||
// Electron apps have two processes: a main process (node) runs first and starts
|
// 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,
|
// a renderer process (essentially a Chrome window). We're in the renderer process,
|
||||||
@@ -169,8 +170,15 @@ function dispatch (action, ...args) {
|
|||||||
if (action === 'seed') {
|
if (action === 'seed') {
|
||||||
seed(args[0] /* files */)
|
seed(args[0] /* files */)
|
||||||
}
|
}
|
||||||
if (action === 'openPlayer') {
|
if (action === 'play') {
|
||||||
openPlayer(args[0] /* torrentSummary */)
|
// TODO: handle audio. video only for now.
|
||||||
|
openPlayer(args[0] /* torrentSummary */, args[1] /* index */)
|
||||||
|
}
|
||||||
|
if (action === 'openFile') {
|
||||||
|
openFile(args[0] /* torrentSummary */, args[1] /* index */)
|
||||||
|
}
|
||||||
|
if (action === 'openFolder') {
|
||||||
|
openFolder(args[0] /* torrentSummary */)
|
||||||
}
|
}
|
||||||
if (action === 'toggleTorrent') {
|
if (action === 'toggleTorrent') {
|
||||||
toggleTorrent(args[0] /* torrentSummary */)
|
toggleTorrent(args[0] /* torrentSummary */)
|
||||||
@@ -453,32 +461,33 @@ function generateTorrentPoster (torrent, torrentSummary) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function startServer (infoHash, cb) {
|
function startServer (infoHash, index, cb) {
|
||||||
if (state.server) return cb()
|
if (state.server) return cb()
|
||||||
|
|
||||||
var torrent = getTorrent(infoHash)
|
var torrent = getTorrent(infoHash)
|
||||||
if (!torrent) torrent = startTorrenting(infoHash)
|
if (!torrent) torrent = startTorrenting(infoHash)
|
||||||
if (torrent.ready) startServerFromReadyTorrent(torrent, cb)
|
if (torrent.ready) startServerFromReadyTorrent(torrent, index, cb)
|
||||||
else torrent.on('ready', () => startServerFromReadyTorrent(torrent, cb))
|
else torrent.on('ready', () => startServerFromReadyTorrent(torrent, index, cb))
|
||||||
}
|
}
|
||||||
|
|
||||||
function startServerFromReadyTorrent (torrent, cb) {
|
function startServerFromReadyTorrent (torrent, index, cb) {
|
||||||
// filter out file formats that the <video> tag definitely can't play
|
// automatically choose which file in the torrent to play, if necessary
|
||||||
var files = torrent.files.filter(function (file) {
|
if (!index) {
|
||||||
var extname = path.extname(file.name)
|
// filter out file formats that the <video> tag definitely can't play
|
||||||
return ['.mp4', '.m4v', '.webm', '.mov', '.mkv'].indexOf(extname) !== -1
|
var files = torrent.files.filter(TorrentPlayer.isPlayable)
|
||||||
})
|
if (files.length === 0) return cb(new Error('cannot play any files in torrent'))
|
||||||
|
// use largest file
|
||||||
|
var largestFile = files.reduce(function (a, b) {
|
||||||
|
return a.length > b.length ? a : b
|
||||||
|
})
|
||||||
|
index = torrent.files.indexOf(largestFile)
|
||||||
|
}
|
||||||
|
|
||||||
if (files.length === 0) return cb(new Error('cannot play any files in torrent'))
|
// update state
|
||||||
|
state.playing.infoHash = torrent.infoHash
|
||||||
|
state.playing.fileIndex = index
|
||||||
|
|
||||||
// use largest file
|
|
||||||
state.torrentPlaying = files.reduce(function (a, b) {
|
|
||||||
return a.length > b.length ? a : b
|
|
||||||
})
|
|
||||||
|
|
||||||
var index = torrent.files.indexOf(state.torrentPlaying)
|
|
||||||
var server = torrent.createServer()
|
var server = torrent.createServer()
|
||||||
|
|
||||||
server.listen(0, function () {
|
server.listen(0, function () {
|
||||||
var port = server.address().port
|
var port = server.address().port
|
||||||
var urlSuffix = ':' + port + '/' + index
|
var urlSuffix = ':' + port + '/' + index
|
||||||
@@ -495,9 +504,12 @@ function stopServer () {
|
|||||||
if (!state.server) return
|
if (!state.server) return
|
||||||
state.server.server.destroy()
|
state.server.server.destroy()
|
||||||
state.server = null
|
state.server = null
|
||||||
|
state.playing.infoHash = null
|
||||||
|
state.playing.fileIndex = null
|
||||||
}
|
}
|
||||||
|
|
||||||
function openPlayer (torrentSummary) {
|
// Opens the video player
|
||||||
|
function openPlayer (torrentSummary, index) {
|
||||||
var torrent = state.client.get(torrentSummary.infoHash)
|
var torrent = state.client.get(torrentSummary.infoHash)
|
||||||
if (!torrent || !torrent.done) playInterfaceSound(config.SOUND_PLAY)
|
if (!torrent || !torrent.done) playInterfaceSound(config.SOUND_PLAY)
|
||||||
torrentSummary.playStatus = 'requested'
|
torrentSummary.playStatus = 'requested'
|
||||||
@@ -509,7 +521,7 @@ function openPlayer (torrentSummary) {
|
|||||||
update()
|
update()
|
||||||
}, 10000) /* give it a few seconds */
|
}, 10000) /* give it a few seconds */
|
||||||
|
|
||||||
startServer(torrentSummary.infoHash, function (err) {
|
startServer(torrentSummary.infoHash, index, function (err) {
|
||||||
if (err) return onError(err)
|
if (err) return onError(err)
|
||||||
|
|
||||||
// if we timed out (user clicked play a long time ago), don't autoplay
|
// if we timed out (user clicked play a long time ago), don't autoplay
|
||||||
@@ -525,6 +537,22 @@ function openPlayer (torrentSummary) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function openFile (torrentSummary, index) {
|
||||||
|
var torrent = state.client.get(torrentSummary.infoHash)
|
||||||
|
if (!torrent) return
|
||||||
|
|
||||||
|
var filePath = path.join(torrent.path, torrent.files[index].path)
|
||||||
|
ipcRenderer.send('openItem', filePath)
|
||||||
|
}
|
||||||
|
|
||||||
|
function openFolder (torrentSummary) {
|
||||||
|
var torrent = state.client.get(torrentSummary.infoHash)
|
||||||
|
if (!torrent) return
|
||||||
|
|
||||||
|
var folderPath = path.join(torrent.path, torrent.name)
|
||||||
|
ipcRenderer.send('openItem', folderPath)
|
||||||
|
}
|
||||||
|
|
||||||
function closePlayer () {
|
function closePlayer () {
|
||||||
state.url = 'home'
|
state.url = 'home'
|
||||||
state.window.title = config.APP_NAME
|
state.window.title = config.APP_NAME
|
||||||
|
|||||||
13
renderer/lib/torrent-player.js
Normal file
13
renderer/lib/torrent-player.js
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
module.exports = {
|
||||||
|
isPlayable: isPlayable
|
||||||
|
}
|
||||||
|
|
||||||
|
var path = require('path')
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines whether a file in a torrent is audio/video we can play
|
||||||
|
*/
|
||||||
|
function isPlayable (file) {
|
||||||
|
var extname = path.extname(file.name)
|
||||||
|
return ['.mp4', '.m4v', '.webm', '.mov', '.mkv'].indexOf(extname) !== -1
|
||||||
|
}
|
||||||
@@ -4,14 +4,18 @@ var path = require('path')
|
|||||||
var config = require('../config')
|
var config = require('../config')
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
/* Temporary state disappears once the program exits.
|
/*
|
||||||
|
* Temporary state disappears once the program exits.
|
||||||
* It can contain complex objects like open connections, etc.
|
* It can contain complex objects like open connections, etc.
|
||||||
*/
|
*/
|
||||||
client: null, /* the WebTorrent client */
|
client: null, /* the WebTorrent client */
|
||||||
prev: {}, /* used for state diffing in updateElectron() */
|
|
||||||
server: null, /* local WebTorrent-to-HTTP server */
|
server: null, /* local WebTorrent-to-HTTP server */
|
||||||
torrentPlaying: null, /* the torrent we're streaming. see client.torrents */
|
prev: {}, /* used for state diffing in updateElectron() */
|
||||||
selectedInfoHash: null, /* the torrent we've selected to view details. see state.torrents */
|
selectedInfoHash: null, /* the torrent we've selected to view details. see state.torrents */
|
||||||
|
playing: {
|
||||||
|
infoHash: null, /* the info hash of the torrent we're playing */
|
||||||
|
fileIndex: null /* the zero-based index within the torrent */
|
||||||
|
},
|
||||||
// history: [], /* track how we got to the current view. enables Back button */
|
// history: [], /* track how we got to the current view. enables Back button */
|
||||||
// historyIndex: 0,
|
// historyIndex: 0,
|
||||||
url: 'home',
|
url: 'home',
|
||||||
@@ -36,7 +40,8 @@ module.exports = {
|
|||||||
mouseStationarySince: 0 /* Unix time in ms */
|
mouseStationarySince: 0 /* Unix time in ms */
|
||||||
},
|
},
|
||||||
|
|
||||||
/* Saved state is read from and written to a file every time the app runs.
|
/*
|
||||||
|
* Saved state is read from and written to a file every time the app runs.
|
||||||
* It should be simple and minimal and must be JSON.
|
* It should be simple and minimal and must be JSON.
|
||||||
*
|
*
|
||||||
* Config path:
|
* Config path:
|
||||||
@@ -49,9 +54,7 @@ module.exports = {
|
|||||||
*
|
*
|
||||||
* Also accessible via `require('application-config')('WebTorrent').filePath`
|
* Also accessible via `require('application-config')('WebTorrent').filePath`
|
||||||
*/
|
*/
|
||||||
saved: {
|
saved: {},
|
||||||
torrents: []
|
|
||||||
},
|
|
||||||
|
|
||||||
/* If the saved state file doesn't exist yet, here's what we use instead */
|
/* If the saved state file doesn't exist yet, here's what we use instead */
|
||||||
defaultSavedState: {
|
defaultSavedState: {
|
||||||
|
|||||||
@@ -64,7 +64,7 @@ function Player (state, dispatch) {
|
|||||||
function renderPlayerControls (state, dispatch) {
|
function renderPlayerControls (state, dispatch) {
|
||||||
var positionPercent = 100 * state.video.currentTime / state.video.duration
|
var positionPercent = 100 * state.video.currentTime / state.video.duration
|
||||||
var playbackCursorStyle = { left: 'calc(' + positionPercent + '% - 8px)' }
|
var playbackCursorStyle = { left: 'calc(' + positionPercent + '% - 8px)' }
|
||||||
var torrent = state.torrentPlaying._torrent
|
var torrent = state.client.get(state.playing.infoHash)
|
||||||
|
|
||||||
var elements = [
|
var elements = [
|
||||||
hx`
|
hx`
|
||||||
@@ -132,7 +132,7 @@ function renderPlayerControls (state, dispatch) {
|
|||||||
// Renders the loading bar. Shows which parts of the torrent are loaded, which
|
// Renders the loading bar. Shows which parts of the torrent are loaded, which
|
||||||
// can be "spongey" / non-contiguous
|
// can be "spongey" / non-contiguous
|
||||||
function renderLoadingBar (state) {
|
function renderLoadingBar (state) {
|
||||||
var torrent = state.torrentPlaying._torrent
|
var torrent = state.client.get(state.playing.infoHash)
|
||||||
if (torrent === null) {
|
if (torrent === null) {
|
||||||
return []
|
return []
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,165 +5,205 @@ var hyperx = require('hyperx')
|
|||||||
var hx = hyperx(h)
|
var hx = hyperx(h)
|
||||||
var prettyBytes = require('prettier-bytes')
|
var prettyBytes = require('prettier-bytes')
|
||||||
|
|
||||||
|
var TorrentPlayer = require('../lib/torrent-player')
|
||||||
|
|
||||||
function TorrentList (state, dispatch) {
|
function TorrentList (state, dispatch) {
|
||||||
var list = state.saved.torrents.map(
|
var torrentRows = state.saved.torrents.map(
|
||||||
(torrentSummary) => renderTorrent(torrentSummary, state, dispatch))
|
(torrentSummary) => renderTorrent(torrentSummary))
|
||||||
return hx`
|
return hx`
|
||||||
<div class='torrent-list'>
|
<div class='torrent-list'>
|
||||||
${list}
|
${torrentRows}
|
||||||
<div class='torrent-placeholder'>
|
<div class='torrent-placeholder'>
|
||||||
<span class='ellipsis'>Drop a torrent file here or paste a magnet link</span>
|
<span class='ellipsis'>Drop a torrent file here or paste a magnet link</span>
|
||||||
</div>
|
</div>
|
||||||
</div>`
|
</div>`
|
||||||
}
|
|
||||||
|
|
||||||
// Renders a torrent in the torrent list
|
// Renders a torrent in the torrent list
|
||||||
// Includes name, download status, play button, background image
|
// Includes name, download status, play button, background image
|
||||||
// May be expanded for additional info, including the list of files inside
|
// May be expanded for additional info, including the list of files inside
|
||||||
function renderTorrent (torrentSummary, state, dispatch) {
|
function renderTorrent (torrentSummary) {
|
||||||
// Get ephemeral data (like progress %) directly from the WebTorrent handle
|
// Get ephemeral data (like progress %) directly from the WebTorrent handle
|
||||||
var infoHash = torrentSummary.infoHash
|
var infoHash = torrentSummary.infoHash
|
||||||
var torrent = state.client.torrents.find((x) => x.infoHash === infoHash)
|
var torrent = state.client.torrents.find((x) => x.infoHash === infoHash)
|
||||||
var isSelected = state.selectedInfoHash === infoHash
|
var isSelected = state.selectedInfoHash === infoHash
|
||||||
|
|
||||||
// Background image: show some nice visuals, like a frame from the movie, if possible
|
// Background image: show some nice visuals, like a frame from the movie, if possible
|
||||||
var style = {}
|
var style = {}
|
||||||
if (torrentSummary.posterURL) {
|
if (torrentSummary.posterURL) {
|
||||||
var gradient = isSelected
|
var gradient = isSelected
|
||||||
? 'linear-gradient(to bottom, rgba(0, 0, 0, 0.8) 0%, rgba(0, 0, 0, 0.4) 100%)'
|
? 'linear-gradient(to bottom, rgba(0, 0, 0, 0.8) 0%, rgba(0, 0, 0, 0.4) 100%)'
|
||||||
: 'linear-gradient(to bottom, rgba(0, 0, 0, 0.5) 0%, rgba(0, 0, 0, 0) 100%)'
|
: 'linear-gradient(to bottom, rgba(0, 0, 0, 0.5) 0%, rgba(0, 0, 0, 0) 100%)'
|
||||||
style['background-image'] = gradient + `, url('${torrentSummary.posterURL}')`
|
style['background-image'] = gradient + `, url('${torrentSummary.posterURL}')`
|
||||||
}
|
|
||||||
|
|
||||||
// Foreground: name of the torrent, basic info like size, play button,
|
|
||||||
// cast buttons if available, and delete
|
|
||||||
var classes = ['torrent']
|
|
||||||
// playStatus turns the play button into a loading spinner or error icon
|
|
||||||
if (torrent && torrent.playStatus) classes.push(torrent.playStatus)
|
|
||||||
if (isSelected) classes.push('selected')
|
|
||||||
classes = classes.join(' ')
|
|
||||||
return hx`
|
|
||||||
<div style=${style} class=${classes} onclick=${() => dispatch('toggleSelectTorrent', infoHash)}>
|
|
||||||
${renderTorrentMetadata(torrent, torrentSummary)}
|
|
||||||
${renderTorrentButtons(torrentSummary, dispatch)}
|
|
||||||
${isSelected ? renderTorrentDetails(torrent, torrentSummary) : ''}
|
|
||||||
</div>
|
|
||||||
`
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show name, download status, % complete
|
|
||||||
function renderTorrentMetadata (torrent, torrentSummary) {
|
|
||||||
var name = torrentSummary.displayName || torrentSummary.name || 'Loading torrent...'
|
|
||||||
var elements = [hx`
|
|
||||||
<div class='name ellipsis'>${name}</div>
|
|
||||||
`]
|
|
||||||
|
|
||||||
// If a torrent is paused and we only get the torrentSummary
|
|
||||||
// If it's downloading/seeding then we have more information
|
|
||||||
if (torrent) {
|
|
||||||
var progress = Math.floor(100 * torrent.progress)
|
|
||||||
var downloaded = prettyBytes(torrent.downloaded)
|
|
||||||
var total = prettyBytes(torrent.length || 0)
|
|
||||||
if (downloaded !== total) downloaded += ` / ${total}`
|
|
||||||
|
|
||||||
elements.push(hx`
|
|
||||||
<div class='status ellipsis'>
|
|
||||||
${getFilesLength()}
|
|
||||||
<span>${getPeers()}</span>
|
|
||||||
<span>↓ ${prettyBytes(torrent.downloadSpeed || 0)}/s</span>
|
|
||||||
<span>↑ ${prettyBytes(torrent.uploadSpeed || 0)}/s</span>
|
|
||||||
</div>
|
|
||||||
`)
|
|
||||||
elements.push(hx`
|
|
||||||
<div class='status2 ellipsis'>
|
|
||||||
<span class='progress'>${progress}%</span>
|
|
||||||
<span>${downloaded}</span>
|
|
||||||
</div>
|
|
||||||
`)
|
|
||||||
}
|
|
||||||
|
|
||||||
return hx`<div class='metadata'>${elements}</div>`
|
|
||||||
|
|
||||||
function getPeers () {
|
|
||||||
var count = torrent.numPeers === 1 ? 'peer' : 'peers'
|
|
||||||
return `${torrent.numPeers} ${count}`
|
|
||||||
}
|
|
||||||
|
|
||||||
function getFilesLength () {
|
|
||||||
if (torrent.ready && torrent.files.length > 1) {
|
|
||||||
return hx`<span class='files'>${torrent.files.length} files</span>`
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Download button toggles between torrenting (DL/seed) and paused
|
// Foreground: name of the torrent, basic info like size, play button,
|
||||||
// Play button starts streaming the torrent immediately, unpausing if needed
|
// cast buttons if available, and delete
|
||||||
function renderTorrentButtons (torrentSummary, dispatch) {
|
var classes = ['torrent']
|
||||||
return hx`
|
// playStatus turns the play button into a loading spinner or error icon
|
||||||
<div class='buttons'>
|
if (torrentSummary.playStatus) classes.push(torrentSummary.playStatus)
|
||||||
<i.btn.icon.download
|
if (isSelected) classes.push('selected')
|
||||||
class='${torrentSummary.status}'
|
classes = classes.join(' ')
|
||||||
onclick=${(e) => handleButton('toggleTorrent', e)}>
|
return hx`
|
||||||
${torrentSummary.status === 'seeding' ? 'file_upload' : 'file_download'}
|
<div style=${style} class=${classes} onclick=${() => dispatch('toggleSelectTorrent', infoHash)}>
|
||||||
</i>
|
${renderTorrentMetadata(torrent, torrentSummary)}
|
||||||
<i.btn.icon.play
|
${renderTorrentButtons(torrentSummary)}
|
||||||
onclick=${(e) => handleButton('openPlayer', e)}>
|
${isSelected ? renderTorrentDetails(torrent, torrentSummary) : ''}
|
||||||
${torrentSummary.playStatus === 'timeout' ? 'warning' : 'play_arrow'}
|
|
||||||
</i>
|
|
||||||
<i
|
|
||||||
class='icon delete'
|
|
||||||
onclick=${(e) => handleButton('deleteTorrent', e)}>
|
|
||||||
close
|
|
||||||
</i>
|
|
||||||
</div>
|
|
||||||
`
|
|
||||||
|
|
||||||
function handleButton (action, e) {
|
|
||||||
// Prevent propagation so that we don't select/unselect the torrent
|
|
||||||
e.stopPropagation()
|
|
||||||
dispatch(action, torrentSummary)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show files, per-file download status and play buttons, and so on
|
|
||||||
function renderTorrentDetails (torrent, torrentSummary) {
|
|
||||||
var filesElement
|
|
||||||
if (!torrent || !torrent.files) {
|
|
||||||
// We don't know what files this torrent contains
|
|
||||||
var message = torrent
|
|
||||||
? 'Downloading torrent data using magnet link...'
|
|
||||||
: 'Failed to download torrent data from magnet link. Click the download button to try again...'
|
|
||||||
filesElement = hx`<div class='files warning'>${message}</div>`
|
|
||||||
} else {
|
|
||||||
// We do know the files. List them and show download stats for each one
|
|
||||||
var fileRows = torrent.files.map(function (file) {
|
|
||||||
var numPieces = file._endPiece - file._startPiece + 1
|
|
||||||
var numPiecesPresent = 0
|
|
||||||
for (var piece = file._startPiece; piece <= file._endPiece; piece++) {
|
|
||||||
if (torrent.bitfield.get(piece)) numPiecesPresent++
|
|
||||||
}
|
|
||||||
var progress = Math.round(100 * numPiecesPresent / numPieces) + '%'
|
|
||||||
return hx`
|
|
||||||
<tr>
|
|
||||||
<td class='col-name'>${file.name}</td>
|
|
||||||
<td class='col-progress'>${progress}</td>
|
|
||||||
<td class='col-size'>${prettyBytes(file.length)}</td>
|
|
||||||
</li>
|
|
||||||
`
|
|
||||||
})
|
|
||||||
filesElement = hx`
|
|
||||||
<div class='files'>
|
|
||||||
<strong>Files</strong>
|
|
||||||
<table>
|
|
||||||
${fileRows}
|
|
||||||
</table>
|
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
}
|
}
|
||||||
|
|
||||||
return hx`
|
// Show name, download status, % complete
|
||||||
<div class='torrent-details'>
|
function renderTorrentMetadata (torrent, torrentSummary) {
|
||||||
${filesElement}
|
var name = torrentSummary.displayName || torrentSummary.name || 'Loading torrent...'
|
||||||
</div>
|
var elements = [hx`
|
||||||
`
|
<div class='name ellipsis'>${name}</div>
|
||||||
|
`]
|
||||||
|
|
||||||
|
// If a torrent is paused and we only get the torrentSummary
|
||||||
|
// If it's downloading/seeding then we have more information
|
||||||
|
if (torrent) {
|
||||||
|
var progress = Math.floor(100 * torrent.progress)
|
||||||
|
var downloaded = prettyBytes(torrent.downloaded)
|
||||||
|
var total = prettyBytes(torrent.length || 0)
|
||||||
|
if (downloaded !== total) downloaded += ` / ${total}`
|
||||||
|
|
||||||
|
elements.push(hx`
|
||||||
|
<div class='status ellipsis'>
|
||||||
|
${getFilesLength()}
|
||||||
|
<span>${getPeers()}</span>
|
||||||
|
<span>↓ ${prettyBytes(torrent.downloadSpeed || 0)}/s</span>
|
||||||
|
<span>↑ ${prettyBytes(torrent.uploadSpeed || 0)}/s</span>
|
||||||
|
</div>
|
||||||
|
`)
|
||||||
|
elements.push(hx`
|
||||||
|
<div class='status2 ellipsis'>
|
||||||
|
<span class='progress'>${progress}%</span>
|
||||||
|
<span>${downloaded}</span>
|
||||||
|
</div>
|
||||||
|
`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return hx`<div class='metadata'>${elements}</div>`
|
||||||
|
|
||||||
|
function getPeers () {
|
||||||
|
var count = torrent.numPeers === 1 ? 'peer' : 'peers'
|
||||||
|
return `${torrent.numPeers} ${count}`
|
||||||
|
}
|
||||||
|
|
||||||
|
function getFilesLength () {
|
||||||
|
if (torrent.ready && torrent.files.length > 1) {
|
||||||
|
return hx`<span class='files'>${torrent.files.length} files</span>`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Download button toggles between torrenting (DL/seed) and paused
|
||||||
|
// Play button starts streaming the torrent immediately, unpausing if needed
|
||||||
|
function renderTorrentButtons (torrentSummary) {
|
||||||
|
return hx`
|
||||||
|
<div class='buttons'>
|
||||||
|
<i.btn.icon.play
|
||||||
|
onclick=${(e) => handleButton('play', e)}>
|
||||||
|
${torrentSummary.playStatus === 'timeout' ? 'warning' : 'play_arrow'}
|
||||||
|
</i>
|
||||||
|
<i.btn.icon.download
|
||||||
|
class='${torrentSummary.status}'
|
||||||
|
onclick=${(e) => handleButton('toggleTorrent', e)}>
|
||||||
|
${torrentSummary.status === 'seeding' ? 'file_upload' : 'file_download'}
|
||||||
|
</i>
|
||||||
|
<i
|
||||||
|
class='icon delete'
|
||||||
|
onclick=${(e) => handleButton('deleteTorrent', e)}>
|
||||||
|
close
|
||||||
|
</i>
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
|
||||||
|
function handleButton (action, e) {
|
||||||
|
// Prevent propagation so that we don't select/unselect the torrent
|
||||||
|
e.stopPropagation()
|
||||||
|
dispatch(action, torrentSummary)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show files, per-file download status and play buttons, and so on
|
||||||
|
function renderTorrentDetails (torrent, torrentSummary) {
|
||||||
|
var filesElement
|
||||||
|
if (!torrent || !torrent.files) {
|
||||||
|
// We don't know what files this torrent contains
|
||||||
|
var message = torrent
|
||||||
|
? 'Downloading torrent data using magnet link...'
|
||||||
|
: 'Failed to download torrent data from magnet link. Click the download button to try again...'
|
||||||
|
filesElement = hx`<div class='files warning'>${message}</div>`
|
||||||
|
} else {
|
||||||
|
// We do know the files. List them and show download stats for each one
|
||||||
|
var fileRows = torrent.files.map((file, index) => renderFileRow(torrent, torrentSummary, file, index))
|
||||||
|
filesElement = hx`
|
||||||
|
<div class='files'>
|
||||||
|
<strong>Files</strong>
|
||||||
|
<span class='open-folder' onclick=${handleOpenFolder}>Open folder</span>
|
||||||
|
<table>
|
||||||
|
${fileRows}
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
}
|
||||||
|
|
||||||
|
return hx`
|
||||||
|
<div class='torrent-details'>
|
||||||
|
${filesElement}
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
|
||||||
|
function handleOpenFolder (e) {
|
||||||
|
e.stopPropagation()
|
||||||
|
dispatch('openFolder', torrentSummary)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show a single file in the details view for a single torrent
|
||||||
|
function renderFileRow (torrent, torrentSummary, file, index) {
|
||||||
|
// First, find out how much of the file we've downloaded
|
||||||
|
var numPieces = file._endPiece - file._startPiece + 1
|
||||||
|
var numPiecesPresent = 0
|
||||||
|
for (var piece = file._startPiece; piece <= file._endPiece; piece++) {
|
||||||
|
if (torrent.bitfield.get(piece)) numPiecesPresent++
|
||||||
|
}
|
||||||
|
var progress = Math.round(100 * numPiecesPresent / numPieces) + '%'
|
||||||
|
var isDone = numPieces === numPiecesPresent
|
||||||
|
|
||||||
|
// Second, render the file as a table row
|
||||||
|
var icon
|
||||||
|
var iconClass = ''
|
||||||
|
if (state.playing.infoHash === torrent.infoHash && state.playing.fileIndex === index) {
|
||||||
|
icon = 'pause_arrow' /* playing? add option to pause */
|
||||||
|
} else if (TorrentPlayer.isPlayable(file)) {
|
||||||
|
icon = 'play_arrow' /* playable? add option to play */
|
||||||
|
} else {
|
||||||
|
icon = 'description' /* file icon, opens in OS default app */
|
||||||
|
iconClass = isDone ? '' : 'disabled'
|
||||||
|
}
|
||||||
|
return hx`
|
||||||
|
<tr onclick=${handleClick}>
|
||||||
|
<td class='col-icon'>
|
||||||
|
<i class='icon ${iconClass}'>${icon}</i>
|
||||||
|
</td>
|
||||||
|
<td class='col-name'>${file.name}</td>
|
||||||
|
<td class='col-progress'>${progress}</td>
|
||||||
|
<td class='col-size'>${prettyBytes(file.length)}</td>
|
||||||
|
</tr>
|
||||||
|
`
|
||||||
|
|
||||||
|
// Finally, let the user click on the row to play media or open files
|
||||||
|
function handleClick (e) {
|
||||||
|
e.stopPropagation()
|
||||||
|
if (icon === 'pause_arrow') {
|
||||||
|
throw new Error('Unimplemented') // TODO: pause audio
|
||||||
|
} else if (icon === 'play_arrow') {
|
||||||
|
dispatch('play', torrentSummary, index)
|
||||||
|
} else if (isDone) {
|
||||||
|
dispatch('openFile', torrentSummary, index)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user