Compare commits
10 Commits
m/mkv-subt
...
v0.13.0
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
74349129f4 | ||
|
|
cdc2c1d718 | ||
|
|
f8cc155650 | ||
|
|
b6bdeab50b | ||
|
|
f528f6033f | ||
|
|
ef1bc13c38 | ||
|
|
2f54feac74 | ||
|
|
75d30baaa5 | ||
|
|
990fb57839 | ||
|
|
1883341ddb |
@@ -1,4 +1,4 @@
|
|||||||
language: node_js
|
language: node_js
|
||||||
node_js:
|
node_js:
|
||||||
- 'node'
|
- 'node'
|
||||||
install: npm install standard
|
install: npm install standard depcheck
|
||||||
|
|||||||
@@ -1,108 +0,0 @@
|
|||||||
#!/usr/bin/env node
|
|
||||||
|
|
||||||
var fs = require('fs')
|
|
||||||
var cp = require('child_process')
|
|
||||||
|
|
||||||
// We can't use `builtin-modules` here since our TravisCI
|
|
||||||
// setup expects this file to run with no dependencies
|
|
||||||
var BUILT_IN_NODE_MODULES = [
|
|
||||||
'assert',
|
|
||||||
'buffer',
|
|
||||||
'child_process',
|
|
||||||
'cluster',
|
|
||||||
'console',
|
|
||||||
'constants',
|
|
||||||
'crypto',
|
|
||||||
'dgram',
|
|
||||||
'dns',
|
|
||||||
'domain',
|
|
||||||
'events',
|
|
||||||
'fs',
|
|
||||||
'http',
|
|
||||||
'https',
|
|
||||||
'module',
|
|
||||||
'net',
|
|
||||||
'os',
|
|
||||||
'path',
|
|
||||||
'process',
|
|
||||||
'punycode',
|
|
||||||
'querystring',
|
|
||||||
'readline',
|
|
||||||
'repl',
|
|
||||||
'stream',
|
|
||||||
'string_decoder',
|
|
||||||
'timers',
|
|
||||||
'tls',
|
|
||||||
'tty',
|
|
||||||
'url',
|
|
||||||
'util',
|
|
||||||
'v8',
|
|
||||||
'vm',
|
|
||||||
'zlib'
|
|
||||||
]
|
|
||||||
|
|
||||||
var BUILT_IN_ELECTRON_MODULES = [ 'electron' ]
|
|
||||||
|
|
||||||
var BUILT_IN_DEPS = [].concat(BUILT_IN_NODE_MODULES, BUILT_IN_ELECTRON_MODULES)
|
|
||||||
|
|
||||||
var EXECUTABLE_DEPS = [
|
|
||||||
'babel-cli',
|
|
||||||
'babel-plugin-syntax-jsx',
|
|
||||||
'babel-plugin-transform-es2015-destructuring',
|
|
||||||
'babel-plugin-transform-object-rest-spread',
|
|
||||||
'babel-plugin-transform-react-jsx',
|
|
||||||
'gh-release',
|
|
||||||
'nodemon',
|
|
||||||
'standard'
|
|
||||||
]
|
|
||||||
|
|
||||||
main()
|
|
||||||
|
|
||||||
// Scans codebase for missing or unused dependencies. Exits with code 0 on success.
|
|
||||||
function main () {
|
|
||||||
if (process.platform === 'win32') {
|
|
||||||
console.error('Sorry, check-deps only works on Mac and Linux')
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var usedDeps = findUsedDeps()
|
|
||||||
var packageDeps = findPackageDeps()
|
|
||||||
|
|
||||||
var missingDeps = usedDeps.filter(
|
|
||||||
(dep) => !includes(packageDeps, dep) && !includes(BUILT_IN_DEPS, dep)
|
|
||||||
)
|
|
||||||
var unusedDeps = packageDeps.filter(
|
|
||||||
(dep) => !includes(usedDeps, dep) && !includes(EXECUTABLE_DEPS, dep)
|
|
||||||
)
|
|
||||||
|
|
||||||
if (missingDeps.length > 0) {
|
|
||||||
console.error('Missing package dependencies: ' + missingDeps)
|
|
||||||
}
|
|
||||||
if (unusedDeps.length > 0) {
|
|
||||||
console.error('Unused package dependencies: ' + unusedDeps)
|
|
||||||
}
|
|
||||||
if (missingDeps.length + unusedDeps.length > 0) {
|
|
||||||
process.exitCode = 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Finds all dependencies specified in `package.json`
|
|
||||||
function findPackageDeps () {
|
|
||||||
var pkg = JSON.parse(fs.readFileSync('package.json', 'utf8'))
|
|
||||||
|
|
||||||
var deps = Object.keys(pkg.dependencies)
|
|
||||||
var devDeps = Object.keys(pkg.devDependencies)
|
|
||||||
var optionalDeps = Object.keys(pkg.optionalDependencies)
|
|
||||||
|
|
||||||
return [].concat(deps, devDeps, optionalDeps)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Finds all dependencies that used with `require()`
|
|
||||||
function findUsedDeps () {
|
|
||||||
var stdout = cp.execSync('./bin/list-deps.sh')
|
|
||||||
return stdout.toString().trim().split('\n')
|
|
||||||
}
|
|
||||||
|
|
||||||
function includes (arr, elem) {
|
|
||||||
return arr.indexOf(elem) >= 0
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
# This is a truly heinous hack, but it works pretty nicely.
|
|
||||||
# Find all modules we're requiring---even conditional requires.
|
|
||||||
|
|
||||||
grep "require('" src/ bin/ -R |
|
|
||||||
grep '.js:' |
|
|
||||||
sed "s/.*require('\([^'\/]*\).*/\1/" |
|
|
||||||
grep -v '^\.' |
|
|
||||||
sort |
|
|
||||||
uniq
|
|
||||||
@@ -7,4 +7,4 @@ npm run package -- --sign
|
|||||||
git push
|
git push
|
||||||
git push --tags
|
git push --tags
|
||||||
npm publish
|
npm publish
|
||||||
./node_modules/.bin/gh-release
|
npm run gh-release
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "webtorrent-desktop",
|
"name": "webtorrent-desktop",
|
||||||
"description": "WebTorrent, the streaming torrent client. For Mac, Windows, and Linux.",
|
"description": "WebTorrent, the streaming torrent client. For Mac, Windows, and Linux.",
|
||||||
"version": "0.12.0",
|
"version": "0.13.0",
|
||||||
"author": {
|
"author": {
|
||||||
"name": "WebTorrent, LLC",
|
"name": "WebTorrent, LLC",
|
||||||
"email": "feross@webtorrent.io",
|
"email": "feross@webtorrent.io",
|
||||||
@@ -52,6 +52,7 @@
|
|||||||
"babel-plugin-transform-object-rest-spread": "^6.8.0",
|
"babel-plugin-transform-object-rest-spread": "^6.8.0",
|
||||||
"babel-plugin-transform-react-jsx": "^6.8.0",
|
"babel-plugin-transform-react-jsx": "^6.8.0",
|
||||||
"cross-zip": "^2.0.1",
|
"cross-zip": "^2.0.1",
|
||||||
|
"depcheck": "^0.6.4",
|
||||||
"electron-osx-sign": "^0.3.0",
|
"electron-osx-sign": "^0.3.0",
|
||||||
"electron-packager": "^7.0.0",
|
"electron-packager": "^7.0.0",
|
||||||
"electron-winstaller": "^2.3.0",
|
"electron-winstaller": "^2.3.0",
|
||||||
@@ -97,7 +98,8 @@
|
|||||||
"package": "node ./bin/package.js",
|
"package": "node ./bin/package.js",
|
||||||
"prepublish": "npm run build",
|
"prepublish": "npm run build",
|
||||||
"start": "npm run build && electron .",
|
"start": "npm run build && electron .",
|
||||||
"test": "standard && node ./bin/check-deps.js",
|
"test": "standard && depcheck --ignores=babel-cli,nodemon,gh-release --ignore-dirs=build,dist",
|
||||||
|
"gh-release": "gh-release",
|
||||||
"update-authors": "./bin/update-authors.sh",
|
"update-authors": "./bin/update-authors.sh",
|
||||||
"watch": "nodemon --exec 'npm run start' --ext js,pug,css"
|
"watch": "nodemon --exec 'npm run start' --ext js,pug,css"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,14 +39,11 @@ class ShowMore extends React.Component {
|
|||||||
? this.props.hideLabel
|
? this.props.hideLabel
|
||||||
: this.props.showLabel
|
: this.props.showLabel
|
||||||
return (
|
return (
|
||||||
<div
|
<div style={this.props.style}>
|
||||||
style={this.props.style}
|
|
||||||
>
|
|
||||||
{this.state.expanded ? this.props.children : null}
|
{this.state.expanded ? this.props.children : null}
|
||||||
<FlatButton
|
<FlatButton
|
||||||
onClick={this.handleClick}
|
onClick={this.handleClick}
|
||||||
label={label}
|
label={label} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -126,7 +126,9 @@ function logUncaughtError (procName, e) {
|
|||||||
|
|
||||||
var message
|
var message
|
||||||
var stack = ''
|
var stack = ''
|
||||||
if (e.message) {
|
if (e == null) {
|
||||||
|
message = 'Unexpected undefined error'
|
||||||
|
} else if (e.message) {
|
||||||
// err is either an Error or a plain object {message, stack}
|
// err is either an Error or a plain object {message, stack}
|
||||||
message = e.message
|
message = e.message
|
||||||
stack = e.stack
|
stack = e.stack
|
||||||
@@ -149,11 +151,13 @@ function logUncaughtError (procName, e) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (typeof stack !== 'string') stack = 'Unexpected stack: ' + stack
|
||||||
|
if (typeof message !== 'string') message = 'Unexpected message: ' + message
|
||||||
|
|
||||||
// Remove the first part of each file path in the stack trace.
|
// Remove the first part of each file path in the stack trace.
|
||||||
// - Privacy: remove personal info like C:\Users\<full name>
|
// - Privacy: remove personal info like C:\Users\<full name>
|
||||||
// - Aggregation: this lets us find which stacktraces occur often
|
// - Aggregation: this lets us find which stacktraces occur often
|
||||||
if (stack && typeof stack === 'string') stack = stack.replace(/\(.*app.asar/g, '(...')
|
stack = stack.replace(/\(.*app.asar/g, '(...')
|
||||||
else if (stack) stack = 'Unexpected stack: ' + stack
|
|
||||||
|
|
||||||
// We need to POST the telemetry object, make sure it stays < 100kb
|
// We need to POST the telemetry object, make sure it stays < 100kb
|
||||||
if (telemetry.uncaughtErrors.length > 20) return
|
if (telemetry.uncaughtErrors.length > 20) return
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ const {dispatch, dispatcher} = require('../lib/dispatcher')
|
|||||||
const FlatButton = require('material-ui/FlatButton').default
|
const FlatButton = require('material-ui/FlatButton').default
|
||||||
const RaisedButton = require('material-ui/RaisedButton').default
|
const RaisedButton = require('material-ui/RaisedButton').default
|
||||||
const TextField = require('material-ui/TextField').default
|
const TextField = require('material-ui/TextField').default
|
||||||
|
const Checkbox = require('material-ui/Checkbox').default
|
||||||
|
|
||||||
const CreateTorrentErrorPage = require('../components/create-torrent-error-page')
|
const CreateTorrentErrorPage = require('../components/create-torrent-error-page')
|
||||||
const Heading = require('../components/Heading')
|
const Heading = require('../components/Heading')
|
||||||
@@ -20,27 +21,20 @@ class CreateTorrentPage extends React.Component {
|
|||||||
var state = this.props.state
|
var state = this.props.state
|
||||||
var info = state.location.current()
|
var info = state.location.current()
|
||||||
|
|
||||||
// Preprocess: exclude .DS_Store and other dotfiles
|
|
||||||
var files = info.files
|
|
||||||
.filter((f) => !f.name.startsWith('.'))
|
|
||||||
.map((f) => ({name: f.name, path: f.path, size: f.size}))
|
|
||||||
if (files.length === 0) return (<CreateTorrentErrorPage state={state} />)
|
|
||||||
|
|
||||||
// First, extract the base folder that the files are all in
|
// First, extract the base folder that the files are all in
|
||||||
var pathPrefix = info.folderPath
|
var pathPrefix = info.folderPath
|
||||||
if (!pathPrefix) {
|
if (!pathPrefix) {
|
||||||
pathPrefix = files.map((x) => x.path).reduce(findCommonPrefix)
|
pathPrefix = info.files.map((x) => x.path).reduce(findCommonPrefix)
|
||||||
if (!pathPrefix.endsWith('/') && !pathPrefix.endsWith('\\')) {
|
if (!pathPrefix.endsWith('/') && !pathPrefix.endsWith('\\')) {
|
||||||
pathPrefix = path.dirname(pathPrefix)
|
pathPrefix = path.dirname(pathPrefix)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sanity check: show the number of files and total size
|
// Then, exclude .DS_Store and other dotfiles
|
||||||
var numFiles = files.length
|
var files = info.files
|
||||||
var totalBytes = files
|
.filter((f) => !containsDots(f.path, pathPrefix))
|
||||||
.map((f) => f.size)
|
.map((f) => ({name: f.name, path: f.path, size: f.size}))
|
||||||
.reduce((a, b) => a + b, 0)
|
if (files.length === 0) return (<CreateTorrentErrorPage state={state} />)
|
||||||
var torrentInfo = `${numFiles} files, ${prettyBytes(totalBytes)}`
|
|
||||||
|
|
||||||
// Then, use the name of the base folder (or sole file, for a single file torrent)
|
// Then, use the name of the base folder (or sole file, for a single file torrent)
|
||||||
// as the default name. Show all files relative to the base folder.
|
// as the default name. Show all files relative to the base folder.
|
||||||
@@ -54,90 +48,70 @@ class CreateTorrentPage extends React.Component {
|
|||||||
defaultName = path.basename(pathPrefix)
|
defaultName = path.basename(pathPrefix)
|
||||||
basePath = path.dirname(pathPrefix)
|
basePath = path.dirname(pathPrefix)
|
||||||
}
|
}
|
||||||
var maxFileElems = 100
|
|
||||||
var fileElems = files.slice(0, maxFileElems).map(function (file, i) {
|
// Default trackers
|
||||||
var relativePath = files.length === 0 ? file.name : path.relative(pathPrefix, file.path)
|
|
||||||
return (<div key={i}>{relativePath}</div>)
|
|
||||||
})
|
|
||||||
if (files.length > maxFileElems) {
|
|
||||||
fileElems.push(<div key='more'>+ {maxFileElems - files.length} more</div>)
|
|
||||||
}
|
|
||||||
var trackers = createTorrent.announceList.join('\n')
|
var trackers = createTorrent.announceList.join('\n')
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
|
comment: '',
|
||||||
|
isPrivate: false,
|
||||||
|
pathPrefix,
|
||||||
basePath,
|
basePath,
|
||||||
defaultName,
|
defaultName,
|
||||||
fileElems,
|
files,
|
||||||
torrentInfo,
|
|
||||||
trackers
|
trackers
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create React event handlers only once
|
||||||
|
this.setIsPrivate = (_, isPrivate) => this.setState({isPrivate})
|
||||||
|
this.setComment = (_, comment) => this.setState({comment})
|
||||||
|
this.setTrackers = (_, trackers) => this.setState({trackers})
|
||||||
|
this.handleSubmit = () => this.handleSubmit
|
||||||
}
|
}
|
||||||
|
|
||||||
handleSubmit () {
|
handleSubmit () {
|
||||||
var announceList = document.querySelector('.torrent-trackers').value
|
var announceList = this.state.trackers
|
||||||
.split('\n')
|
.split('\n')
|
||||||
.map((s) => s.trim())
|
.map((s) => s.trim())
|
||||||
.filter((s) => s !== '')
|
.filter((s) => s !== '')
|
||||||
var isPrivate = document.querySelector('.torrent-is-private').checked
|
|
||||||
var comment = document.querySelector('.torrent-comment').value.trim()
|
|
||||||
var options = {
|
var options = {
|
||||||
// We can't let the user choose their own name if we want WebTorrent
|
// We can't let the user choose their own name if we want WebTorrent
|
||||||
// to use the files in place rather than creating a new folder.
|
// to use the files in place rather than creating a new folder.
|
||||||
// If we ever want to add support for that:
|
|
||||||
// name: document.querySelector('.torrent-name').value
|
|
||||||
name: this.state.defaultName,
|
name: this.state.defaultName,
|
||||||
path: this.state.basePath,
|
path: this.state.basePath,
|
||||||
files: this.state.files,
|
files: this.state.files,
|
||||||
announce: announceList,
|
announce: announceList,
|
||||||
private: isPrivate,
|
private: this.state.isPrivate,
|
||||||
comment: comment
|
comment: this.state.comment.trim()
|
||||||
}
|
}
|
||||||
dispatch('createTorrent', options)
|
dispatch('createTorrent', options)
|
||||||
}
|
}
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
|
var files = this.state.files
|
||||||
|
|
||||||
|
// Sanity check: show the number of files and total size
|
||||||
|
var numFiles = files.length
|
||||||
|
var totalBytes = files
|
||||||
|
.map((f) => f.size)
|
||||||
|
.reduce((a, b) => a + b, 0)
|
||||||
|
var torrentInfo = `${numFiles} files, ${prettyBytes(totalBytes)}`
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='create-torrent'>
|
<div className='create-torrent'>
|
||||||
<Heading level={1}>
|
<Heading level={1}>Create torrent {this.state.defaultName}</Heading>
|
||||||
Create torrent "{this.state.defaultName}"
|
<div className='torrent-info'>{torrentInfo}</div>
|
||||||
</Heading>
|
|
||||||
<div className='torrent-info'>
|
|
||||||
{this.state.torrentInfo}
|
|
||||||
</div>
|
|
||||||
<div className='torrent-attribute'>
|
<div className='torrent-attribute'>
|
||||||
<label>Path:</label>
|
<label>Path:</label>
|
||||||
<div className='torrent-attribute'>{this.state.pathPrefix}</div>
|
<div>{this.state.pathPrefix}</div>
|
||||||
</div>
|
</div>
|
||||||
<ShowMore
|
<ShowMore
|
||||||
style={{
|
style={{
|
||||||
marginBottom: 10
|
marginBottom: 10
|
||||||
}}
|
}}
|
||||||
hideLabel='Hide advanced settings...'
|
hideLabel='Hide advanced settings...'
|
||||||
showLabel='Show advanced settings...'
|
showLabel='Show advanced settings...' >
|
||||||
>
|
{this.renderAdvanced()}
|
||||||
<div key='advanced' className='create-torrent-advanced'>
|
|
||||||
<div key='private' className='torrent-attribute'>
|
|
||||||
<label>Private:</label>
|
|
||||||
<input type='checkbox' className='torrent-is-private' value='torrent-is-private' />
|
|
||||||
</div>
|
|
||||||
<Heading level={2}>Trackers:</Heading>
|
|
||||||
<TextField
|
|
||||||
className='torrent-trackers'
|
|
||||||
hintText='Tracker'
|
|
||||||
multiLine
|
|
||||||
rows={2}
|
|
||||||
rowsMax={10}
|
|
||||||
defaultValue={this.state.trackers}
|
|
||||||
/>
|
|
||||||
<div key='comment' className='torrent-attribute'>
|
|
||||||
<label>Comment:</label>
|
|
||||||
<textarea className='torrent-attribute torrent-comment' />
|
|
||||||
</div>
|
|
||||||
<div key='files' className='torrent-attribute'>
|
|
||||||
<label>Files:</label>
|
|
||||||
<div>{this.state.fileElems}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</ShowMore>
|
</ShowMore>
|
||||||
<div className='float-right'>
|
<div className='float-right'>
|
||||||
<FlatButton
|
<FlatButton
|
||||||
@@ -156,6 +130,65 @@ class CreateTorrentPage extends React.Component {
|
|||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
renderAdvanced () {
|
||||||
|
// Create file list
|
||||||
|
var maxFileElems = 100
|
||||||
|
var files = this.state.files
|
||||||
|
var fileElems = files.slice(0, maxFileElems).map((file, i) => {
|
||||||
|
var relativePath = path.relative(this.state.pathPrefix, file.path)
|
||||||
|
return (<div key={i}>{relativePath}</div>)
|
||||||
|
})
|
||||||
|
if (files.length > maxFileElems) {
|
||||||
|
fileElems.push(<div key='more'>+ {files.length - maxFileElems} more</div>)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Align the text fields
|
||||||
|
var textFieldStyle = { width: '' }
|
||||||
|
var textareaStyle = { margin: 0 }
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div key='advanced' className='create-torrent-advanced'>
|
||||||
|
<div key='private' className='torrent-attribute'>
|
||||||
|
<label>Private:</label>
|
||||||
|
<Checkbox
|
||||||
|
className='torrent-is-private'
|
||||||
|
style={{display: ''}}
|
||||||
|
value={this.state.isPrivate}
|
||||||
|
onChange={this.setIsPrivate} />
|
||||||
|
</div>
|
||||||
|
<div key='trackers' className='torrent-attribute'>
|
||||||
|
<label>Trackers:</label>
|
||||||
|
<TextField
|
||||||
|
className='torrent-trackers'
|
||||||
|
style={textFieldStyle}
|
||||||
|
textareaStyle={textareaStyle}
|
||||||
|
multiLine
|
||||||
|
rows={2}
|
||||||
|
rowsMax={10}
|
||||||
|
value={this.state.trackers}
|
||||||
|
onChange={this.setTrackers} />
|
||||||
|
</div>
|
||||||
|
<div key='comment' className='torrent-attribute'>
|
||||||
|
<label>Comment:</label>
|
||||||
|
<TextField
|
||||||
|
className='torrent-comment'
|
||||||
|
style={textFieldStyle}
|
||||||
|
textareaStyle={textareaStyle}
|
||||||
|
hintText='Optionally describe your torrent...'
|
||||||
|
multiLine
|
||||||
|
rows={2}
|
||||||
|
rowsMax={10}
|
||||||
|
value={this.state.comment}
|
||||||
|
onChange={this.setComment} />
|
||||||
|
</div>
|
||||||
|
<div key='files' className='torrent-attribute'>
|
||||||
|
<label>Files:</label>
|
||||||
|
<div>{fileElems}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Finds the longest common prefix
|
// Finds the longest common prefix
|
||||||
@@ -168,4 +201,10 @@ function findCommonPrefix (a, b) {
|
|||||||
return a.substring(0, i)
|
return a.substring(0, i)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function containsDots (path, pathPrefix) {
|
||||||
|
var suffix = path.substring(pathPrefix.length).replace(/\\/g, '/')
|
||||||
|
console.log('SUFFIX ' + suffix)
|
||||||
|
return ('/' + suffix).includes('/.')
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = CreateTorrentPage
|
module.exports = CreateTorrentPage
|
||||||
|
|||||||
@@ -96,6 +96,14 @@ class PreferencesPage extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
setDefaultAppButton () {
|
setDefaultAppButton () {
|
||||||
|
var isFileHandler = this.props.state.unsaved.prefs.isFileHandler
|
||||||
|
if (isFileHandler) {
|
||||||
|
return (
|
||||||
|
<Preference>
|
||||||
|
<p>WebTorrent is your default torrent app. Hooray!</p>
|
||||||
|
</Preference>
|
||||||
|
)
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<Preference>
|
<Preference>
|
||||||
<p>WebTorrent is not currently the default torrent app.</p>
|
<p>WebTorrent is not currently the default torrent app.</p>
|
||||||
@@ -109,20 +117,17 @@ class PreferencesPage extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
handleSetDefaultApp () {
|
handleSetDefaultApp () {
|
||||||
window.alert('TODO')
|
dispatch('updatePreferences', 'isFileHandler', true)
|
||||||
// var isFileHandler = state.unsaved.prefs.isFileHandler
|
|
||||||
// dispatch('updatePreferences', 'isFileHandler', !isFileHandler)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
|
var style = {
|
||||||
|
color: colors.grey400,
|
||||||
|
marginLeft: 25,
|
||||||
|
marginRight: 25
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<div
|
<div style={style} >
|
||||||
style={{
|
|
||||||
color: colors.grey400,
|
|
||||||
marginLeft: 25,
|
|
||||||
marginRight: 25
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<PreferencesSection title='Downloads'>
|
<PreferencesSection title='Downloads'>
|
||||||
{this.downloadPathSelector()}
|
{this.downloadPathSelector()}
|
||||||
</PreferencesSection>
|
</PreferencesSection>
|
||||||
@@ -146,13 +151,12 @@ class PreferencesSection extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render () {
|
render () {
|
||||||
|
var style = {
|
||||||
|
marginBottom: 25,
|
||||||
|
marginTop: 25
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<div
|
<div style={style}>
|
||||||
style={{
|
|
||||||
marginBottom: 25,
|
|
||||||
marginTop: 25
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Heading level={2}>{this.props.title}</Heading>
|
<Heading level={2}>{this.props.title}</Heading>
|
||||||
{this.props.children}
|
{this.props.children}
|
||||||
</div>
|
</div>
|
||||||
@@ -162,15 +166,8 @@ class PreferencesSection extends React.Component {
|
|||||||
|
|
||||||
class Preference extends React.Component {
|
class Preference extends React.Component {
|
||||||
render () {
|
render () {
|
||||||
return (
|
var style = { marginBottom: 10 }
|
||||||
<div
|
return (<div style={style}>{this.props.children}</div>)
|
||||||
style={{
|
|
||||||
marginBottom: 10
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{this.props.children}
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -272,10 +272,12 @@ table {
|
|||||||
.create-torrent {
|
.create-torrent {
|
||||||
padding: 10px 25px;
|
padding: 10px 25px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
font: 16px/24px BlinkMacSystemFont, "Helvetica Neue", Helvetica, sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-torrent .torrent-attribute {
|
.create-torrent .torrent-attribute {
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
margin: 8px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-torrent .torrent-attribute>* {
|
.create-torrent .torrent-attribute>* {
|
||||||
@@ -283,13 +285,12 @@ table {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.create-torrent .torrent-attribute label {
|
.create-torrent .torrent-attribute label {
|
||||||
width: 60px;
|
width: 100px;
|
||||||
margin-right: 10px;
|
|
||||||
vertical-align: top;
|
vertical-align: top;
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-torrent .torrent-attribute>div {
|
.create-torrent .torrent-attribute>div {
|
||||||
width: calc(100% - 90px);
|
width: calc(100% - 100px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-torrent .torrent-attribute div {
|
.create-torrent .torrent-attribute div {
|
||||||
@@ -298,18 +299,6 @@ table {
|
|||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-torrent .torrent-attribute textarea {
|
|
||||||
width: calc(100% - 80px);
|
|
||||||
height: 80px;
|
|
||||||
color: #eee;
|
|
||||||
background-color: transparent;
|
|
||||||
line-height: 1.5;
|
|
||||||
font-size: 14px;
|
|
||||||
font-family: inherit;
|
|
||||||
border-radius: 2px;
|
|
||||||
padding: 4px 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* BUTTONS
|
* BUTTONS
|
||||||
*/
|
*/
|
||||||
|
|||||||
Reference in New Issue
Block a user