Compare commits
1 Commits
fix-startu
...
redownload
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fb33d569ca |
2
.github/FUNDING.yml
vendored
2
.github/FUNDING.yml
vendored
@@ -1,2 +0,0 @@
|
|||||||
github: feross
|
|
||||||
custom: https://paypal.me/borewit
|
|
||||||
4
.github/config.yml
vendored
4
.github/config.yml
vendored
@@ -13,3 +13,7 @@ newPRWelcomeComment: >
|
|||||||
firstPRMergeComment: >
|
firstPRMergeComment: >
|
||||||
🎉 Congrats on getting your first pull request landed! <br><br>
|
🎉 Congrats on getting your first pull request landed! <br><br>
|
||||||

|

|
||||||
|
|
||||||
|
# ProBot WIP (https://probot.github.io/apps/wip/)
|
||||||
|
|
||||||
|
# (Requires no configuration)
|
||||||
|
|||||||
2
.github/lock.yml
vendored
2
.github/lock.yml
vendored
@@ -1,4 +1,4 @@
|
|||||||
# ProBot Lock (https://probot.github.io/apps/lock/)
|
# ProBot Lock (https://probot.github.io/apps/lock/)
|
||||||
|
|
||||||
daysUntilLock: 365
|
daysUntilLock: 90
|
||||||
lockComment: false
|
lockComment: false
|
||||||
|
|||||||
2
.github/no-response.yml
vendored
2
.github/no-response.yml
vendored
@@ -1,6 +1,6 @@
|
|||||||
# ProBot No Response (https://probot.github.io/apps/no-response/)
|
# ProBot No Response (https://probot.github.io/apps/no-response/)
|
||||||
|
|
||||||
daysUntilClose: 14
|
daysUntilClose: 7
|
||||||
responseRequiredLabel: 'need more info'
|
responseRequiredLabel: 'need more info'
|
||||||
closeComment: >
|
closeComment: >
|
||||||
This issue has been automatically closed because there was no response to a
|
This issue has been automatically closed because there was no response to a
|
||||||
|
|||||||
5
.github/stale.yml
vendored
5
.github/stale.yml
vendored
@@ -1,14 +1,11 @@
|
|||||||
# ProBot Stale (https://probot.github.io/apps/stale/)
|
# ProBot Stale (https://probot.github.io/apps/stale/)
|
||||||
|
|
||||||
daysUntilStale: 90
|
daysUntilStale: 90
|
||||||
daysUntilClose: 14
|
daysUntilClose: 7
|
||||||
exemptLabels:
|
exemptLabels:
|
||||||
- accepted
|
- accepted
|
||||||
- blocked
|
|
||||||
- bug
|
- bug
|
||||||
- enhancement
|
- enhancement
|
||||||
- greenkeeper
|
|
||||||
- 'help wanted'
|
|
||||||
- meta
|
- meta
|
||||||
staleLabel: stale
|
staleLabel: stale
|
||||||
markComment: >
|
markComment: >
|
||||||
|
|||||||
19
AUTHORS.md
19
AUTHORS.md
@@ -43,28 +43,9 @@
|
|||||||
- Borewit (borewit@users.noreply.github.com)
|
- Borewit (borewit@users.noreply.github.com)
|
||||||
- greenkeeper[bot] (greenkeeper[bot]@users.noreply.github.com)
|
- greenkeeper[bot] (greenkeeper[bot]@users.noreply.github.com)
|
||||||
- Auyer (rafa_auyer@icloud.com)
|
- Auyer (rafa_auyer@icloud.com)
|
||||||
- Jon Koops (jonkoops@gmail.com)
|
|
||||||
- Michael George Attard (michaelgeorgeattard@gmail.com)
|
|
||||||
- SimplyAhmazing (ahmad19526@gmail.com)
|
- SimplyAhmazing (ahmad19526@gmail.com)
|
||||||
- Cezar Carneiro (cezargcarneiro@gmail.com)
|
- Cezar Carneiro (cezargcarneiro@gmail.com)
|
||||||
- Bilal Elmoussaoui (bil.elmoussaoui@gmail.com)
|
|
||||||
- Terry Hau (terryhau@gmail.com)
|
- Terry Hau (terryhau@gmail.com)
|
||||||
- Vítor Galvão (info@vitorgalvao.com)
|
- Vítor Galvão (info@vitorgalvao.com)
|
||||||
- Borewit (Borewit@users.noreply.github.com)
|
|
||||||
- Diego Rodríguez (diegorbaquero@gmail.com)
|
|
||||||
- Dan Flettre (flettre@gmail.com)
|
|
||||||
- Sibiraj (dev.sibiraj@outlook.com)
|
|
||||||
- clujin (clujin@gmail.com)
|
|
||||||
- Linus Unnebäck (linus@folkdatorn.se)
|
|
||||||
- Adrian Tombu (adrian@otso.fr)
|
|
||||||
- Lucas (5874806+RecoX@users.noreply.github.com)
|
|
||||||
- David Ernst (dsernst@users.noreply.github.com)
|
|
||||||
- David Ernst (git@dsernst.com)
|
|
||||||
- Jimmy Wärting (jimmy@warting.se)
|
|
||||||
- Recox (5874806+RecoX@users.noreply.github.com)
|
|
||||||
- greenkeeper[bot] (23040076+greenkeeper[bot]@users.noreply.github.com)
|
|
||||||
- hicom150 (hicom150@gmail.com)
|
|
||||||
- Jimmy Wärting (jimmy@warting.se)
|
|
||||||
- Feross (feross@feross.org)
|
|
||||||
|
|
||||||
#### Generated by bin/update-authors.sh.
|
#### Generated by bin/update-authors.sh.
|
||||||
|
|||||||
29
CHANGELOG.md
29
CHANGELOG.md
@@ -1,34 +1,5 @@
|
|||||||
# WebTorrent Desktop Version History
|
# WebTorrent Desktop Version History
|
||||||
|
|
||||||
## v0.21.0 - 2019-08
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Add YouTube style hotkeys [\#1579](https://github.com/webtorrent/webtorrent-desktop/pull/1579) ([dsernst](https://github.com/dsernst))
|
|
||||||
- Toggle sound notifications on/off [\#1536](https://github.com/webtorrent/webtorrent-desktop/pull/1536) ([adriantombu](https://github.com/adriantombu))
|
|
||||||
- Ability to play MPEG-4 Audio Book \(.m4b\) [\#1450](https://github.com/webtorrent/webtorrent-desktop/pull/1450) ([Borewit](https://github.com/Borewit))
|
|
||||||
- Add support for subtitles on Chromecast [\#1165](https://github.com/webtorrent/webtorrent-desktop/pull/1165) ([janza](https://github.com/janza))
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Update to Electron 4 [\#1590](https://github.com/webtorrent/webtorrent-desktop/pull/1590) ([Borewit](https://github.com/Borewit))
|
|
||||||
- Remove '\(BETA\)' from app window title [\#1562](https://github.com/webtorrent/webtorrent-desktop/pull/1562) ([dsernst](https://github.com/dsernst))
|
|
||||||
- Update React (v16) and Material-UI (v0.20) [\#1483](https://github.com/webtorrent/webtorrent-desktop/pull/1483) ([mathiasvr](https://github.com/mathiasvr))
|
|
||||||
- Show audio track and disk number [\#1454](https://github.com/webtorrent/webtorrent-desktop/pull/1454) ([Borewit](https://github.com/Borewit))
|
|
||||||
- Asynchronous music metadata updates while streaming [\#1449](https://github.com/webtorrent/webtorrent-desktop/pull/1449) ([Borewit](https://github.com/Borewit))
|
|
||||||
- If torrent is not private, leave private flag unset [\#1411](https://github.com/webtorrent/webtorrent-desktop/pull/1411) ([feross](https://github.com/feross))
|
|
||||||
- Improve audio poster selection: [\#1368](https://github.com/webtorrent/webtorrent-desktop/pull/1368) ([Borewit](https://github.com/Borewit))
|
|
||||||
- Save preferences immediately when changed [\#1042](https://github.com/webtorrent/webtorrent-desktop/pull/1042) ([Flet](https://github.com/Flet))
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Ensure that the minutes field in playback indicator is zero-padded. [\#1506](https://github.com/webtorrent/webtorrent-desktop/pull/1506) ([bnjmnt4n](https://github.com/bnjmnt4n))
|
|
||||||
- Bug Fix: Empty Array Reduce [\#1494](https://github.com/webtorrent/webtorrent-desktop/pull/1494) ([clujin](https://github.com/clujin))
|
|
||||||
- Fix startup problems [\#1419](https://github.com/webtorrent/webtorrent-desktop/pull/1419) ([Borewit](https://github.com/Borewit))
|
|
||||||
- Add back loading spinner for player page. [\#1311](https://github.com/webtorrent/webtorrent-desktop/pull/1311) ([bnjmnt4n](https://github.com/bnjmnt4n))
|
|
||||||
- Fix Linux desktop file [\#1309](https://github.com/webtorrent/webtorrent-desktop/pull/1309) ([bilelmoussaoui](https://github.com/bilelmoussaoui))
|
|
||||||
|
|
||||||
|
|
||||||
## v0.20.0 - 2018-04-26
|
## v0.20.0 - 2018-04-26
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|||||||
188
CONTRIBUTING.md
Normal file
188
CONTRIBUTING.md
Normal file
@@ -0,0 +1,188 @@
|
|||||||
|
# Contributing Guidelines
|
||||||
|
|
||||||
|
Contributions welcome!
|
||||||
|
|
||||||
|
**Before spending lots of time on something, ask for feedback on your idea first!**
|
||||||
|
|
||||||
|
Please search issues and pull requests before adding something new to avoid duplicating
|
||||||
|
efforts and conversations.
|
||||||
|
|
||||||
|
This project welcomes non-code contributions, too! The following types of contributions
|
||||||
|
are welcome:
|
||||||
|
|
||||||
|
- **Ideas**: participate in an issue thread or start your own to have your voice heard.
|
||||||
|
- **Writing**: contribute your expertise in an area by helping expand the included docs.
|
||||||
|
- **Copy editing**: fix typos, clarify language, and improve the quality of the docs.
|
||||||
|
- **Formatting**: help keep docs easy to read with consistent formatting.
|
||||||
|
|
||||||
|
## Code Style
|
||||||
|
|
||||||
|
[![standard][standard-image]][standard-url]
|
||||||
|
|
||||||
|
This repository uses [`standard`][standard-url] to maintain code style and consistency,
|
||||||
|
and to avoid style arguments. `npm test` runs `standard` automatically, so you don't have
|
||||||
|
to!
|
||||||
|
|
||||||
|
[standard-image]: https://cdn.rawgit.com/feross/standard/master/badge.svg
|
||||||
|
[standard-url]: https://standardjs.com
|
||||||
|
|
||||||
|
## Project Governance
|
||||||
|
|
||||||
|
Individuals making significant and valuable contributions are given commit-access to the
|
||||||
|
project to contribute as they see fit. This project is more like an open wiki than a
|
||||||
|
standard guarded open source project.
|
||||||
|
|
||||||
|
### Rules
|
||||||
|
|
||||||
|
There are a few basic ground-rules for contributors:
|
||||||
|
|
||||||
|
1. **No `--force` pushes to master** or modifying history in any way. Rebasing and force pushing your own PR branch is fine.
|
||||||
|
2. **Non-master branches** should be used for ongoing work.
|
||||||
|
3. **Significant modifications** like API changes should be subject to a **pull request**
|
||||||
|
to solicit feedback from other contributors.
|
||||||
|
4. **Pull requests** are *encouraged* for all contributions to solicit feedback, but left to
|
||||||
|
the discretion of the contributor.
|
||||||
|
|
||||||
|
### Releases
|
||||||
|
|
||||||
|
Declaring formal releases remains the prerogative of the project maintainer.
|
||||||
|
|
||||||
|
### Changes to this arrangement
|
||||||
|
|
||||||
|
This is an experiment and feedback is welcome! This document may also be subject to pull-
|
||||||
|
requests or changes by contributors where you believe you have something valuable to add
|
||||||
|
or change.
|
||||||
|
|
||||||
|
## Developer's Certificate of Origin 1.1
|
||||||
|
|
||||||
|
By making a contribution to this project, I certify that:
|
||||||
|
|
||||||
|
- (a) The contribution was created in whole or in part by me and I have the right to
|
||||||
|
submit it under the open source license indicated in the file; or
|
||||||
|
|
||||||
|
- (b) The contribution is based upon previous work that, to the best of my knowledge, is
|
||||||
|
covered under an appropriate open source license and I have the right under that license
|
||||||
|
to submit that work with modifications, whether created in whole or in part by me, under
|
||||||
|
the same open source license (unless I am permitted to submit under a different
|
||||||
|
license), as indicated in the file; or
|
||||||
|
|
||||||
|
- (c) The contribution was provided directly to me by some other person who certified
|
||||||
|
(a), (b) or (c) and I have not modified it.
|
||||||
|
|
||||||
|
- (d) I understand and agree that this project and the contribution are public and that a
|
||||||
|
record of the contribution (including all personal information I submit with it,
|
||||||
|
including my sign-off) is maintained indefinitely and may be redistributed consistent
|
||||||
|
with this project or the open source license(s) involved.
|
||||||
|
|
||||||
|
## Release Procedure
|
||||||
|
|
||||||
|
### 1. Create a new version
|
||||||
|
|
||||||
|
- Update `AUTHORS`
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run update-authors
|
||||||
|
```
|
||||||
|
|
||||||
|
Commit if necessary. The commit message should be "authors".
|
||||||
|
|
||||||
|
- Write the changelog
|
||||||
|
|
||||||
|
You can use `git log --oneline <last version tag>..HEAD` to get a list of changes.
|
||||||
|
|
||||||
|
Summarize them concisely in `CHANGELOG.md`. The commit message should be "changelog".
|
||||||
|
|
||||||
|
- Update the version
|
||||||
|
|
||||||
|
```
|
||||||
|
npm version [major|minor|patch]
|
||||||
|
```
|
||||||
|
|
||||||
|
This creates both a commit and a git tag.
|
||||||
|
|
||||||
|
- Make a PR
|
||||||
|
|
||||||
|
Once the PR is reviewed, merge it:
|
||||||
|
|
||||||
|
```
|
||||||
|
git push origin <branch-name>:master
|
||||||
|
```
|
||||||
|
|
||||||
|
This makes it so that the commit hash on master matches the commit hash of the version tag.
|
||||||
|
|
||||||
|
Finally, run:
|
||||||
|
|
||||||
|
```
|
||||||
|
git push --tags
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. Create the release binaries
|
||||||
|
|
||||||
|
- On a Mac:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run package -- darwin --sign
|
||||||
|
npm run package -- linux --sign
|
||||||
|
```
|
||||||
|
|
||||||
|
- On Windows, or in a Windows VM:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run package -- win32 --sign
|
||||||
|
```
|
||||||
|
|
||||||
|
- Then, upload the release binaries to Github:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run gh-release
|
||||||
|
```
|
||||||
|
|
||||||
|
Follow the URL to a newly created Github release page. Manually upload the binaries from
|
||||||
|
`webtorrent-desktop/dist/`. Open the previous release in another tab, and make sure that you
|
||||||
|
are uploading the same set of files, no more, no less.
|
||||||
|
|
||||||
|
### 3. Test it
|
||||||
|
|
||||||
|
**This is the most important part.**
|
||||||
|
|
||||||
|
- Manually download the binaries for each platform from Github.
|
||||||
|
|
||||||
|
**Do not use your locally built binaries.** Modern OSs treat executables differently if they've
|
||||||
|
been downloaded, even though the files are byte for byte identical. This ensures that the
|
||||||
|
codesigning worked and is valid.
|
||||||
|
|
||||||
|
- Smoke test WebTorrent Desktop on each platform.
|
||||||
|
|
||||||
|
See Smoke Tests below for details. Open DevTools
|
||||||
|
on Windows and Mac, and ensure that the auto updater is running. If the auto updater does not
|
||||||
|
run, users will successfully auto update to this new version, and then be stuck there forever.
|
||||||
|
|
||||||
|
### 4. Ship it
|
||||||
|
|
||||||
|
- Update the website
|
||||||
|
|
||||||
|
Create a pull request in [webtorrent.io](https://github.com/webtorrent/webtorrent.io). Update
|
||||||
|
`config.js`, updating the desktop app version.
|
||||||
|
|
||||||
|
As soon as this PR is merged, Jenkins will automatically redeploy the WebTorrent website, and
|
||||||
|
hundreds of thousands of users around the world will start auto updating. **Merge with care.**
|
||||||
|
|
||||||
|
## Smoke Tests
|
||||||
|
|
||||||
|
Before a release, check that the following basic use cases work correctly:
|
||||||
|
|
||||||
|
1. Click "Play" to stream a built-in torrent (e.g. Sintel)
|
||||||
|
- Ensure that seeking to undownloaded region works and plays immediately.
|
||||||
|
- Ensure that sintel.mp4 gets downloaded to `~/Downloads`.
|
||||||
|
|
||||||
|
2. Check that the auto-updater works
|
||||||
|
- Open the console and check for the line "No update available" to indicate
|
||||||
|
|
||||||
|
3. Add a new .torrent file via drag-and-drop.
|
||||||
|
- Ensure that it gets added to the list and starts downloading
|
||||||
|
|
||||||
|
4. Remove a torrent from the client
|
||||||
|
- Ensure that the file is removed from `~/Downloads`
|
||||||
|
|
||||||
|
5. Create and seed a new a torrent via drag-and-drop.
|
||||||
|
- Ensure that the torrent gets created and seeding begins.
|
||||||
14
README.md
14
README.md
@@ -12,7 +12,7 @@
|
|||||||
<h4 align="center">The streaming torrent app. For Mac, Windows, and Linux.</h4>
|
<h4 align="center">The streaming torrent app. For Mac, Windows, and Linux.</h4>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<a href="https://discord.gg/cnXkm4Z"><img src="https://img.shields.io/discord/612575111718895616" alt="discord"></a>
|
<a href="https://gitter.im/webtorrent/webtorrent"><img src="https://img.shields.io/badge/gitter-join%20chat%20%E2%86%92-brightgreen.svg" alt="gitter"></a>
|
||||||
<a href="https://travis-ci.org/webtorrent/webtorrent-desktop"><img src="https://img.shields.io/travis/webtorrent/webtorrent-desktop/master.svg" alt="travis"></a>
|
<a href="https://travis-ci.org/webtorrent/webtorrent-desktop"><img src="https://img.shields.io/travis/webtorrent/webtorrent-desktop/master.svg" alt="travis"></a>
|
||||||
<a href="https://github.com/webtorrent/webtorrent-desktop/releases"><img src="https://img.shields.io/github/release/webtorrent/webtorrent-desktop.svg" alt="github release version"></a>
|
<a href="https://github.com/webtorrent/webtorrent-desktop/releases"><img src="https://img.shields.io/github/release/webtorrent/webtorrent-desktop.svg" alt="github release version"></a>
|
||||||
<a href="https://github.com/webtorrent/webtorrent-desktop/releases"><img src="https://img.shields.io/github/downloads/webtorrent/webtorrent-desktop/total.svg" alt="github release downloads"></a>
|
<a href="https://github.com/webtorrent/webtorrent-desktop/releases"><img src="https://img.shields.io/github/downloads/webtorrent/webtorrent-desktop/total.svg" alt="github release downloads"></a>
|
||||||
@@ -158,18 +158,6 @@ The Mac app can only be packaged from **macOS**.
|
|||||||
|
|
||||||
The Linux app can be packaged from **any** platform.
|
The Linux app can be packaged from **any** platform.
|
||||||
|
|
||||||
|
|
||||||
#### Recommended readings to start working in the app
|
|
||||||
|
|
||||||
Electron (Framework to make native apps for Windows, OSX and Linux in Javascript):
|
|
||||||
https://electronjs.org/docs/tutorial/quick-start
|
|
||||||
|
|
||||||
React.js (Framework to work with Frontend UI):
|
|
||||||
https://reactjs.org/docs/getting-started.html
|
|
||||||
|
|
||||||
Material UI (React components that implement Google's Material Design.):
|
|
||||||
https://material-ui.com/getting-started
|
|
||||||
|
|
||||||
### Privacy
|
### Privacy
|
||||||
|
|
||||||
WebTorrent Desktop collects some basic usage stats to help us make the app better.
|
WebTorrent Desktop collects some basic usage stats to help us make the app better.
|
||||||
|
|||||||
@@ -1,112 +0,0 @@
|
|||||||
## Release Process
|
|
||||||
|
|
||||||
### 1. Create a new version
|
|
||||||
|
|
||||||
- Update `AUTHORS`
|
|
||||||
|
|
||||||
```
|
|
||||||
npm run update-authors
|
|
||||||
```
|
|
||||||
|
|
||||||
Commit if necessary. The commit message should be "authors".
|
|
||||||
|
|
||||||
- Write the changelog
|
|
||||||
|
|
||||||
You can use `git log --oneline <last version tag>..HEAD` to get a list of changes.
|
|
||||||
|
|
||||||
Summarize them concisely in `CHANGELOG.md`. The commit message should be "changelog".
|
|
||||||
|
|
||||||
- Update the version
|
|
||||||
|
|
||||||
```
|
|
||||||
npm version [major|minor|patch]
|
|
||||||
```
|
|
||||||
|
|
||||||
This creates both a commit and a git tag.
|
|
||||||
|
|
||||||
- Make a PR
|
|
||||||
|
|
||||||
Once the PR is reviewed, merge it:
|
|
||||||
|
|
||||||
```
|
|
||||||
git push origin <branch-name>:master
|
|
||||||
```
|
|
||||||
|
|
||||||
This makes it so that the commit hash on master matches the commit hash of the version tag.
|
|
||||||
|
|
||||||
Finally, run:
|
|
||||||
|
|
||||||
```
|
|
||||||
git push --tags
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Create the release binaries
|
|
||||||
|
|
||||||
- On a Mac:
|
|
||||||
|
|
||||||
```
|
|
||||||
npm run package -- darwin --sign
|
|
||||||
npm run package -- linux --sign
|
|
||||||
```
|
|
||||||
|
|
||||||
- On Windows, or in a Windows VM:
|
|
||||||
|
|
||||||
```
|
|
||||||
npm run package -- win32 --sign
|
|
||||||
```
|
|
||||||
|
|
||||||
- Then, upload the release binaries to Github:
|
|
||||||
|
|
||||||
```
|
|
||||||
npm run gh-release
|
|
||||||
```
|
|
||||||
|
|
||||||
Follow the URL to a newly created Github release page. Manually upload the binaries from
|
|
||||||
`webtorrent-desktop/dist/`. Open the previous release in another tab, and make sure that you
|
|
||||||
are uploading the same set of files, no more, no less.
|
|
||||||
|
|
||||||
### 3. Test it
|
|
||||||
|
|
||||||
**This is the most important part.**
|
|
||||||
|
|
||||||
- Manually download the binaries for each platform from Github.
|
|
||||||
|
|
||||||
**Do not use your locally built binaries.** Modern OSs treat executables differently if they've
|
|
||||||
been downloaded, even though the files are byte for byte identical. This ensures that the
|
|
||||||
codesigning worked and is valid.
|
|
||||||
|
|
||||||
- Smoke test WebTorrent Desktop on each platform.
|
|
||||||
|
|
||||||
See Smoke Tests below for details. Open DevTools
|
|
||||||
on Windows and Mac, and ensure that the auto updater is running. If the auto updater does not
|
|
||||||
run, users will successfully auto update to this new version, and then be stuck there forever.
|
|
||||||
|
|
||||||
### 4. Ship it
|
|
||||||
|
|
||||||
- Update the website
|
|
||||||
|
|
||||||
Create a pull request in [webtorrent.io](https://github.com/webtorrent/webtorrent.io). Update
|
|
||||||
`config.js`, updating the desktop app version.
|
|
||||||
|
|
||||||
As soon as this PR is merged, Jenkins will automatically redeploy the WebTorrent website, and
|
|
||||||
hundreds of thousands of users around the world will start auto updating. **Merge with care.**
|
|
||||||
|
|
||||||
## Smoke Tests
|
|
||||||
|
|
||||||
Before a release, check that the following basic use cases work correctly:
|
|
||||||
|
|
||||||
1. Click "Play" to stream a built-in torrent (e.g. Sintel)
|
|
||||||
- Ensure that seeking to undownloaded region works and plays immediately.
|
|
||||||
- Ensure that sintel.mp4 gets downloaded to `~/Downloads`.
|
|
||||||
|
|
||||||
2. Check that the auto-updater works
|
|
||||||
- Open the console and check for the line "No update available" to indicate
|
|
||||||
|
|
||||||
3. Add a new .torrent file via drag-and-drop.
|
|
||||||
- Ensure that it gets added to the list and starts downloading
|
|
||||||
|
|
||||||
4. Remove a torrent from the client
|
|
||||||
- Ensure that the file is removed from `~/Downloads`
|
|
||||||
|
|
||||||
5. Create and seed a new a torrent via drag-and-drop.
|
|
||||||
- Ensure that the torrent gets created and seeding begins.
|
|
||||||
43
SECURITY.md
Normal file
43
SECURITY.md
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
# Security Policies and Procedures
|
||||||
|
|
||||||
|
This document outlines security procedures and general policies for the WebTorrent
|
||||||
|
project.
|
||||||
|
|
||||||
|
* [Reporting a Bug](#reporting-a-bug)
|
||||||
|
* [Disclosure Policy](#disclosure-policy)
|
||||||
|
* [Comments on this Policy](#comments-on-this-policy)
|
||||||
|
|
||||||
|
## Reporting a Bug
|
||||||
|
|
||||||
|
The WebTorrent team and community take all security bugs in WebTorrent seriously.
|
||||||
|
Thank you for improving the security of WebTorrent. We appreciate your efforts and
|
||||||
|
responsible disclosure and will make every effort to acknowledge your
|
||||||
|
contributions.
|
||||||
|
|
||||||
|
Report security bugs by emailing the lead maintainer at feross@feross.org.
|
||||||
|
|
||||||
|
The lead maintainer will acknowledge your email within 48 hours, and will send a
|
||||||
|
more detailed response within 48 hours indicating the next steps in handling
|
||||||
|
your report. After the initial reply to your report, the security team will
|
||||||
|
endeavor to keep you informed of the progress towards a fix and full
|
||||||
|
announcement, and may ask for additional information or guidance.
|
||||||
|
|
||||||
|
Report security bugs in third-party modules to the person or team maintaining
|
||||||
|
the module. You can also report a vulnerability through the
|
||||||
|
[Node Security Project](https://nodesecurity.io/report).
|
||||||
|
|
||||||
|
## Disclosure Policy
|
||||||
|
|
||||||
|
When the security team receives a security bug report, they will assign it to a
|
||||||
|
primary handler. This person will coordinate the fix and release process,
|
||||||
|
involving the following steps:
|
||||||
|
|
||||||
|
* Confirm the problem and determine the affected versions.
|
||||||
|
* Audit code to find any potential similar problems.
|
||||||
|
* Prepare fixes for all releases still under maintenance. These fixes will be
|
||||||
|
released as fast as possible to npm.
|
||||||
|
|
||||||
|
## Comments on this Policy
|
||||||
|
|
||||||
|
If you have suggestions on how this process could be improved please submit a
|
||||||
|
pull request.
|
||||||
@@ -18,6 +18,13 @@ files.forEach(function (file) {
|
|||||||
lines.forEach(function (line, i) {
|
lines.forEach(function (line, i) {
|
||||||
let error
|
let error
|
||||||
|
|
||||||
|
// Consistent JSX tag closing
|
||||||
|
if (line.match(/' {2}\/> *$/) ||
|
||||||
|
line.match('[^ ]/> *$') ||
|
||||||
|
line.match(' > *$')) {
|
||||||
|
error = 'JSX tag spacing'
|
||||||
|
}
|
||||||
|
|
||||||
// No lines over 100 characters
|
// No lines over 100 characters
|
||||||
if (line.length > 100) {
|
if (line.length > 100) {
|
||||||
error = 'Line >100 chars'
|
error = 'Line >100 chars'
|
||||||
@@ -28,7 +35,7 @@ files.forEach(function (file) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
const name = path.basename(file)
|
let name = path.basename(file)
|
||||||
console.log('%s:%d - %s:\n%s', name, i + 1, error, line)
|
console.log('%s:%d - %s:\n%s', name, i + 1, error, line)
|
||||||
hasErrors = true
|
hasErrors = true
|
||||||
}
|
}
|
||||||
|
|||||||
143
bin/package.js
143
bin/package.js
@@ -39,7 +39,8 @@ const argv = minimist(process.argv.slice(2), {
|
|||||||
function build () {
|
function build () {
|
||||||
console.log('Reinstalling node_modules...')
|
console.log('Reinstalling node_modules...')
|
||||||
rimraf.sync(NODE_MODULES_PATH)
|
rimraf.sync(NODE_MODULES_PATH)
|
||||||
cp.execSync('npm ci', { stdio: 'inherit' })
|
cp.execSync('npm install', { stdio: 'inherit' })
|
||||||
|
cp.execSync('npm dedupe', { stdio: 'inherit' })
|
||||||
|
|
||||||
console.log('Nuking dist/ and build/...')
|
console.log('Nuking dist/ and build/...')
|
||||||
rimraf.sync(DIST_PATH)
|
rimraf.sync(DIST_PATH)
|
||||||
@@ -70,11 +71,11 @@ function build () {
|
|||||||
const all = {
|
const all = {
|
||||||
// The human-readable copyright line for the app. Maps to the `LegalCopyright` metadata
|
// The human-readable copyright line for the app. Maps to the `LegalCopyright` metadata
|
||||||
// property on Windows, and `NSHumanReadableCopyright` on Mac.
|
// property on Windows, and `NSHumanReadableCopyright` on Mac.
|
||||||
appCopyright: config.APP_COPYRIGHT,
|
'app-copyright': config.APP_COPYRIGHT,
|
||||||
|
|
||||||
// The release version of the application. Maps to the `ProductVersion` metadata
|
// The release version of the application. Maps to the `ProductVersion` metadata
|
||||||
// property on Windows, and `CFBundleShortVersionString` on Mac.
|
// property on Windows, and `CFBundleShortVersionString` on Mac.
|
||||||
appVersion: pkg.version,
|
'app-version': pkg.version,
|
||||||
|
|
||||||
// Package the application's source code into an archive, using Electron's archive
|
// Package the application's source code into an archive, using Electron's archive
|
||||||
// format. Mitigates issues around long path names on Windows and slightly speeds up
|
// format. Mitigates issues around long path names on Windows and slightly speeds up
|
||||||
@@ -88,7 +89,7 @@ const all = {
|
|||||||
// The build version of the application. Maps to the FileVersion metadata property on
|
// The build version of the application. Maps to the FileVersion metadata property on
|
||||||
// Windows, and CFBundleVersion on Mac. Note: Windows requires the build version to
|
// Windows, and CFBundleVersion on Mac. Note: Windows requires the build version to
|
||||||
// start with a number. We're using the version of the underlying WebTorrent library.
|
// start with a number. We're using the version of the underlying WebTorrent library.
|
||||||
buildVersion: require('webtorrent/package.json').version,
|
'build-version': require('webtorrent/package.json').version,
|
||||||
|
|
||||||
// The application source directory.
|
// The application source directory.
|
||||||
dir: config.ROOT_PATH,
|
dir: config.ROOT_PATH,
|
||||||
@@ -118,18 +119,18 @@ const darwin = {
|
|||||||
// Build for Mac
|
// Build for Mac
|
||||||
platform: 'darwin',
|
platform: 'darwin',
|
||||||
|
|
||||||
// Build x64 binary only.
|
// Build x64 binaries only.
|
||||||
arch: 'x64',
|
arch: 'x64',
|
||||||
|
|
||||||
// The bundle identifier to use in the application's plist (Mac only).
|
// The bundle identifier to use in the application's plist (Mac only).
|
||||||
appBundleId: 'io.webtorrent.webtorrent',
|
'app-bundle-id': 'io.webtorrent.webtorrent',
|
||||||
|
|
||||||
// The application category type, as shown in the Finder via "View" -> "Arrange by
|
// The application category type, as shown in the Finder via "View" -> "Arrange by
|
||||||
// Application Category" when viewing the Applications directory (Mac only).
|
// Application Category" when viewing the Applications directory (Mac only).
|
||||||
appCategoryType: 'public.app-category.utilities',
|
'app-category-type': 'public.app-category.utilities',
|
||||||
|
|
||||||
// The bundle identifier to use in the application helper's plist (Mac only).
|
// The bundle identifier to use in the application helper's plist (Mac only).
|
||||||
helperBundleId: 'io.webtorrent.webtorrent-helper',
|
'helper-bundle-id': 'io.webtorrent.webtorrent-helper',
|
||||||
|
|
||||||
// Application icon.
|
// Application icon.
|
||||||
icon: config.APP_ICON + '.icns'
|
icon: config.APP_ICON + '.icns'
|
||||||
@@ -139,8 +140,8 @@ const win32 = {
|
|||||||
// Build for Windows.
|
// Build for Windows.
|
||||||
platform: 'win32',
|
platform: 'win32',
|
||||||
|
|
||||||
// Build x64 binary only.
|
// Build ia32 and x64 binaries.
|
||||||
arch: 'x64',
|
arch: ['ia32', 'x64'],
|
||||||
|
|
||||||
// Object hash of application metadata to embed into the executable (Windows only)
|
// Object hash of application metadata to embed into the executable (Windows only)
|
||||||
win32metadata: {
|
win32metadata: {
|
||||||
@@ -173,8 +174,8 @@ const linux = {
|
|||||||
// Build for Linux.
|
// Build for Linux.
|
||||||
platform: 'linux',
|
platform: 'linux',
|
||||||
|
|
||||||
// Build x64 binary onle.
|
// Build ia32 and x64 binaries.
|
||||||
arch: 'x64'
|
arch: ['ia32', 'x64']
|
||||||
|
|
||||||
// Note: Application icon for Linux is specified via the BrowserWindow `icon` option.
|
// Note: Application icon for Linux is specified via the BrowserWindow `icon` option.
|
||||||
}
|
}
|
||||||
@@ -185,7 +186,8 @@ function buildDarwin (cb) {
|
|||||||
const plist = require('plist')
|
const plist = require('plist')
|
||||||
|
|
||||||
console.log('Mac: Packaging electron...')
|
console.log('Mac: Packaging electron...')
|
||||||
electronPackager(Object.assign({}, all, darwin)).then(function (buildPath) {
|
electronPackager(Object.assign({}, all, darwin), function (err, buildPath) {
|
||||||
|
if (err) return cb(err)
|
||||||
console.log('Mac: Packaged electron. ' + buildPath)
|
console.log('Mac: Packaged electron. ' + buildPath)
|
||||||
|
|
||||||
const appPath = path.join(buildPath[0], config.APP_NAME + '.app')
|
const appPath = path.join(buildPath[0], config.APP_NAME + '.app')
|
||||||
@@ -196,16 +198,16 @@ function buildDarwin (cb) {
|
|||||||
|
|
||||||
infoPlist.CFBundleDocumentTypes = [
|
infoPlist.CFBundleDocumentTypes = [
|
||||||
{
|
{
|
||||||
CFBundleTypeExtensions: ['torrent'],
|
CFBundleTypeExtensions: [ 'torrent' ],
|
||||||
CFBundleTypeIconFile: path.basename(config.APP_FILE_ICON) + '.icns',
|
CFBundleTypeIconFile: path.basename(config.APP_FILE_ICON) + '.icns',
|
||||||
CFBundleTypeName: 'BitTorrent Document',
|
CFBundleTypeName: 'BitTorrent Document',
|
||||||
CFBundleTypeRole: 'Editor',
|
CFBundleTypeRole: 'Editor',
|
||||||
LSHandlerRank: 'Owner',
|
LSHandlerRank: 'Owner',
|
||||||
LSItemContentTypes: ['org.bittorrent.torrent']
|
LSItemContentTypes: [ 'org.bittorrent.torrent' ]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
CFBundleTypeName: 'Any',
|
CFBundleTypeName: 'Any',
|
||||||
CFBundleTypeOSTypes: ['****'],
|
CFBundleTypeOSTypes: [ '****' ],
|
||||||
CFBundleTypeRole: 'Editor',
|
CFBundleTypeRole: 'Editor',
|
||||||
LSHandlerRank: 'Owner',
|
LSHandlerRank: 'Owner',
|
||||||
LSTypeIsPackage: false
|
LSTypeIsPackage: false
|
||||||
@@ -217,13 +219,13 @@ function buildDarwin (cb) {
|
|||||||
CFBundleTypeRole: 'Editor',
|
CFBundleTypeRole: 'Editor',
|
||||||
CFBundleURLIconFile: path.basename(config.APP_FILE_ICON) + '.icns',
|
CFBundleURLIconFile: path.basename(config.APP_FILE_ICON) + '.icns',
|
||||||
CFBundleURLName: 'BitTorrent Magnet URL',
|
CFBundleURLName: 'BitTorrent Magnet URL',
|
||||||
CFBundleURLSchemes: ['magnet']
|
CFBundleURLSchemes: [ 'magnet' ]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
CFBundleTypeRole: 'Editor',
|
CFBundleTypeRole: 'Editor',
|
||||||
CFBundleURLIconFile: path.basename(config.APP_FILE_ICON) + '.icns',
|
CFBundleURLIconFile: path.basename(config.APP_FILE_ICON) + '.icns',
|
||||||
CFBundleURLName: 'BitTorrent Stream-Magnet URL',
|
CFBundleURLName: 'BitTorrent Stream-Magnet URL',
|
||||||
CFBundleURLSchemes: ['stream-magnet']
|
CFBundleURLSchemes: [ 'stream-magnet' ]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -240,7 +242,7 @@ function buildDarwin (cb) {
|
|||||||
UTTypeReferenceURL: 'http://www.bittorrent.org/beps/bep_0000.html',
|
UTTypeReferenceURL: 'http://www.bittorrent.org/beps/bep_0000.html',
|
||||||
UTTypeTagSpecification: {
|
UTTypeTagSpecification: {
|
||||||
'com.apple.ostype': 'TORR',
|
'com.apple.ostype': 'TORR',
|
||||||
'public.filename-extension': ['torrent'],
|
'public.filename-extension': [ 'torrent' ],
|
||||||
'public.mime-type': 'application/x-bittorrent'
|
'public.mime-type': 'application/x-bittorrent'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -354,8 +356,6 @@ function buildDarwin (cb) {
|
|||||||
cb(null)
|
cb(null)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}).catch(function (err) {
|
|
||||||
cb(err)
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -376,7 +376,8 @@ function buildWin32 (cb) {
|
|||||||
CERT_PATH = path.join(os.homedir(), 'Desktop')
|
CERT_PATH = path.join(os.homedir(), 'Desktop')
|
||||||
}
|
}
|
||||||
|
|
||||||
electronPackager(Object.assign({}, all, win32)).then(function (buildPath) {
|
electronPackager(Object.assign({}, all, win32), function (err, buildPath) {
|
||||||
|
if (err) return cb(err)
|
||||||
console.log('Windows: Packaged electron. ' + buildPath)
|
console.log('Windows: Packaged electron. ' + buildPath)
|
||||||
|
|
||||||
let signWithParams
|
let signWithParams
|
||||||
@@ -395,17 +396,21 @@ function buildWin32 (cb) {
|
|||||||
|
|
||||||
const tasks = []
|
const tasks = []
|
||||||
buildPath.forEach(function (filesPath) {
|
buildPath.forEach(function (filesPath) {
|
||||||
|
const destArch = filesPath.split('-').pop()
|
||||||
|
|
||||||
if (argv.package === 'exe' || argv.package === 'all') {
|
if (argv.package === 'exe' || argv.package === 'all') {
|
||||||
tasks.push((cb) => packageInstaller(filesPath, cb))
|
tasks.push((cb) => packageInstaller(filesPath, destArch, cb))
|
||||||
}
|
}
|
||||||
if (argv.package === 'portable' || argv.package === 'all') {
|
if (argv.package === 'portable' || argv.package === 'all') {
|
||||||
tasks.push((cb) => packagePortable(filesPath, cb))
|
tasks.push((cb) => packagePortable(filesPath, destArch, cb))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
series(tasks, cb)
|
series(tasks, cb)
|
||||||
|
|
||||||
function packageInstaller (filesPath, cb) {
|
function packageInstaller (filesPath, destArch, cb) {
|
||||||
console.log('Windows: Creating installer...')
|
console.log(`Windows: Creating ${destArch} installer...`)
|
||||||
|
|
||||||
|
const archStr = destArch === 'ia32' ? '-ia32' : ''
|
||||||
|
|
||||||
installer.createWindowsInstaller({
|
installer.createWindowsInstaller({
|
||||||
appDirectory: filesPath,
|
appDirectory: filesPath,
|
||||||
@@ -418,17 +423,26 @@ function buildWin32 (cb) {
|
|||||||
noMsi: true,
|
noMsi: true,
|
||||||
outputDirectory: DIST_PATH,
|
outputDirectory: DIST_PATH,
|
||||||
productName: config.APP_NAME,
|
productName: config.APP_NAME,
|
||||||
|
/**
|
||||||
|
* Only create delta updates for the Windows x64 build because 90% of our
|
||||||
|
* users have Windows x64 and the delta files take a *very* long time to
|
||||||
|
* generate. Also, the ia32 files on GitHub have non-standard Squirrel
|
||||||
|
* names (i.e. RELEASES-ia32 instead of RELEASES) and so Squirrel won't
|
||||||
|
* find them unless we proxy the requests.
|
||||||
|
*/
|
||||||
// TODO: Re-enable Windows 64-bit delta updates when we confirm that they
|
// TODO: Re-enable Windows 64-bit delta updates when we confirm that they
|
||||||
// work correctly in the presence of the "ia32" .nupkg files. I
|
// work correctly in the presence of the "ia32" .nupkg files. I
|
||||||
// (feross) noticed them listed in the 64-bit RELEASES file and
|
// (feross) noticed them listed in the 64-bit RELEASES file and
|
||||||
// manually edited them out for the v0.17 release. Shipping only
|
// manually edited them out for the v0.17 release. Shipping only
|
||||||
// full updates for now will work fine, with no ill-effects.
|
// full updates for now will work fine, with no ill-effects.
|
||||||
// remoteReleases: config.GITHUB_URL,
|
// remoteReleases: destArch === 'x64'
|
||||||
|
// ? config.GITHUB_URL
|
||||||
|
// : undefined,
|
||||||
/**
|
/**
|
||||||
* If you hit a "GitHub API rate limit exceeded" error, set this token!
|
* If you hit a "GitHub API rate limit exceeded" error, set this token!
|
||||||
*/
|
*/
|
||||||
// remoteToken: process.env.WEBTORRENT_GITHUB_API_TOKEN,
|
// remoteToken: process.env.WEBTORRENT_GITHUB_API_TOKEN,
|
||||||
setupExe: config.APP_NAME + 'Setup-v' + config.APP_VERSION + '.exe',
|
setupExe: config.APP_NAME + 'Setup-v' + config.APP_VERSION + archStr + '.exe',
|
||||||
setupIcon: config.APP_ICON + '.ico',
|
setupIcon: config.APP_ICON + '.ico',
|
||||||
signWithParams: signWithParams,
|
signWithParams: signWithParams,
|
||||||
title: config.APP_NAME,
|
title: config.APP_NAME,
|
||||||
@@ -436,7 +450,7 @@ function buildWin32 (cb) {
|
|||||||
version: pkg.version
|
version: pkg.version
|
||||||
})
|
})
|
||||||
.then(function () {
|
.then(function () {
|
||||||
console.log('Windows: Created installer.')
|
console.log(`Windows: Created ${destArch} installer.`)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete extraneous Squirrel files (i.e. *.nupkg delta files for older
|
* Delete extraneous Squirrel files (i.e. *.nupkg delta files for older
|
||||||
@@ -448,13 +462,42 @@ function buildWin32 (cb) {
|
|||||||
fs.unlinkSync(path.join(DIST_PATH, filename))
|
fs.unlinkSync(path.join(DIST_PATH, filename))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if (destArch === 'ia32') {
|
||||||
|
console.log('Windows: Renaming ia32 installer files...')
|
||||||
|
|
||||||
|
// RELEASES -> RELEASES-ia32
|
||||||
|
const relPath = path.join(DIST_PATH, 'RELEASES-ia32')
|
||||||
|
fs.renameSync(
|
||||||
|
path.join(DIST_PATH, 'RELEASES'),
|
||||||
|
relPath
|
||||||
|
)
|
||||||
|
|
||||||
|
// WebTorrent-vX.X.X-full.nupkg -> WebTorrent-vX.X.X-ia32-full.nupkg
|
||||||
|
fs.renameSync(
|
||||||
|
path.join(DIST_PATH, `${config.APP_NAME}-${config.APP_VERSION}-full.nupkg`),
|
||||||
|
path.join(DIST_PATH, `${config.APP_NAME}-${config.APP_VERSION}-ia32-full.nupkg`)
|
||||||
|
)
|
||||||
|
|
||||||
|
// Change file name inside RELEASES-ia32 to match renamed file
|
||||||
|
const relContent = fs.readFileSync(relPath, 'utf8')
|
||||||
|
const relContent32 = relContent.replace('full.nupkg', 'ia32-full.nupkg')
|
||||||
|
fs.writeFileSync(relPath, relContent32)
|
||||||
|
|
||||||
|
if (relContent === relContent32) {
|
||||||
|
// Sanity check
|
||||||
|
throw new Error('Fixing RELEASES-ia32 failed. Replacement did not modify the file.')
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Windows: Renamed ia32 installer files.')
|
||||||
|
}
|
||||||
|
|
||||||
cb(null)
|
cb(null)
|
||||||
})
|
})
|
||||||
.catch(cb)
|
.catch(cb)
|
||||||
}
|
}
|
||||||
|
|
||||||
function packagePortable (filesPath, cb) {
|
function packagePortable (filesPath, destArch, cb) {
|
||||||
console.log('Windows: Creating portable app...')
|
console.log(`Windows: Creating ${destArch} portable app...`)
|
||||||
|
|
||||||
const portablePath = path.join(filesPath, 'Portable Settings')
|
const portablePath = path.join(filesPath, 'Portable Settings')
|
||||||
mkdirp.sync(portablePath)
|
mkdirp.sync(portablePath)
|
||||||
@@ -465,41 +508,41 @@ function buildWin32 (cb) {
|
|||||||
const tempPath = path.join(portablePath, 'Temp')
|
const tempPath = path.join(portablePath, 'Temp')
|
||||||
mkdirp.sync(tempPath)
|
mkdirp.sync(tempPath)
|
||||||
|
|
||||||
|
const archStr = destArch === 'ia32' ? '-ia32' : ''
|
||||||
|
|
||||||
const inPath = path.join(DIST_PATH, path.basename(filesPath))
|
const inPath = path.join(DIST_PATH, path.basename(filesPath))
|
||||||
const outPath = path.join(DIST_PATH, BUILD_NAME + '-win.zip')
|
const outPath = path.join(DIST_PATH, BUILD_NAME + '-win' + archStr + '.zip')
|
||||||
zip.zipSync(inPath, outPath)
|
zip.zipSync(inPath, outPath)
|
||||||
|
|
||||||
console.log('Windows: Created portable app.')
|
console.log(`Windows: Created ${destArch} portable app.`)
|
||||||
cb(null)
|
cb(null)
|
||||||
}
|
}
|
||||||
}).catch(function (err) {
|
|
||||||
cb(err)
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function buildLinux (cb) {
|
function buildLinux (cb) {
|
||||||
console.log('Linux: Packaging electron...')
|
console.log('Linux: Packaging electron...')
|
||||||
|
electronPackager(Object.assign({}, all, linux), function (err, buildPath) {
|
||||||
electronPackager(Object.assign({}, all, linux)).then(function (buildPath) {
|
if (err) return cb(err)
|
||||||
console.log('Linux: Packaged electron. ' + buildPath)
|
console.log('Linux: Packaged electron. ' + buildPath)
|
||||||
|
|
||||||
const tasks = []
|
const tasks = []
|
||||||
buildPath.forEach(function (filesPath) {
|
buildPath.forEach(function (filesPath) {
|
||||||
|
const destArch = filesPath.split('-').pop()
|
||||||
|
|
||||||
if (argv.package === 'deb' || argv.package === 'all') {
|
if (argv.package === 'deb' || argv.package === 'all') {
|
||||||
tasks.push((cb) => packageDeb(filesPath, cb))
|
tasks.push((cb) => packageDeb(filesPath, destArch, cb))
|
||||||
}
|
}
|
||||||
if (argv.package === 'zip' || argv.package === 'all') {
|
if (argv.package === 'zip' || argv.package === 'all') {
|
||||||
tasks.push((cb) => packageZip(filesPath, cb))
|
tasks.push((cb) => packageZip(filesPath, destArch, cb))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
series(tasks, cb)
|
series(tasks, cb)
|
||||||
}).catch(function (err) {
|
|
||||||
cb(err)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
function packageDeb (filesPath, cb) {
|
function packageDeb (filesPath, destArch, cb) {
|
||||||
// Create .deb file for Debian-based platforms
|
// Create .deb file for Debian-based platforms
|
||||||
console.log('Linux: Creating deb...')
|
console.log(`Linux: Creating ${destArch} deb...`)
|
||||||
|
|
||||||
const deb = require('nobin-debian-installer')()
|
const deb = require('nobin-debian-installer')()
|
||||||
const destPath = path.join('/opt', pkg.name)
|
const destPath = path.join('/opt', pkg.name)
|
||||||
@@ -507,7 +550,7 @@ function buildLinux (cb) {
|
|||||||
deb.pack({
|
deb.pack({
|
||||||
package: pkg,
|
package: pkg,
|
||||||
info: {
|
info: {
|
||||||
arch: 'amd64',
|
arch: destArch === 'x64' ? 'amd64' : 'i386',
|
||||||
targetDir: DIST_PATH,
|
targetDir: DIST_PATH,
|
||||||
depends: 'gconf2, libgtk2.0-0, libnss3, libxss1',
|
depends: 'gconf2, libgtk2.0-0, libnss3, libxss1',
|
||||||
scripts: {
|
scripts: {
|
||||||
@@ -527,20 +570,22 @@ function buildLinux (cb) {
|
|||||||
cwd: path.join(config.STATIC_PATH, 'linux', 'share')
|
cwd: path.join(config.STATIC_PATH, 'linux', 'share')
|
||||||
}], function (err) {
|
}], function (err) {
|
||||||
if (err) return cb(err)
|
if (err) return cb(err)
|
||||||
console.log('Linux: Created deb.')
|
console.log(`Linux: Created ${destArch} deb.`)
|
||||||
cb(null)
|
cb(null)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
function packageZip (filesPath, cb) {
|
function packageZip (filesPath, destArch, cb) {
|
||||||
// Create .zip file for Linux
|
// Create .zip file for Linux
|
||||||
console.log('Linux: Creating zip...')
|
console.log(`Linux: Creating ${destArch} zip...`)
|
||||||
|
|
||||||
|
const archStr = destArch === 'ia32' ? '-ia32' : ''
|
||||||
|
|
||||||
const inPath = path.join(DIST_PATH, path.basename(filesPath))
|
const inPath = path.join(DIST_PATH, path.basename(filesPath))
|
||||||
const outPath = path.join(DIST_PATH, BUILD_NAME + '-linux.zip')
|
const outPath = path.join(DIST_PATH, BUILD_NAME + '-linux' + archStr + '.zip')
|
||||||
zip.zipSync(inPath, outPath)
|
zip.zipSync(inPath, outPath)
|
||||||
|
|
||||||
console.log('Linux: Created zip.')
|
console.log(`Linux: Created ${destArch} zip.`)
|
||||||
cb(null)
|
cb(null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
9238
package-lock.json
generated
9238
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
97
package.json
97
package.json
@@ -12,64 +12,65 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"airplayer": "^2.0.0",
|
"airplayer": "^2.0.0",
|
||||||
"application-config": "^1.0.1",
|
"application-config": "^1.0.0",
|
||||||
"arch": "^2.1.1",
|
"arch": "^2.0.0",
|
||||||
"auto-launch": "^5.0.5",
|
"auto-launch": "^4.0.1",
|
||||||
"bitfield": "^3.0.0",
|
"bitfield": "^1.0.2",
|
||||||
"capture-frame": "^3.0.0",
|
"capture-frame": "^2.0.0",
|
||||||
"chokidar": "^3.0.2",
|
"chokidar": "^2.0.4",
|
||||||
"chromecasts": "^1.9.1",
|
"chromecasts": "^1.9.1",
|
||||||
"cp-file": "^7.0.0",
|
"cp-file": "^6.0.0",
|
||||||
"create-torrent": "^4.3.1",
|
"create-torrent": "^3.24.5",
|
||||||
"debounce": "^1.2.0",
|
"debounce": "^1.0.0",
|
||||||
"deep-equal": "^1.1.0",
|
"deep-equal": "^1.0.1",
|
||||||
"dlnacasts": "^0.1.0",
|
"dlnacasts": "^0.1.0",
|
||||||
"drag-drop": "^5.0.1",
|
"drag-drop": "^4.1.0",
|
||||||
"es6-error": "^4.1.1",
|
"es6-error": "^4.0.0",
|
||||||
"fn-getter": "^1.0.0",
|
"fn-getter": "^1.0.0",
|
||||||
"iso-639-1": "^2.1.0",
|
"iso-639-1": "^1.2.1",
|
||||||
"languagedetect": "^1.2.0",
|
"languagedetect": "^1.1.1",
|
||||||
"location-history": "^1.1.1",
|
"location-history": "^1.0.0",
|
||||||
"material-ui": "^0.20.2",
|
"material-ui": "^0.17.0",
|
||||||
"mkdirp": "^0.5.1",
|
"mkdirp": "^0.5.1",
|
||||||
"music-metadata": "^4.5.0",
|
"music-metadata": "^3.1.0",
|
||||||
"network-address": "^1.1.2",
|
"network-address": "^1.1.0",
|
||||||
"nobin-debian-installer": "github:webtorrent/nobin-debian-installer",
|
"parse-torrent": "^6.0.1",
|
||||||
"parse-torrent": "^7.0.1",
|
"prettier-bytes": "^1.0.1",
|
||||||
"prettier-bytes": "^1.0.4",
|
"react": "^15.4.2",
|
||||||
"prop-types": "^15.7.2",
|
"react-dom": "^15.4.2",
|
||||||
"react": "^16.9.0",
|
"react-tap-event-plugin": "^2.0.1",
|
||||||
"react-dom": "^16.9.0",
|
"rimraf": "^2.5.2",
|
||||||
"rimraf": "^3.0.0",
|
"run-parallel": "^1.1.6",
|
||||||
"run-parallel": "^1.1.9",
|
"semver": "^5.1.0",
|
||||||
"semver": "^6.3.0",
|
|
||||||
"simple-concat": "^1.0.0",
|
"simple-concat": "^1.0.0",
|
||||||
"simple-get": "^3.0.3",
|
"simple-get": "^3.0.3",
|
||||||
"srt-to-vtt": "^1.1.3",
|
"srt-to-vtt": "^1.1.1",
|
||||||
"vlc-command": "^1.2.0",
|
"vlc-command": "^1.0.1",
|
||||||
"webtorrent": ">=0.107.7",
|
"webtorrent": "0.x",
|
||||||
"winreg": "^1.2.4"
|
"winreg": "^1.2.0",
|
||||||
|
"zero-fill": "^2.2.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"babel-eslint": "^10.0.3",
|
"babel-eslint": "^9.0.0",
|
||||||
"buble": "^0.19.8",
|
"buble": "^0.19.3",
|
||||||
"cross-zip": "^2.1.6",
|
"cross-zip": "^2.0.1",
|
||||||
"depcheck": "^0.8.3",
|
"depcheck": "^0.6.4",
|
||||||
"electron": "~6.0.7",
|
"electron": "1.6.16",
|
||||||
"electron-osx-sign": "^0.4.12",
|
"electron-osx-sign": "^0.4.10",
|
||||||
"electron-packager": "^14.0.5",
|
"electron-packager": "~8.5.1",
|
||||||
"electron-winstaller": "^4.0.0",
|
"electron-winstaller": "^2.6.4",
|
||||||
"gh-release": "^3.5.0",
|
"gh-release": "^3.2.1",
|
||||||
"minimist": "^1.2.0",
|
"minimist": "^1.2.0",
|
||||||
"nodemon": "^1.19.2",
|
"nobin-debian-installer": "0.0.10",
|
||||||
"opn": "^6.0.0",
|
"nodemon": "^1.10.2",
|
||||||
|
"opn": "^5.4.0",
|
||||||
"plist": "^3.0.1",
|
"plist": "^3.0.1",
|
||||||
"pngjs": "^3.4.0",
|
"pngjs": "^3.0.0",
|
||||||
"run-series": "^1.1.8",
|
"run-series": "^1.1.4",
|
||||||
"spectron": "^8.0.0",
|
"spectron": "^3.3.0",
|
||||||
"standard": "*",
|
"standard": "*",
|
||||||
"tape": "^4.11.0",
|
"tape": "^4.6.0",
|
||||||
"walk-sync": "^2.0.2"
|
"walk-sync": "^0.3.1"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=4.0.0"
|
"node": ">=4.0.0"
|
||||||
@@ -88,7 +89,7 @@
|
|||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"appdmg": "^0.6.0"
|
"appdmg": "^0.4.3"
|
||||||
},
|
},
|
||||||
"private": true,
|
"private": true,
|
||||||
"productName": "WebTorrent",
|
"productName": "WebTorrent",
|
||||||
|
|||||||
@@ -23,13 +23,13 @@ module.exports = {
|
|||||||
CRASH_REPORT_URL: 'https://webtorrent.io/desktop/crash-report',
|
CRASH_REPORT_URL: 'https://webtorrent.io/desktop/crash-report',
|
||||||
TELEMETRY_URL: 'https://webtorrent.io/desktop/telemetry',
|
TELEMETRY_URL: 'https://webtorrent.io/desktop/telemetry',
|
||||||
|
|
||||||
APP_COPYRIGHT: 'Copyright © 2014-2019 ' + APP_TEAM,
|
APP_COPYRIGHT: 'Copyright © 2014-2018 ' + APP_TEAM,
|
||||||
APP_FILE_ICON: path.join(__dirname, '..', 'static', 'WebTorrentFile'),
|
APP_FILE_ICON: path.join(__dirname, '..', 'static', 'WebTorrentFile'),
|
||||||
APP_ICON: path.join(__dirname, '..', 'static', 'WebTorrent'),
|
APP_ICON: path.join(__dirname, '..', 'static', 'WebTorrent'),
|
||||||
APP_NAME: APP_NAME,
|
APP_NAME: APP_NAME,
|
||||||
APP_TEAM: APP_TEAM,
|
APP_TEAM: APP_TEAM,
|
||||||
APP_VERSION: APP_VERSION,
|
APP_VERSION: APP_VERSION,
|
||||||
APP_WINDOW_TITLE: APP_NAME,
|
APP_WINDOW_TITLE: APP_NAME + ' (BETA)',
|
||||||
|
|
||||||
CONFIG_PATH: getConfigPath(),
|
CONFIG_PATH: getConfigPath(),
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ function openSeedFile () {
|
|||||||
log('openSeedFile')
|
log('openSeedFile')
|
||||||
const opts = {
|
const opts = {
|
||||||
title: 'Select a file for the torrent.',
|
title: 'Select a file for the torrent.',
|
||||||
properties: ['openFile']
|
properties: [ 'openFile' ]
|
||||||
}
|
}
|
||||||
showOpenSeed(opts)
|
showOpenSeed(opts)
|
||||||
}
|
}
|
||||||
@@ -35,11 +35,11 @@ function openSeedDirectory () {
|
|||||||
const opts = process.platform === 'darwin'
|
const opts = process.platform === 'darwin'
|
||||||
? {
|
? {
|
||||||
title: 'Select a file or folder for the torrent.',
|
title: 'Select a file or folder for the torrent.',
|
||||||
properties: ['openFile', 'openDirectory']
|
properties: [ 'openFile', 'openDirectory' ]
|
||||||
}
|
}
|
||||||
: {
|
: {
|
||||||
title: 'Select a folder for the torrent.',
|
title: 'Select a folder for the torrent.',
|
||||||
properties: ['openDirectory']
|
properties: [ 'openDirectory' ]
|
||||||
}
|
}
|
||||||
showOpenSeed(opts)
|
showOpenSeed(opts)
|
||||||
}
|
}
|
||||||
@@ -54,11 +54,11 @@ function openFiles () {
|
|||||||
const opts = process.platform === 'darwin'
|
const opts = process.platform === 'darwin'
|
||||||
? {
|
? {
|
||||||
title: 'Select a file or folder to add.',
|
title: 'Select a file or folder to add.',
|
||||||
properties: ['openFile', 'openDirectory']
|
properties: [ 'openFile', 'openDirectory' ]
|
||||||
}
|
}
|
||||||
: {
|
: {
|
||||||
title: 'Select a file to add.',
|
title: 'Select a file to add.',
|
||||||
properties: ['openFile']
|
properties: [ 'openFile' ]
|
||||||
}
|
}
|
||||||
setTitle(opts.title)
|
setTitle(opts.title)
|
||||||
electron.dialog.showOpenDialog(windows.main.win, opts, function (selectedPaths) {
|
electron.dialog.showOpenDialog(windows.main.win, opts, function (selectedPaths) {
|
||||||
@@ -77,7 +77,7 @@ function openTorrentFile () {
|
|||||||
const opts = {
|
const opts = {
|
||||||
title: 'Select a .torrent file.',
|
title: 'Select a .torrent file.',
|
||||||
filters: [{ name: 'Torrent Files', extensions: ['torrent'] }],
|
filters: [{ name: 'Torrent Files', extensions: ['torrent'] }],
|
||||||
properties: ['openFile', 'multiSelections']
|
properties: [ 'openFile', 'multiSelections' ]
|
||||||
}
|
}
|
||||||
setTitle(opts.title)
|
setTitle(opts.title)
|
||||||
electron.dialog.showOpenDialog(windows.main.win, opts, function (selectedPaths) {
|
electron.dialog.showOpenDialog(windows.main.win, opts, function (selectedPaths) {
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ function installDarwin () {
|
|||||||
|
|
||||||
function uninstallDarwin () {}
|
function uninstallDarwin () {}
|
||||||
|
|
||||||
const EXEC_COMMAND = [process.execPath, '--']
|
const EXEC_COMMAND = [ process.execPath, '--' ]
|
||||||
|
|
||||||
if (!config.IS_PRODUCTION) {
|
if (!config.IS_PRODUCTION) {
|
||||||
EXEC_COMMAND.push(config.ROOT_PATH)
|
EXEC_COMMAND.push(config.ROOT_PATH)
|
||||||
@@ -355,7 +355,7 @@ function uninstallLinux () {
|
|||||||
'applications',
|
'applications',
|
||||||
'webtorrent-desktop.desktop'
|
'webtorrent-desktop.desktop'
|
||||||
)
|
)
|
||||||
rimraf.sync(desktopFilePath)
|
rimraf(desktopFilePath)
|
||||||
|
|
||||||
const iconFilePath = path.join(
|
const iconFilePath = path.join(
|
||||||
os.homedir(),
|
os.homedir(),
|
||||||
@@ -364,5 +364,5 @@ function uninstallLinux () {
|
|||||||
'icons',
|
'icons',
|
||||||
'webtorrent-desktop.png'
|
'webtorrent-desktop.png'
|
||||||
)
|
)
|
||||||
rimraf.sync(iconFilePath)
|
rimraf(iconFilePath)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,14 +13,9 @@ const menu = require('./menu')
|
|||||||
const State = require('../renderer/lib/state')
|
const State = require('../renderer/lib/state')
|
||||||
const windows = require('./windows')
|
const windows = require('./windows')
|
||||||
|
|
||||||
const WEBTORRENT_VERSION = require('webtorrent/package.json').version
|
|
||||||
|
|
||||||
let shouldQuit = false
|
let shouldQuit = false
|
||||||
let argv = sliceArgv(process.argv)
|
let argv = sliceArgv(process.argv)
|
||||||
|
|
||||||
// allow electron/chromium to play startup sounds (without user interaction)
|
|
||||||
app.commandLine.appendSwitch('autoplay-policy', 'no-user-gesture-required')
|
|
||||||
|
|
||||||
// Start the app without showing the main window when auto launching on login
|
// Start the app without showing the main window when auto launching on login
|
||||||
// (On Windows and Linux, we get a flag. On MacOS, we get special API.)
|
// (On Windows and Linux, we get a flag. On MacOS, we get special API.)
|
||||||
const hidden = argv.includes('--hidden') ||
|
const hidden = argv.includes('--hidden') ||
|
||||||
@@ -43,19 +38,17 @@ if (!shouldQuit && !config.IS_PORTABLE) {
|
|||||||
// signal this instance and quit. Note: This feature creates a lock file in
|
// signal this instance and quit. Note: This feature creates a lock file in
|
||||||
// %APPDATA%\Roaming\WebTorrent so we do not do it for the Portable App since
|
// %APPDATA%\Roaming\WebTorrent so we do not do it for the Portable App since
|
||||||
// we want to be "silent" as well as "portable".
|
// we want to be "silent" as well as "portable".
|
||||||
if (!app.requestSingleInstanceLock()) {
|
shouldQuit = app.makeSingleInstance(onAppOpen)
|
||||||
shouldQuit = true
|
if (shouldQuit) {
|
||||||
|
app.quit()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (shouldQuit) {
|
if (!shouldQuit) {
|
||||||
app.quit()
|
|
||||||
} else {
|
|
||||||
init()
|
init()
|
||||||
}
|
}
|
||||||
|
|
||||||
function init () {
|
function init () {
|
||||||
app.on('second-instance', (event, commandLine, workingDirectory) => onAppOpen(commandLine))
|
|
||||||
if (config.IS_PORTABLE) {
|
if (config.IS_PORTABLE) {
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
// Put all user data into the "Portable Settings" folder
|
// Put all user data into the "Portable Settings" folder
|
||||||
@@ -98,12 +91,6 @@ function init () {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// Enable app logging into default directory, i.e. /Library/Logs/WebTorrent
|
|
||||||
// on Mac, %APPDATA% on Windows, $XDG_CONFIG_HOME or ~/.config on Linux.
|
|
||||||
app.setAppLogsPath()
|
|
||||||
|
|
||||||
app.userAgentFallback = `WebTorrent/${WEBTORRENT_VERSION} (https://webtorrent.io)`
|
|
||||||
|
|
||||||
app.on('open-file', onOpen)
|
app.on('open-file', onOpen)
|
||||||
app.on('open-url', onOpen)
|
app.on('open-url', onOpen)
|
||||||
|
|
||||||
@@ -175,7 +162,7 @@ function onOpen (e, torrentId) {
|
|||||||
// Electron issue: https://github.com/atom/electron/issues/4338
|
// Electron issue: https://github.com/atom/electron/issues/4338
|
||||||
setTimeout(() => windows.main.show(), 100)
|
setTimeout(() => windows.main.show(), 100)
|
||||||
|
|
||||||
processArgv([torrentId])
|
processArgv([ torrentId ])
|
||||||
} else {
|
} else {
|
||||||
argv.push(torrentId)
|
argv.push(torrentId)
|
||||||
}
|
}
|
||||||
@@ -205,7 +192,7 @@ function sliceArgv (argv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function processArgv (argv) {
|
function processArgv (argv) {
|
||||||
const torrentIds = []
|
let torrentIds = []
|
||||||
argv.forEach(function (arg) {
|
argv.forEach(function (arg) {
|
||||||
if (arg === '-n' || arg === '-o' || arg === '-u') {
|
if (arg === '-n' || arg === '-o' || arg === '-u') {
|
||||||
// Critical path: Only load the 'dialog' package if it is needed
|
// Critical path: Only load the 'dialog' package if it is needed
|
||||||
|
|||||||
@@ -120,21 +120,21 @@ function init () {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
ipc.on('startFolderWatcher', function () {
|
ipc.on('startFolderWatcher', function () {
|
||||||
if (!modules.folderWatcher) {
|
if (!modules['folderWatcher']) {
|
||||||
log('IPC ERR: folderWatcher module is not defined.')
|
log('IPC ERR: folderWatcher module is not defined.')
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
modules.folderWatcher.start()
|
modules['folderWatcher'].start()
|
||||||
})
|
})
|
||||||
|
|
||||||
ipc.on('stopFolderWatcher', function () {
|
ipc.on('stopFolderWatcher', function () {
|
||||||
if (!modules.folderWatcher) {
|
if (!modules['folderWatcher']) {
|
||||||
log('IPC ERR: folderWatcher module is not defined.')
|
log('IPC ERR: folderWatcher module is not defined.')
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
modules.folderWatcher.stop()
|
modules['folderWatcher'].stop()
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -3,11 +3,19 @@ module.exports = {
|
|||||||
uninstall
|
uninstall
|
||||||
}
|
}
|
||||||
|
|
||||||
const { APP_NAME } = require('../config')
|
const config = require('../config')
|
||||||
const AutoLaunch = require('auto-launch')
|
const AutoLaunch = require('auto-launch')
|
||||||
|
const { app } = require('electron')
|
||||||
|
|
||||||
|
// On Mac, work around a bug in auto-launch where it opens a Terminal window
|
||||||
|
// See https://github.com/Teamwork/node-auto-launch/issues/28#issuecomment-222194437
|
||||||
|
const appPath = process.platform === 'darwin'
|
||||||
|
? app.getPath('exe').replace(/\.app\/Content.*/, '.app')
|
||||||
|
: undefined // Use the default
|
||||||
|
|
||||||
const appLauncher = new AutoLaunch({
|
const appLauncher = new AutoLaunch({
|
||||||
name: APP_NAME,
|
name: config.APP_NAME,
|
||||||
|
path: appPath,
|
||||||
isHidden: true
|
isHidden: true
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -77,8 +77,8 @@ function onPlayerPlay () {
|
|||||||
|
|
||||||
function onPlayerUpdate (state) {
|
function onPlayerUpdate (state) {
|
||||||
if (!isEnabled()) return
|
if (!isEnabled()) return
|
||||||
buttons[PREV].flags = [state.hasPrevious ? 'enabled' : 'disabled']
|
buttons[PREV].flags = [ state.hasPrevious ? 'enabled' : 'disabled' ]
|
||||||
buttons[NEXT].flags = [state.hasNext ? 'enabled' : 'disabled']
|
buttons[NEXT].flags = [ state.hasNext ? 'enabled' : 'disabled' ]
|
||||||
update()
|
update()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -24,9 +24,6 @@ function init () {
|
|||||||
skipTaskbar: true,
|
skipTaskbar: true,
|
||||||
title: 'About ' + config.APP_WINDOW_TITLE,
|
title: 'About ' + config.APP_WINDOW_TITLE,
|
||||||
useContentSize: true,
|
useContentSize: true,
|
||||||
webPreferences: {
|
|
||||||
nodeIntegration: true
|
|
||||||
},
|
|
||||||
width: 300
|
width: 300
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -40,12 +40,9 @@ function init (state, options) {
|
|||||||
minWidth: config.WINDOW_MIN_WIDTH,
|
minWidth: config.WINDOW_MIN_WIDTH,
|
||||||
show: false,
|
show: false,
|
||||||
title: config.APP_WINDOW_TITLE,
|
title: config.APP_WINDOW_TITLE,
|
||||||
titleBarStyle: 'hiddenInset', // Hide title bar (Mac)
|
titleBarStyle: 'hidden-inset', // Hide title bar (Mac)
|
||||||
useContentSize: true, // Specify web page size without OS chrome
|
useContentSize: true, // Specify web page size without OS chrome
|
||||||
width: initialBounds.width,
|
width: initialBounds.width,
|
||||||
webPreferences: {
|
|
||||||
nodeIntegration: true
|
|
||||||
},
|
|
||||||
x: initialBounds.x,
|
x: initialBounds.x,
|
||||||
y: initialBounds.y
|
y: initialBounds.y
|
||||||
})
|
})
|
||||||
@@ -141,7 +138,7 @@ function setAspectRatio (aspectRatio) {
|
|||||||
function setBounds (bounds, maximize) {
|
function setBounds (bounds, maximize) {
|
||||||
// Do nothing in fullscreen
|
// Do nothing in fullscreen
|
||||||
if (!main.win || main.win.isFullScreen()) {
|
if (!main.win || main.win.isFullScreen()) {
|
||||||
log('setBounds: not setting bounds because already in full screen mode')
|
log(`setBounds: not setting bounds because we're in full screen`)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -212,7 +209,7 @@ function toggleDevTools () {
|
|||||||
if (main.win.webContents.isDevToolsOpened()) {
|
if (main.win.webContents.isDevToolsOpened()) {
|
||||||
main.win.webContents.closeDevTools()
|
main.win.webContents.closeDevTools()
|
||||||
} else {
|
} else {
|
||||||
main.win.webContents.openDevTools({ mode: 'detach' })
|
main.win.webContents.openDevTools({ detach: true })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -25,9 +25,6 @@ function init () {
|
|||||||
skipTaskbar: true,
|
skipTaskbar: true,
|
||||||
title: 'webtorrent-hidden-window',
|
title: 'webtorrent-hidden-window',
|
||||||
useContentSize: true,
|
useContentSize: true,
|
||||||
webPreferences: {
|
|
||||||
nodeIntegration: true
|
|
||||||
},
|
|
||||||
width: 150
|
width: 150
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -59,6 +56,6 @@ function toggleDevTools () {
|
|||||||
webtorrent.win.webContents.closeDevTools()
|
webtorrent.win.webContents.closeDevTools()
|
||||||
webtorrent.win.hide()
|
webtorrent.win.hide()
|
||||||
} else {
|
} else {
|
||||||
webtorrent.win.webContents.openDevTools({ mode: 'detach' })
|
webtorrent.win.webContents.openDevTools({ detach: true })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,26 +6,22 @@ class Header extends React.Component {
|
|||||||
render () {
|
render () {
|
||||||
const loc = this.props.state.location
|
const loc = this.props.state.location
|
||||||
return (
|
return (
|
||||||
<div
|
<div className='header'
|
||||||
className='header'
|
|
||||||
onMouseMove={dispatcher('mediaMouseMoved')}
|
onMouseMove={dispatcher('mediaMouseMoved')}
|
||||||
onMouseEnter={dispatcher('mediaControlsMouseEnter')}
|
onMouseEnter={dispatcher('mediaControlsMouseEnter')}
|
||||||
onMouseLeave={dispatcher('mediaControlsMouseLeave')}
|
onMouseLeave={dispatcher('mediaControlsMouseLeave')}>
|
||||||
>
|
|
||||||
{this.getTitle()}
|
{this.getTitle()}
|
||||||
<div className='nav left float-left'>
|
<div className='nav left float-left'>
|
||||||
<i
|
<i
|
||||||
className={'icon back ' + (loc.hasBack() ? '' : 'disabled')}
|
className={'icon back ' + (loc.hasBack() ? '' : 'disabled')}
|
||||||
title='Back'
|
title='Back'
|
||||||
onClick={dispatcher('back')}
|
onClick={dispatcher('back')}>
|
||||||
>
|
|
||||||
chevron_left
|
chevron_left
|
||||||
</i>
|
</i>
|
||||||
<i
|
<i
|
||||||
className={'icon forward ' + (loc.hasForward() ? '' : 'disabled')}
|
className={'icon forward ' + (loc.hasForward() ? '' : 'disabled')}
|
||||||
title='Forward'
|
title='Forward'
|
||||||
onClick={dispatcher('forward')}
|
onClick={dispatcher('forward')}>
|
||||||
>
|
|
||||||
chevron_right
|
chevron_right
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
@@ -49,8 +45,7 @@ class Header extends React.Component {
|
|||||||
<i
|
<i
|
||||||
className='icon add'
|
className='icon add'
|
||||||
title='Add torrent'
|
title='Add torrent'
|
||||||
onClick={dispatcher('openFiles')}
|
onClick={dispatcher('openFiles')}>
|
||||||
>
|
|
||||||
add
|
add
|
||||||
</i>
|
</i>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
const React = require('react')
|
const React = require('react')
|
||||||
const PropTypes = require('prop-types')
|
|
||||||
|
|
||||||
const colors = require('material-ui/styles/colors')
|
const colors = require('material-ui/styles/colors')
|
||||||
|
|
||||||
class Heading extends React.Component {
|
class Heading extends React.Component {
|
||||||
static get propTypes () {
|
static get propTypes () {
|
||||||
return {
|
return {
|
||||||
level: PropTypes.number
|
level: React.PropTypes.number
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -12,15 +12,13 @@ module.exports = class ModalOKCancel extends React.Component {
|
|||||||
className='control cancel'
|
className='control cancel'
|
||||||
style={cancelStyle}
|
style={cancelStyle}
|
||||||
label={cancelText}
|
label={cancelText}
|
||||||
onClick={onCancel}
|
onClick={onCancel} />
|
||||||
/>
|
|
||||||
<RaisedButton
|
<RaisedButton
|
||||||
className='control ok'
|
className='control ok'
|
||||||
primary
|
primary
|
||||||
label={okText}
|
label={okText}
|
||||||
onClick={onOK}
|
onClick={onOK}
|
||||||
autoFocus
|
autoFocus />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,15 +15,13 @@ module.exports = class OpenTorrentAddressModal extends React.Component {
|
|||||||
className='control'
|
className='control'
|
||||||
ref={(c) => { this.torrentURL = c }}
|
ref={(c) => { this.torrentURL = c }}
|
||||||
fullWidth
|
fullWidth
|
||||||
onKeyDown={handleKeyDown.bind(this)}
|
onKeyDown={handleKeyDown.bind(this)} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
<ModalOKCancel
|
<ModalOKCancel
|
||||||
cancelText='CANCEL'
|
cancelText='CANCEL'
|
||||||
onCancel={dispatcher('exitModal')}
|
onCancel={dispatcher('exitModal')}
|
||||||
okText='OK'
|
okText='OK'
|
||||||
onOK={handleOK.bind(this)}
|
onOK={handleOK.bind(this)} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,6 @@
|
|||||||
const path = require('path')
|
|
||||||
|
|
||||||
const colors = require('material-ui/styles/colors')
|
const colors = require('material-ui/styles/colors')
|
||||||
const electron = require('electron')
|
const electron = require('electron')
|
||||||
const React = require('react')
|
const React = require('react')
|
||||||
const PropTypes = require('prop-types')
|
|
||||||
|
|
||||||
const remote = electron.remote
|
const remote = electron.remote
|
||||||
|
|
||||||
@@ -14,14 +11,15 @@ const TextField = require('material-ui/TextField').default
|
|||||||
// Uses the system Open File dialog.
|
// Uses the system Open File dialog.
|
||||||
// You can't edit the text field directly.
|
// You can't edit the text field directly.
|
||||||
class PathSelector extends React.Component {
|
class PathSelector extends React.Component {
|
||||||
static propTypes () {
|
static get propTypes () {
|
||||||
return {
|
return {
|
||||||
className: PropTypes.string,
|
className: React.PropTypes.string,
|
||||||
dialog: PropTypes.object,
|
dialog: React.PropTypes.object,
|
||||||
id: PropTypes.string,
|
displayValue: React.PropTypes.string,
|
||||||
onChange: PropTypes.func,
|
id: React.PropTypes.string,
|
||||||
title: PropTypes.string.isRequired,
|
onChange: React.PropTypes.func,
|
||||||
value: PropTypes.string
|
title: React.PropTypes.string.isRequired,
|
||||||
|
value: React.PropTypes.string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -32,8 +30,8 @@ class PathSelector extends React.Component {
|
|||||||
|
|
||||||
handleClick () {
|
handleClick () {
|
||||||
const opts = Object.assign({
|
const opts = Object.assign({
|
||||||
defaultPath: this.props.value && path.dirname(this.props.value),
|
defaultPath: this.props.value,
|
||||||
properties: ['openFile', 'openDirectory']
|
properties: [ 'openFile', 'openDirectory' ]
|
||||||
}, this.props.dialog)
|
}, this.props.dialog)
|
||||||
|
|
||||||
remote.dialog.showOpenDialog(
|
remote.dialog.showOpenDialog(
|
||||||
@@ -66,7 +64,7 @@ class PathSelector extends React.Component {
|
|||||||
const textFieldStyle = {
|
const textFieldStyle = {
|
||||||
flex: '1'
|
flex: '1'
|
||||||
}
|
}
|
||||||
const text = this.props.value || ''
|
const text = this.props.displayValue || this.props.value
|
||||||
const buttonStyle = {
|
const buttonStyle = {
|
||||||
marginLeft: 10
|
marginLeft: 10
|
||||||
}
|
}
|
||||||
@@ -76,14 +74,10 @@ class PathSelector extends React.Component {
|
|||||||
<div className='label' style={labelStyle}>
|
<div className='label' style={labelStyle}>
|
||||||
{this.props.title}:
|
{this.props.title}:
|
||||||
</div>
|
</div>
|
||||||
<TextField
|
<TextField className='control' disabled id={id} value={text}
|
||||||
className='control' disabled id={id} value={text}
|
inputStyle={textareaStyle} style={textFieldStyle} />
|
||||||
inputStyle={textareaStyle} style={textFieldStyle}
|
<RaisedButton className='control' label='Change' onClick={this.handleClick}
|
||||||
/>
|
style={buttonStyle} />
|
||||||
<RaisedButton
|
|
||||||
className='control' label='Change' onClick={this.handleClick}
|
|
||||||
style={buttonStyle}
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,8 +18,7 @@ module.exports = class RemoveTorrentModal extends React.Component {
|
|||||||
cancelText='CANCEL'
|
cancelText='CANCEL'
|
||||||
onCancel={dispatcher('exitModal')}
|
onCancel={dispatcher('exitModal')}
|
||||||
okText={buttonText}
|
okText={buttonText}
|
||||||
onOK={handleRemove}
|
onOK={handleRemove} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,13 @@
|
|||||||
const React = require('react')
|
const React = require('react')
|
||||||
const PropTypes = require('prop-types')
|
|
||||||
|
|
||||||
const RaisedButton = require('material-ui/RaisedButton').default
|
const RaisedButton = require('material-ui/RaisedButton').default
|
||||||
|
|
||||||
class ShowMore extends React.Component {
|
class ShowMore extends React.Component {
|
||||||
static get propTypes () {
|
static get propTypes () {
|
||||||
return {
|
return {
|
||||||
defaultExpanded: PropTypes.bool,
|
defaultExpanded: React.PropTypes.bool,
|
||||||
hideLabel: PropTypes.string,
|
hideLabel: React.PropTypes.string,
|
||||||
showLabel: PropTypes.string
|
showLabel: React.PropTypes.string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -45,8 +44,7 @@ class ShowMore extends React.Component {
|
|||||||
<RaisedButton
|
<RaisedButton
|
||||||
className='control'
|
className='control'
|
||||||
onClick={this.handleClick}
|
onClick={this.handleClick}
|
||||||
label={label}
|
label={label} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,8 +28,7 @@ module.exports = class UnsupportedMediaModal extends React.Component {
|
|||||||
cancelText='CANCEL'
|
cancelText='CANCEL'
|
||||||
onCancel={dispatcher('backToList')}
|
onCancel={dispatcher('backToList')}
|
||||||
okText={actionText}
|
okText={actionText}
|
||||||
onOK={onAction}
|
onOK={onAction} />
|
||||||
/>
|
|
||||||
<p className='error-text'>{errorMessage}</p>
|
<p className='error-text'>{errorMessage}</p>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -18,8 +18,7 @@ module.exports = class UpdateAvailableModal extends React.Component {
|
|||||||
cancelText='SKIP THIS RELEASE'
|
cancelText='SKIP THIS RELEASE'
|
||||||
onCancel={handleSkip}
|
onCancel={handleSkip}
|
||||||
okText='SHOW DOWNLOAD PAGE'
|
okText='SHOW DOWNLOAD PAGE'
|
||||||
onOK={handleShow}
|
onOK={handleShow} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -269,7 +269,6 @@ module.exports = class PlaybackController {
|
|||||||
// update state
|
// update state
|
||||||
state.playing.infoHash = infoHash
|
state.playing.infoHash = infoHash
|
||||||
state.playing.fileIndex = index
|
state.playing.fileIndex = index
|
||||||
state.playing.fileName = fileSummary.name
|
|
||||||
state.playing.type = TorrentPlayer.isVideo(fileSummary) ? 'video'
|
state.playing.type = TorrentPlayer.isVideo(fileSummary) ? 'video'
|
||||||
: TorrentPlayer.isAudio(fileSummary) ? 'audio'
|
: TorrentPlayer.isAudio(fileSummary) ? 'audio'
|
||||||
: 'other'
|
: 'other'
|
||||||
|
|||||||
@@ -15,8 +15,8 @@ module.exports = class SubtitlesController {
|
|||||||
openSubtitles () {
|
openSubtitles () {
|
||||||
remote.dialog.showOpenDialog({
|
remote.dialog.showOpenDialog({
|
||||||
title: 'Select a subtitles file.',
|
title: 'Select a subtitles file.',
|
||||||
filters: [{ name: 'Subtitles', extensions: ['vtt', 'srt'] }],
|
filters: [ { name: 'Subtitles', extensions: ['vtt', 'srt'] } ],
|
||||||
properties: ['openFile']
|
properties: [ 'openFile' ]
|
||||||
}, (filenames) => {
|
}, (filenames) => {
|
||||||
if (!Array.isArray(filenames)) return
|
if (!Array.isArray(filenames)) return
|
||||||
this.addSubtitles(filenames, true)
|
this.addSubtitles(filenames, true)
|
||||||
|
|||||||
@@ -318,7 +318,7 @@ module.exports = class TorrentListController {
|
|||||||
function findFilesRecursive (paths, cb_) {
|
function findFilesRecursive (paths, cb_) {
|
||||||
if (paths.length > 1) {
|
if (paths.length > 1) {
|
||||||
let numComplete = 0
|
let numComplete = 0
|
||||||
const ret = []
|
let ret = []
|
||||||
paths.forEach(function (path) {
|
paths.forEach(function (path) {
|
||||||
findFilesRecursive([path], function (fileObjs) {
|
findFilesRecursive([path], function (fileObjs) {
|
||||||
ret.push(...fileObjs)
|
ret.push(...fileObjs)
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
const mediaExtensions = {
|
const mediaExtensions = {
|
||||||
audio: [
|
audio: [
|
||||||
'.aac', '.aif', '.aiff', '.asf', '.dff', '.dsf', '.flac', '.m2a',
|
'.aac', '.asf', '.flac', '.m2a', '.m4a', '.m4b', '.mp2', '.mp4',
|
||||||
'.m4a', '.m4b', '.mp2', '.mp3', '.mpc', '.oga', '.ogg', '.opus',
|
'.mp3', '.oga', '.ogg', '.opus', '.wma', '.wav', '.wv', '.wvp'],
|
||||||
'.spx', '.wma', '.wav', '.wv', '.wvp'],
|
|
||||||
video: [
|
video: [
|
||||||
'.avi', '.mp4', '.m4v', '.webm', '.mov', '.mkv', '.mpg', '.mpeg',
|
'.avi', '.mp4', '.m4v', '.webm', '.mov', '.mkv', 'mpg', 'mpeg',
|
||||||
'.ogv', '.webm', '.wmv'],
|
'.ogv', '.webm', '.wmv'],
|
||||||
image: ['.gif', '.jpg', '.jpeg', '.png']
|
image: ['.gif', '.jpg', '.jpeg', '.png']
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,7 +28,6 @@ function run (state) {
|
|||||||
if (semver.lt(version, '0.14.0')) migrate_0_14_0(saved)
|
if (semver.lt(version, '0.14.0')) migrate_0_14_0(saved)
|
||||||
if (semver.lt(version, '0.17.0')) migrate_0_17_0(saved)
|
if (semver.lt(version, '0.17.0')) migrate_0_17_0(saved)
|
||||||
if (semver.lt(version, '0.17.2')) migrate_0_17_2(saved)
|
if (semver.lt(version, '0.17.2')) migrate_0_17_2(saved)
|
||||||
if (semver.lt(version, '0.21.0')) migrate_0_21_0(saved)
|
|
||||||
|
|
||||||
// Config is now on the new version
|
// Config is now on the new version
|
||||||
state.saved.version = config.APP_VERSION
|
state.saved.version = config.APP_VERSION
|
||||||
@@ -207,10 +206,3 @@ function migrate_0_17_2 (saved) {
|
|||||||
} catch (err) {}
|
} catch (err) {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function migrate_0_21_0 (saved) {
|
|
||||||
if (saved.prefs.soundNotifications == null) {
|
|
||||||
// The app used to always have sound notifications enabled
|
|
||||||
saved.prefs.soundNotifications = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -38,8 +38,7 @@ function getPreviousIndex (state) {
|
|||||||
|
|
||||||
function getCurrentLocalURL (state) {
|
function getCurrentLocalURL (state) {
|
||||||
return state.server
|
return state.server
|
||||||
? state.server.localURL + '/' + state.playing.fileIndex + '/' +
|
? state.server.localURL + '/' + state.playing.fileIndex
|
||||||
encodeURIComponent(state.playing.fileName)
|
|
||||||
: ''
|
: ''
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
init,
|
|
||||||
play
|
play
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -9,9 +8,6 @@ const path = require('path')
|
|||||||
|
|
||||||
const VOLUME = 0.25
|
const VOLUME = 0.25
|
||||||
|
|
||||||
// App state to access the soundNotifications preference
|
|
||||||
let state
|
|
||||||
|
|
||||||
/* Cache of Audio elements, for instant playback */
|
/* Cache of Audio elements, for instant playback */
|
||||||
const cache = {}
|
const cache = {}
|
||||||
|
|
||||||
@@ -50,19 +46,7 @@ const sounds = {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function init (appState) {
|
|
||||||
state = appState
|
|
||||||
}
|
|
||||||
|
|
||||||
function play (name) {
|
function play (name) {
|
||||||
if (state == null) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!state.saved.prefs.soundNotifications) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let audio = cache[name]
|
let audio = cache[name]
|
||||||
if (!audio) {
|
if (!audio) {
|
||||||
const sound = sounds[name]
|
const sound = sounds[name]
|
||||||
|
|||||||
@@ -88,7 +88,6 @@ function getDefaultPlayState () {
|
|||||||
return {
|
return {
|
||||||
infoHash: null, /* the info hash of the torrent we're playing */
|
infoHash: null, /* the info hash of the torrent we're playing */
|
||||||
fileIndex: null, /* the zero-based index within the torrent */
|
fileIndex: null, /* the zero-based index within the torrent */
|
||||||
fileName: null, /* name of the file that is playing */
|
|
||||||
location: 'local', /* 'local', 'chromecast', 'airplay' */
|
location: 'local', /* 'local', 'chromecast', 'airplay' */
|
||||||
type: null, /* 'audio' or 'video', could be 'other' if ever support eg streaming to VLC */
|
type: null, /* 'audio' or 'video', could be 'other' if ever support eg streaming to VLC */
|
||||||
currentTime: 0, /* seconds */
|
currentTime: 0, /* seconds */
|
||||||
@@ -122,7 +121,6 @@ function setupStateSaved (cb) {
|
|||||||
openExternalPlayer: false,
|
openExternalPlayer: false,
|
||||||
externalPlayerPath: null,
|
externalPlayerPath: null,
|
||||||
startup: false,
|
startup: false,
|
||||||
soundNotifications: true,
|
|
||||||
autoAddTorrents: false,
|
autoAddTorrents: false,
|
||||||
torrentsFolderPath: '',
|
torrentsFolderPath: '',
|
||||||
highestPlaybackPriority: true
|
highestPlaybackPriority: true
|
||||||
@@ -238,7 +236,7 @@ function saveImmediate (state, cb) {
|
|||||||
.filter((x) => x.infoHash)
|
.filter((x) => x.infoHash)
|
||||||
.map(function (x) {
|
.map(function (x) {
|
||||||
const torrent = {}
|
const torrent = {}
|
||||||
for (const key in x) {
|
for (let key in x) {
|
||||||
if (key === 'progress' || key === 'torrentKey') {
|
if (key === 'progress' || key === 'torrentKey') {
|
||||||
continue // Don't save progress info or key for the webtorrent process
|
continue // Don't save progress info or key for the webtorrent process
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -74,7 +74,7 @@ function reset () {
|
|||||||
|
|
||||||
// Track screen resolution
|
// Track screen resolution
|
||||||
function getScreenInfo () {
|
function getScreenInfo () {
|
||||||
return electron.remote.screen.getAllDisplays().map((screen) => ({
|
return electron.screen.getAllDisplays().map((screen) => ({
|
||||||
width: screen.size.width,
|
width: screen.size.width,
|
||||||
height: screen.size.height,
|
height: screen.size.height,
|
||||||
scaleFactor: screen.scaleFactor
|
scaleFactor: screen.scaleFactor
|
||||||
@@ -98,7 +98,7 @@ function getSystemInfo () {
|
|||||||
function getTorrentStats (state) {
|
function getTorrentStats (state) {
|
||||||
const count = state.saved.torrents.length
|
const count = state.saved.torrents.length
|
||||||
let sizeMB = 0
|
let sizeMB = 0
|
||||||
const byStatus = {
|
let byStatus = {
|
||||||
new: { count: 0, sizeMB: 0 },
|
new: { count: 0, sizeMB: 0 },
|
||||||
downloading: { count: 0, sizeMB: 0 },
|
downloading: { count: 0, sizeMB: 0 },
|
||||||
seeding: { count: 0, sizeMB: 0 },
|
seeding: { count: 0, sizeMB: 0 },
|
||||||
|
|||||||
@@ -18,8 +18,7 @@ function torrentPoster (torrent, cb) {
|
|||||||
const bestScore = ['audio', 'video', 'image'].map(mediaType => {
|
const bestScore = ['audio', 'video', 'image'].map(mediaType => {
|
||||||
return {
|
return {
|
||||||
type: mediaType,
|
type: mediaType,
|
||||||
size: calculateDataLengthByExtension(torrent, mediaExtensions[mediaType])
|
size: calculateDataLengthByExtension(torrent, mediaExtensions[mediaType]) }
|
||||||
}
|
|
||||||
}).sort((a, b) => { // sort descending on size
|
}).sort((a, b) => { // sort descending on size
|
||||||
return b.size - a.size
|
return b.size - a.size
|
||||||
})[0]
|
})[0]
|
||||||
@@ -99,7 +98,7 @@ function scoreAudioCoverFile (imgFile) {
|
|||||||
spectrogram: -80
|
spectrogram: -80
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const keyword in relevanceScore) {
|
for (let keyword in relevanceScore) {
|
||||||
if (fileName === keyword) {
|
if (fileName === keyword) {
|
||||||
return relevanceScore[keyword]
|
return relevanceScore[keyword]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
* actually used because auto-prefixing is disabled with
|
* actually used because auto-prefixing is disabled with
|
||||||
* `darkBaseTheme.userAgent = false`. Return a fake object.
|
* `darkBaseTheme.userAgent = false`. Return a fake object.
|
||||||
*/
|
*/
|
||||||
const Module = require('module')
|
let Module = require('module')
|
||||||
const _require = Module.prototype.require
|
const _require = Module.prototype.require
|
||||||
Module.prototype.require = function (id) {
|
Module.prototype.require = function (id) {
|
||||||
if (id === 'inline-style-prefixer') return {}
|
if (id === 'inline-style-prefixer') return {}
|
||||||
@@ -39,6 +39,9 @@ const TorrentPlayer = require('./lib/torrent-player')
|
|||||||
// Perf optimization: Needed immediately, so do not lazy load it below
|
// Perf optimization: Needed immediately, so do not lazy load it below
|
||||||
const TorrentListController = require('./controllers/torrent-list-controller')
|
const TorrentListController = require('./controllers/torrent-list-controller')
|
||||||
|
|
||||||
|
// Required by Material UI -- adds `onTouchTap` event
|
||||||
|
require('react-tap-event-plugin')()
|
||||||
|
|
||||||
const App = require('./pages/app')
|
const App = require('./pages/app')
|
||||||
|
|
||||||
// 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
|
||||||
@@ -74,7 +77,6 @@ function onState (err, _state) {
|
|||||||
window.dispatch = dispatch
|
window.dispatch = dispatch
|
||||||
|
|
||||||
telemetry.init(state)
|
telemetry.init(state)
|
||||||
sound.init(state)
|
|
||||||
|
|
||||||
// Log uncaught JS errors
|
// Log uncaught JS errors
|
||||||
window.addEventListener(
|
window.addEventListener(
|
||||||
@@ -148,9 +150,6 @@ function onState (err, _state) {
|
|||||||
// ...same thing if you paste a torrent
|
// ...same thing if you paste a torrent
|
||||||
document.addEventListener('paste', onPaste)
|
document.addEventListener('paste', onPaste)
|
||||||
|
|
||||||
// Add YouTube style hotkey shortcuts
|
|
||||||
window.addEventListener('keydown', onKeydown)
|
|
||||||
|
|
||||||
const debouncedFullscreenToggle = debounce(function () {
|
const debouncedFullscreenToggle = debounce(function () {
|
||||||
dispatch('toggleFullScreen')
|
dispatch('toggleFullScreen')
|
||||||
}, 1000, true)
|
}, 1000, true)
|
||||||
@@ -235,100 +234,100 @@ function updateElectron () {
|
|||||||
|
|
||||||
const dispatchHandlers = {
|
const dispatchHandlers = {
|
||||||
// Torrent list: creating, deleting, selecting torrents
|
// Torrent list: creating, deleting, selecting torrents
|
||||||
openTorrentFile: () => ipcRenderer.send('openTorrentFile'),
|
'openTorrentFile': () => ipcRenderer.send('openTorrentFile'),
|
||||||
openFiles: () => ipcRenderer.send('openFiles'), /* shows the open file dialog */
|
'openFiles': () => ipcRenderer.send('openFiles'), /* shows the open file dialog */
|
||||||
openTorrentAddress: () => { state.modal = { id: 'open-torrent-address-modal' } },
|
'openTorrentAddress': () => { state.modal = { id: 'open-torrent-address-modal' } },
|
||||||
|
|
||||||
addTorrent: (torrentId) => controllers.torrentList().addTorrent(torrentId),
|
'addTorrent': (torrentId) => controllers.torrentList().addTorrent(torrentId),
|
||||||
showCreateTorrent: (paths) => controllers.torrentList().showCreateTorrent(paths),
|
'showCreateTorrent': (paths) => controllers.torrentList().showCreateTorrent(paths),
|
||||||
createTorrent: (options) => controllers.torrentList().createTorrent(options),
|
'createTorrent': (options) => controllers.torrentList().createTorrent(options),
|
||||||
toggleTorrent: (infoHash) => controllers.torrentList().toggleTorrent(infoHash),
|
'toggleTorrent': (infoHash) => controllers.torrentList().toggleTorrent(infoHash),
|
||||||
pauseAllTorrents: () => controllers.torrentList().pauseAllTorrents(),
|
'pauseAllTorrents': () => controllers.torrentList().pauseAllTorrents(),
|
||||||
resumeAllTorrents: () => controllers.torrentList().resumeAllTorrents(),
|
'resumeAllTorrents': () => controllers.torrentList().resumeAllTorrents(),
|
||||||
toggleTorrentFile: (infoHash, index) =>
|
'toggleTorrentFile': (infoHash, index) =>
|
||||||
controllers.torrentList().toggleTorrentFile(infoHash, index),
|
controllers.torrentList().toggleTorrentFile(infoHash, index),
|
||||||
confirmDeleteTorrent: (infoHash, deleteData) =>
|
'confirmDeleteTorrent': (infoHash, deleteData) =>
|
||||||
controllers.torrentList().confirmDeleteTorrent(infoHash, deleteData),
|
controllers.torrentList().confirmDeleteTorrent(infoHash, deleteData),
|
||||||
deleteTorrent: (infoHash, deleteData) =>
|
'deleteTorrent': (infoHash, deleteData) =>
|
||||||
controllers.torrentList().deleteTorrent(infoHash, deleteData),
|
controllers.torrentList().deleteTorrent(infoHash, deleteData),
|
||||||
toggleSelectTorrent: (infoHash) =>
|
'toggleSelectTorrent': (infoHash) =>
|
||||||
controllers.torrentList().toggleSelectTorrent(infoHash),
|
controllers.torrentList().toggleSelectTorrent(infoHash),
|
||||||
openTorrentContextMenu: (infoHash) =>
|
'openTorrentContextMenu': (infoHash) =>
|
||||||
controllers.torrentList().openTorrentContextMenu(infoHash),
|
controllers.torrentList().openTorrentContextMenu(infoHash),
|
||||||
startTorrentingSummary: (torrentKey) =>
|
'startTorrentingSummary': (torrentKey) =>
|
||||||
controllers.torrentList().startTorrentingSummary(torrentKey),
|
controllers.torrentList().startTorrentingSummary(torrentKey),
|
||||||
saveTorrentFileAs: (torrentKey) =>
|
'saveTorrentFileAs': (torrentKey) =>
|
||||||
controllers.torrentList().saveTorrentFileAs(torrentKey),
|
controllers.torrentList().saveTorrentFileAs(torrentKey),
|
||||||
prioritizeTorrent: (infoHash) => controllers.torrentList().prioritizeTorrent(infoHash),
|
'prioritizeTorrent': (infoHash) => controllers.torrentList().prioritizeTorrent(infoHash),
|
||||||
resumePausedTorrents: () => controllers.torrentList().resumePausedTorrents(),
|
'resumePausedTorrents': () => controllers.torrentList().resumePausedTorrents(),
|
||||||
|
|
||||||
// Playback
|
// Playback
|
||||||
playFile: (infoHash, index) => controllers.playback().playFile(infoHash, index),
|
'playFile': (infoHash, index) => controllers.playback().playFile(infoHash, index),
|
||||||
playPause: () => controllers.playback().playPause(),
|
'playPause': () => controllers.playback().playPause(),
|
||||||
nextTrack: () => controllers.playback().nextTrack(),
|
'nextTrack': () => controllers.playback().nextTrack(),
|
||||||
previousTrack: () => controllers.playback().previousTrack(),
|
'previousTrack': () => controllers.playback().previousTrack(),
|
||||||
skip: (time) => controllers.playback().skip(time),
|
'skip': (time) => controllers.playback().skip(time),
|
||||||
skipTo: (time) => controllers.playback().skipTo(time),
|
'skipTo': (time) => controllers.playback().skipTo(time),
|
||||||
changePlaybackRate: (dir) => controllers.playback().changePlaybackRate(dir),
|
'changePlaybackRate': (dir) => controllers.playback().changePlaybackRate(dir),
|
||||||
changeVolume: (delta) => controllers.playback().changeVolume(delta),
|
'changeVolume': (delta) => controllers.playback().changeVolume(delta),
|
||||||
setVolume: (vol) => controllers.playback().setVolume(vol),
|
'setVolume': (vol) => controllers.playback().setVolume(vol),
|
||||||
openItem: (infoHash, index) => controllers.playback().openItem(infoHash, index),
|
'openItem': (infoHash, index) => controllers.playback().openItem(infoHash, index),
|
||||||
|
|
||||||
// Subtitles
|
// Subtitles
|
||||||
openSubtitles: () => controllers.subtitles().openSubtitles(),
|
'openSubtitles': () => controllers.subtitles().openSubtitles(),
|
||||||
selectSubtitle: (index) => controllers.subtitles().selectSubtitle(index),
|
'selectSubtitle': (index) => controllers.subtitles().selectSubtitle(index),
|
||||||
toggleSubtitlesMenu: () => controllers.subtitles().toggleSubtitlesMenu(),
|
'toggleSubtitlesMenu': () => controllers.subtitles().toggleSubtitlesMenu(),
|
||||||
checkForSubtitles: () => controllers.subtitles().checkForSubtitles(),
|
'checkForSubtitles': () => controllers.subtitles().checkForSubtitles(),
|
||||||
addSubtitles: (files, autoSelect) => controllers.subtitles().addSubtitles(files, autoSelect),
|
'addSubtitles': (files, autoSelect) => controllers.subtitles().addSubtitles(files, autoSelect),
|
||||||
|
|
||||||
// Local media: <video>, <audio>, external players
|
// Local media: <video>, <audio>, external players
|
||||||
mediaStalled: () => controllers.media().mediaStalled(),
|
'mediaStalled': () => controllers.media().mediaStalled(),
|
||||||
mediaError: (err) => controllers.media().mediaError(err),
|
'mediaError': (err) => controllers.media().mediaError(err),
|
||||||
mediaSuccess: () => controllers.media().mediaSuccess(),
|
'mediaSuccess': () => controllers.media().mediaSuccess(),
|
||||||
mediaTimeUpdate: () => controllers.media().mediaTimeUpdate(),
|
'mediaTimeUpdate': () => controllers.media().mediaTimeUpdate(),
|
||||||
mediaMouseMoved: () => controllers.media().mediaMouseMoved(),
|
'mediaMouseMoved': () => controllers.media().mediaMouseMoved(),
|
||||||
mediaControlsMouseEnter: () => controllers.media().controlsMouseEnter(),
|
'mediaControlsMouseEnter': () => controllers.media().controlsMouseEnter(),
|
||||||
mediaControlsMouseLeave: () => controllers.media().controlsMouseLeave(),
|
'mediaControlsMouseLeave': () => controllers.media().controlsMouseLeave(),
|
||||||
openExternalPlayer: () => controllers.media().openExternalPlayer(),
|
'openExternalPlayer': () => controllers.media().openExternalPlayer(),
|
||||||
externalPlayerNotFound: () => controllers.media().externalPlayerNotFound(),
|
'externalPlayerNotFound': () => controllers.media().externalPlayerNotFound(),
|
||||||
|
|
||||||
// Remote casting: Chromecast, Airplay, etc
|
// Remote casting: Chromecast, Airplay, etc
|
||||||
toggleCastMenu: (deviceType) => lazyLoadCast().toggleMenu(deviceType),
|
'toggleCastMenu': (deviceType) => lazyLoadCast().toggleMenu(deviceType),
|
||||||
selectCastDevice: (index) => lazyLoadCast().selectDevice(index),
|
'selectCastDevice': (index) => lazyLoadCast().selectDevice(index),
|
||||||
stopCasting: () => lazyLoadCast().stop(),
|
'stopCasting': () => lazyLoadCast().stop(),
|
||||||
|
|
||||||
// Preferences screen
|
// Preferences screen
|
||||||
preferences: () => controllers.prefs().show(),
|
'preferences': () => controllers.prefs().show(),
|
||||||
updatePreferences: (key, value) => controllers.prefs().update(key, value),
|
'updatePreferences': (key, value) => controllers.prefs().update(key, value),
|
||||||
checkDownloadPath: checkDownloadPath,
|
'checkDownloadPath': checkDownloadPath,
|
||||||
startFolderWatcher: () => controllers.folderWatcher().start(),
|
'startFolderWatcher': () => controllers.folderWatcher().start(),
|
||||||
stopFolderWatcher: () => controllers.folderWatcher().stop(),
|
'stopFolderWatcher': () => controllers.folderWatcher().stop(),
|
||||||
|
|
||||||
// Update (check for new versions on Linux, where there's no auto updater)
|
// Update (check for new versions on Linux, where there's no auto updater)
|
||||||
updateAvailable: (version) => controllers.update().updateAvailable(version),
|
'updateAvailable': (version) => controllers.update().updateAvailable(version),
|
||||||
skipVersion: (version) => controllers.update().skipVersion(version),
|
'skipVersion': (version) => controllers.update().skipVersion(version),
|
||||||
|
|
||||||
// Navigation between screens (back, forward, ESC, etc)
|
// Navigation between screens (back, forward, ESC, etc)
|
||||||
exitModal: () => { state.modal = null },
|
'exitModal': () => { state.modal = null },
|
||||||
backToList: backToList,
|
'backToList': backToList,
|
||||||
escapeBack: escapeBack,
|
'escapeBack': escapeBack,
|
||||||
back: () => state.location.back(),
|
'back': () => state.location.back(),
|
||||||
forward: () => state.location.forward(),
|
'forward': () => state.location.forward(),
|
||||||
cancel: () => state.location.cancel(),
|
'cancel': () => state.location.cancel(),
|
||||||
|
|
||||||
// Controlling the window
|
// Controlling the window
|
||||||
setDimensions: setDimensions,
|
'setDimensions': setDimensions,
|
||||||
toggleFullScreen: (setTo) => ipcRenderer.send('toggleFullScreen', setTo),
|
'toggleFullScreen': (setTo) => ipcRenderer.send('toggleFullScreen', setTo),
|
||||||
setTitle: (title) => { state.window.title = title },
|
'setTitle': (title) => { state.window.title = title },
|
||||||
resetTitle: () => { state.window.title = config.APP_WINDOW_TITLE },
|
'resetTitle': () => { state.window.title = config.APP_WINDOW_TITLE },
|
||||||
|
|
||||||
// Everything else
|
// Everything else
|
||||||
onOpen: onOpen,
|
'onOpen': onOpen,
|
||||||
error: onError,
|
'error': onError,
|
||||||
uncaughtError: (proc, err) => telemetry.logUncaughtError(proc, err),
|
'uncaughtError': (proc, err) => telemetry.logUncaughtError(proc, err),
|
||||||
stateSave: () => State.save(state),
|
'stateSave': () => State.save(state),
|
||||||
stateSaveImmediate: () => State.saveImmediate(state),
|
'stateSaveImmediate': () => State.saveImmediate(state),
|
||||||
update: () => {} // No-op, just trigger an update
|
'update': () => {} // No-op, just trigger an update
|
||||||
}
|
}
|
||||||
|
|
||||||
// Events from the UI never modify state directly. Instead they call dispatch()
|
// Events from the UI never modify state directly. Instead they call dispatch()
|
||||||
@@ -458,7 +457,7 @@ function setDimensions (dimensions) {
|
|||||||
// Called when the user adds files (.torrent, files to seed, subtitles) to the app
|
// 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)
|
// via any method (drag-drop, drag to app icon, command line)
|
||||||
function onOpen (files) {
|
function onOpen (files) {
|
||||||
if (!Array.isArray(files)) files = [files]
|
if (!Array.isArray(files)) files = [ files ]
|
||||||
|
|
||||||
// File API seems to transform "magnet:?foo" in "magnet:///?foo"
|
// File API seems to transform "magnet:?foo" in "magnet:///?foo"
|
||||||
// this is a sanitization
|
// this is a sanitization
|
||||||
@@ -511,34 +510,6 @@ function onPaste (e) {
|
|||||||
update()
|
update()
|
||||||
}
|
}
|
||||||
|
|
||||||
function onKeydown (e) {
|
|
||||||
const key = e.key
|
|
||||||
|
|
||||||
if (key === 'ArrowLeft') {
|
|
||||||
dispatch('skip', -5)
|
|
||||||
} else if (key === 'ArrowRight') {
|
|
||||||
dispatch('skip', 5)
|
|
||||||
} else if (key === 'ArrowUp') {
|
|
||||||
dispatch('changeVolume', 0.1)
|
|
||||||
} else if (key === 'ArrowDown') {
|
|
||||||
dispatch('changeVolume', -0.1)
|
|
||||||
} else if (key === 'j') {
|
|
||||||
dispatch('skip', -10)
|
|
||||||
} else if (key === 'l') {
|
|
||||||
dispatch('skip', 10)
|
|
||||||
} else if (key === 'k') {
|
|
||||||
dispatch('playPause')
|
|
||||||
} else if (key === '>') {
|
|
||||||
dispatch('changePlaybackRate', 1)
|
|
||||||
} else if (key === '<') {
|
|
||||||
dispatch('changePlaybackRate', -1)
|
|
||||||
} else if (key === 'f') {
|
|
||||||
dispatch('toggleFullScreen')
|
|
||||||
}
|
|
||||||
|
|
||||||
update()
|
|
||||||
}
|
|
||||||
|
|
||||||
function onFocus (e) {
|
function onFocus (e) {
|
||||||
state.window.isFocused = true
|
state.window.isFocused = true
|
||||||
state.dock.badge = 0
|
state.dock.badge = 0
|
||||||
|
|||||||
@@ -12,10 +12,10 @@ const Header = require('../components/header')
|
|||||||
const TorrentListPage = require('./torrent-list-page')
|
const TorrentListPage = require('./torrent-list-page')
|
||||||
|
|
||||||
const Views = {
|
const Views = {
|
||||||
home: createGetter(() => TorrentListPage),
|
'home': createGetter(() => TorrentListPage),
|
||||||
player: createGetter(() => require('./player-page')),
|
'player': createGetter(() => require('./player-page')),
|
||||||
'create-torrent': createGetter(() => require('./create-torrent-page')),
|
'create-torrent': createGetter(() => require('./create-torrent-page')),
|
||||||
preferences: createGetter(() => require('./preferences-page'))
|
'preferences': createGetter(() => require('./preferences-page'))
|
||||||
}
|
}
|
||||||
|
|
||||||
const Modals = {
|
const Modals = {
|
||||||
@@ -88,10 +88,8 @@ class App extends React.Component {
|
|||||||
return (<div key={i} className='error'>{error.message}</div>)
|
return (<div key={i} className='error'>{error.message}</div>)
|
||||||
})
|
})
|
||||||
return (
|
return (
|
||||||
<div
|
<div key='errors'
|
||||||
key='errors'
|
className={'error-popover ' + (hasErrors ? 'visible' : 'hidden')}>
|
||||||
className={'error-popover ' + (hasErrors ? 'visible' : 'hidden')}
|
|
||||||
>
|
|
||||||
<div key='title' className='title'>Error</div>
|
<div key='title' className='title'>Error</div>
|
||||||
{errorElems}
|
{errorElems}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -65,9 +65,9 @@ class CreateTorrentPage extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Create React event handlers only once
|
// Create React event handlers only once
|
||||||
this.handleSetIsPrivate = (_, isPrivate) => this.setState({ isPrivate })
|
this.setIsPrivate = (_, isPrivate) => this.setState({ isPrivate })
|
||||||
this.handleSetComment = (_, comment) => this.setState({ comment })
|
this.setComment = (_, comment) => this.setState({ comment })
|
||||||
this.handleSetTrackers = (_, trackers) => this.setState({ trackers })
|
this.setTrackers = (_, trackers) => this.setState({ trackers })
|
||||||
this.handleSubmit = handleSubmit.bind(this)
|
this.handleSubmit = handleSubmit.bind(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -94,8 +94,7 @@ class CreateTorrentPage extends React.Component {
|
|||||||
marginBottom: 10
|
marginBottom: 10
|
||||||
}}
|
}}
|
||||||
hideLabel='Hide advanced settings...'
|
hideLabel='Hide advanced settings...'
|
||||||
showLabel='Show advanced settings...'
|
showLabel='Show advanced settings...'>
|
||||||
>
|
|
||||||
{this.renderAdvanced()}
|
{this.renderAdvanced()}
|
||||||
</ShowMore>
|
</ShowMore>
|
||||||
<div className='float-right'>
|
<div className='float-right'>
|
||||||
@@ -105,14 +104,12 @@ class CreateTorrentPage extends React.Component {
|
|||||||
style={{
|
style={{
|
||||||
marginRight: 10
|
marginRight: 10
|
||||||
}}
|
}}
|
||||||
onClick={dispatcher('cancel')}
|
onClick={dispatcher('cancel')} />
|
||||||
/>
|
|
||||||
<RaisedButton
|
<RaisedButton
|
||||||
className='control create-torrent-button'
|
className='control create-torrent-button'
|
||||||
label='Create Torrent'
|
label='Create Torrent'
|
||||||
primary
|
primary
|
||||||
onClick={this.handleSubmit}
|
onClick={this.handleSubmit} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
@@ -146,8 +143,7 @@ class CreateTorrentPage extends React.Component {
|
|||||||
className='torrent-is-private control'
|
className='torrent-is-private control'
|
||||||
style={{ display: '' }}
|
style={{ display: '' }}
|
||||||
checked={this.state.isPrivate}
|
checked={this.state.isPrivate}
|
||||||
onCheck={this.handleSetIsPrivate}
|
onCheck={this.setIsPrivate} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
<div key='trackers' className='torrent-attribute'>
|
<div key='trackers' className='torrent-attribute'>
|
||||||
<label>Trackers:</label>
|
<label>Trackers:</label>
|
||||||
@@ -159,8 +155,7 @@ class CreateTorrentPage extends React.Component {
|
|||||||
rows={2}
|
rows={2}
|
||||||
rowsMax={10}
|
rowsMax={10}
|
||||||
value={this.state.trackers}
|
value={this.state.trackers}
|
||||||
onChange={this.handleSetTrackers}
|
onChange={this.setTrackers} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
<div key='comment' className='torrent-attribute'>
|
<div key='comment' className='torrent-attribute'>
|
||||||
<label>Comment:</label>
|
<label>Comment:</label>
|
||||||
@@ -173,8 +168,7 @@ class CreateTorrentPage extends React.Component {
|
|||||||
rows={2}
|
rows={2}
|
||||||
rowsMax={10}
|
rowsMax={10}
|
||||||
value={this.state.comment}
|
value={this.state.comment}
|
||||||
onChange={this.handleSetComment}
|
onChange={this.setComment} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
<div key='files' className='torrent-attribute'>
|
<div key='files' className='torrent-attribute'>
|
||||||
<label>Files:</label>
|
<label>Files:</label>
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
/* global HTMLMediaElement */
|
|
||||||
|
|
||||||
const React = require('react')
|
const React = require('react')
|
||||||
const Bitfield = require('bitfield')
|
const Bitfield = require('bitfield')
|
||||||
const prettyBytes = require('prettier-bytes')
|
const prettyBytes = require('prettier-bytes')
|
||||||
|
const zeroFill = require('zero-fill')
|
||||||
|
|
||||||
const TorrentSummary = require('../lib/torrent-summary')
|
const TorrentSummary = require('../lib/torrent-summary')
|
||||||
const Playlist = require('../lib/playlist')
|
const Playlist = require('../lib/playlist')
|
||||||
@@ -21,8 +20,7 @@ module.exports = class Player extends React.Component {
|
|||||||
<div
|
<div
|
||||||
className='player'
|
className='player'
|
||||||
onWheel={handleVolumeWheel}
|
onWheel={handleVolumeWheel}
|
||||||
onMouseMove={dispatcher('mediaMouseMoved')}
|
onMouseMove={dispatcher('mediaMouseMoved')}>
|
||||||
>
|
|
||||||
{showVideo ? renderMedia(state) : renderCastScreen(state)}
|
{showVideo ? renderMedia(state) : renderCastScreen(state)}
|
||||||
{showControls ? renderPlayerControls(state) : null}
|
{showControls ? renderPlayerControls(state) : null}
|
||||||
</div>
|
</div>
|
||||||
@@ -112,8 +110,7 @@ function renderMedia (state) {
|
|||||||
default={isSelected ? 'default' : ''}
|
default={isSelected ? 'default' : ''}
|
||||||
label={track.label}
|
label={track.label}
|
||||||
type='subtitles'
|
type='subtitles'
|
||||||
src={track.buffer}
|
src={track.buffer} />
|
||||||
/>
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -130,8 +127,7 @@ function renderMedia (state) {
|
|||||||
onError={dispatcher('mediaError')}
|
onError={dispatcher('mediaError')}
|
||||||
onTimeUpdate={dispatcher('mediaTimeUpdate')}
|
onTimeUpdate={dispatcher('mediaTimeUpdate')}
|
||||||
onEncrypted={dispatcher('mediaEncrypted')}
|
onEncrypted={dispatcher('mediaEncrypted')}
|
||||||
onCanPlay={onCanPlay}
|
onCanPlay={onCanPlay}>
|
||||||
>
|
|
||||||
{trackTags}
|
{trackTags}
|
||||||
</MediaTagName>
|
</MediaTagName>
|
||||||
)
|
)
|
||||||
@@ -141,8 +137,7 @@ function renderMedia (state) {
|
|||||||
<div
|
<div
|
||||||
key='letterbox'
|
key='letterbox'
|
||||||
className='letterbox'
|
className='letterbox'
|
||||||
onMouseMove={dispatcher('mediaMouseMoved')}
|
onMouseMove={dispatcher('mediaMouseMoved')}>
|
||||||
>
|
|
||||||
{mediaTag}
|
{mediaTag}
|
||||||
{renderOverlay(state)}
|
{renderOverlay(state)}
|
||||||
</div>
|
</div>
|
||||||
@@ -171,7 +166,6 @@ function renderMedia (state) {
|
|||||||
|
|
||||||
function onCanPlay (e) {
|
function onCanPlay (e) {
|
||||||
const elem = e.target
|
const elem = e.target
|
||||||
if (elem.readyState < HTMLMediaElement.HAVE_FUTURE_DATA) return
|
|
||||||
if (state.playing.type === 'video' &&
|
if (state.playing.type === 'video' &&
|
||||||
elem.webkitVideoDecodedByteCount === 0) {
|
elem.webkitVideoDecodedByteCount === 0) {
|
||||||
dispatch('mediaError', 'Video codec unsupported')
|
dispatch('mediaError', 'Video codec unsupported')
|
||||||
@@ -299,7 +293,7 @@ function renderAudioMetadata (state) {
|
|||||||
}
|
}
|
||||||
elems.push((
|
elems.push((
|
||||||
<div key='release' className='audio-release'>
|
<div key='release' className='audio-release'>
|
||||||
<label>Release</label>{releaseInfo.join(', ')}
|
<label>Release</label>{ releaseInfo.join(', ') }
|
||||||
</div>
|
</div>
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@@ -307,18 +301,14 @@ function renderAudioMetadata (state) {
|
|||||||
// Audio metadata: format
|
// Audio metadata: format
|
||||||
const format = []
|
const format = []
|
||||||
fileSummary.audioInfo.format = fileSummary.audioInfo.format || ''
|
fileSummary.audioInfo.format = fileSummary.audioInfo.format || ''
|
||||||
if (fileSummary.audioInfo.format.container) {
|
if (fileSummary.audioInfo.format.dataformat) {
|
||||||
format.push(fileSummary.audioInfo.format.container)
|
format.push(fileSummary.audioInfo.format.dataformat)
|
||||||
}
|
|
||||||
if (fileSummary.audioInfo.format.codec &&
|
|
||||||
fileSummary.audioInfo.format.container !== fileSummary.audioInfo.format.codec) {
|
|
||||||
format.push(fileSummary.audioInfo.format.codec)
|
|
||||||
}
|
}
|
||||||
if (fileSummary.audioInfo.format.bitrate) {
|
if (fileSummary.audioInfo.format.bitrate) {
|
||||||
format.push(Math.round(fileSummary.audioInfo.format.bitrate / 1000) + ' kbps') // 128 kbps
|
format.push(Math.round(fileSummary.audioInfo.format.bitrate / 1000) + ' kbps') // 128 kbps
|
||||||
}
|
}
|
||||||
if (fileSummary.audioInfo.format.sampleRate) {
|
if (fileSummary.audioInfo.format.sampleRate) {
|
||||||
format.push(Math.round(fileSummary.audioInfo.format.sampleRate / 100) / 10 + ' kHz')
|
format.push(Math.round(fileSummary.audioInfo.format.sampleRate / 100) / 10 + ' kHz') // 44.1 kHz
|
||||||
}
|
}
|
||||||
if (fileSummary.audioInfo.format.bitsPerSample) {
|
if (fileSummary.audioInfo.format.bitsPerSample) {
|
||||||
format.push(fileSummary.audioInfo.format.bitsPerSample + ' bit')
|
format.push(fileSummary.audioInfo.format.bitsPerSample + ' bit')
|
||||||
@@ -326,7 +316,7 @@ function renderAudioMetadata (state) {
|
|||||||
if (format.length > 0) {
|
if (format.length > 0) {
|
||||||
elems.push((
|
elems.push((
|
||||||
<div key='format' className='audio-format'>
|
<div key='format' className='audio-format'>
|
||||||
<label>Format</label>{format.join(', ')}
|
<label>Format</label>{ format.join(', ') }
|
||||||
</div>
|
</div>
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@@ -368,7 +358,7 @@ function renderLoadingSpinner (state) {
|
|||||||
<div key='loading' className='media-stalled'>
|
<div key='loading' className='media-stalled'>
|
||||||
<div key='loading-spinner' className='loading-spinner' />
|
<div key='loading-spinner' className='loading-spinner' />
|
||||||
<div key='loading-progress' className='loading-status ellipsis'>
|
<div key='loading-progress' className='loading-status ellipsis'>
|
||||||
<span><span className='progress'>{fileProgress}%</span> downloaded</span>
|
<span className='progress'>{fileProgress}%</span> downloaded
|
||||||
<span> ↓ {prettyBytes(prog.downloadSpeed || 0)}/s</span>
|
<span> ↓ {prettyBytes(prog.downloadSpeed || 0)}/s</span>
|
||||||
<span> ↑ {prettyBytes(prog.uploadSpeed || 0)}/s</span>
|
<span> ↑ {prettyBytes(prog.uploadSpeed || 0)}/s</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -435,7 +425,6 @@ function renderCastOptions (state) {
|
|||||||
return (
|
return (
|
||||||
<li key={ix} onClick={dispatcher('selectCastDevice', ix)}>
|
<li key={ix} onClick={dispatcher('selectCastDevice', ix)}>
|
||||||
<i className='icon'>{isSelected ? 'radio_button_checked' : 'radio_button_unchecked'}</i>
|
<i className='icon'>{isSelected ? 'radio_button_checked' : 'radio_button_unchecked'}</i>
|
||||||
{' '}
|
|
||||||
{name}
|
{name}
|
||||||
</li>
|
</li>
|
||||||
)
|
)
|
||||||
@@ -492,47 +481,41 @@ function renderPlayerControls (state) {
|
|||||||
<div
|
<div
|
||||||
key='cursor'
|
key='cursor'
|
||||||
className='playback-cursor'
|
className='playback-cursor'
|
||||||
style={playbackCursorStyle}
|
style={playbackCursorStyle} />
|
||||||
/>
|
|
||||||
<div
|
<div
|
||||||
key='scrub-bar'
|
key='scrub-bar'
|
||||||
className='scrub-bar'
|
className='scrub-bar'
|
||||||
draggable='true'
|
draggable='true'
|
||||||
onDragStart={handleDragStart}
|
onDragStart={handleDragStart}
|
||||||
onClick={handleScrub}
|
onClick={handleScrub}
|
||||||
onDrag={handleScrub}
|
onDrag={handleScrub} />
|
||||||
/>
|
|
||||||
</div>,
|
</div>,
|
||||||
|
|
||||||
<i
|
<i
|
||||||
key='skip-previous'
|
key='skip-previous'
|
||||||
className={'icon skip-previous float-left ' + prevClass}
|
className={'icon skip-previous float-left ' + prevClass}
|
||||||
onClick={dispatcher('previousTrack')}
|
onClick={dispatcher('previousTrack')}>
|
||||||
>
|
|
||||||
skip_previous
|
skip_previous
|
||||||
</i>,
|
</i>,
|
||||||
|
|
||||||
<i
|
<i
|
||||||
key='play'
|
key='play'
|
||||||
className='icon play-pause float-left'
|
className='icon play-pause float-left'
|
||||||
onClick={dispatcher('playPause')}
|
onClick={dispatcher('playPause')}>
|
||||||
>
|
|
||||||
{state.playing.isPaused ? 'play_arrow' : 'pause'}
|
{state.playing.isPaused ? 'play_arrow' : 'pause'}
|
||||||
</i>,
|
</i>,
|
||||||
|
|
||||||
<i
|
<i
|
||||||
key='skip-next'
|
key='skip-next'
|
||||||
className={'icon skip-next float-left ' + nextClass}
|
className={'icon skip-next float-left ' + nextClass}
|
||||||
onClick={dispatcher('nextTrack')}
|
onClick={dispatcher('nextTrack')}>
|
||||||
>
|
|
||||||
skip_next
|
skip_next
|
||||||
</i>,
|
</i>,
|
||||||
|
|
||||||
<i
|
<i
|
||||||
key='fullscreen'
|
key='fullscreen'
|
||||||
className='icon fullscreen float-right'
|
className='icon fullscreen float-right'
|
||||||
onClick={dispatcher('toggleFullScreen')}
|
onClick={dispatcher('toggleFullScreen')}>
|
||||||
>
|
|
||||||
{state.window.isFullScreen ? 'fullscreen_exit' : 'fullscreen'}
|
{state.window.isFullScreen ? 'fullscreen_exit' : 'fullscreen'}
|
||||||
</i>
|
</i>
|
||||||
]
|
]
|
||||||
@@ -543,8 +526,7 @@ function renderPlayerControls (state) {
|
|||||||
<i
|
<i
|
||||||
key='subtitles'
|
key='subtitles'
|
||||||
className={'icon closed-caption float-right ' + captionsClass}
|
className={'icon closed-caption float-right ' + captionsClass}
|
||||||
onClick={handleSubtitles}
|
onClick={handleSubtitles}>
|
||||||
>
|
|
||||||
closed_caption
|
closed_caption
|
||||||
</i>
|
</i>
|
||||||
))
|
))
|
||||||
@@ -557,9 +539,9 @@ function renderPlayerControls (state) {
|
|||||||
|
|
||||||
// Add the cast buttons. Icons for each cast type, connected/disconnected:
|
// Add the cast buttons. Icons for each cast type, connected/disconnected:
|
||||||
const buttonIcons = {
|
const buttonIcons = {
|
||||||
chromecast: { true: 'cast_connected', false: 'cast' },
|
'chromecast': { true: 'cast_connected', false: 'cast' },
|
||||||
airplay: { true: 'airplay', false: 'airplay' },
|
'airplay': { true: 'airplay', false: 'airplay' },
|
||||||
dlna: { true: 'tv', false: 'tv' }
|
'dlna': { true: 'tv', false: 'tv' }
|
||||||
}
|
}
|
||||||
castTypes.forEach(function (castType) {
|
castTypes.forEach(function (castType) {
|
||||||
// Do we show this button (eg. the Chromecast button) at all?
|
// Do we show this button (eg. the Chromecast button) at all?
|
||||||
@@ -588,8 +570,7 @@ function renderPlayerControls (state) {
|
|||||||
<i
|
<i
|
||||||
key={castType}
|
key={castType}
|
||||||
className={'icon device float-right ' + buttonClass}
|
className={'icon device float-right ' + buttonClass}
|
||||||
onClick={buttonHandler}
|
onClick={buttonHandler}>
|
||||||
>
|
|
||||||
{buttonIcon}
|
{buttonIcon}
|
||||||
</i>
|
</i>
|
||||||
))
|
))
|
||||||
@@ -612,8 +593,7 @@ function renderPlayerControls (state) {
|
|||||||
<div key='volume' className='volume float-left'>
|
<div key='volume' className='volume float-left'>
|
||||||
<i
|
<i
|
||||||
className='icon volume-icon float-left'
|
className='icon volume-icon float-left'
|
||||||
onMouseDown={handleVolumeMute}
|
onMouseDown={handleVolumeMute}>
|
||||||
>
|
|
||||||
{volumeIcon}
|
{volumeIcon}
|
||||||
</i>
|
</i>
|
||||||
<input
|
<input
|
||||||
@@ -621,8 +601,7 @@ function renderPlayerControls (state) {
|
|||||||
type='range' min='0' max='1' step='0.05'
|
type='range' min='0' max='1' step='0.05'
|
||||||
value={volume}
|
value={volume}
|
||||||
onChange={handleVolumeScrub}
|
onChange={handleVolumeScrub}
|
||||||
style={volumeStyle}
|
style={volumeStyle} />
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
))
|
))
|
||||||
|
|
||||||
@@ -645,11 +624,9 @@ function renderPlayerControls (state) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div key='controls' className='controls'
|
||||||
key='controls' className='controls'
|
|
||||||
onMouseEnter={dispatcher('mediaControlsMouseEnter')}
|
onMouseEnter={dispatcher('mediaControlsMouseEnter')}
|
||||||
onMouseLeave={dispatcher('mediaControlsMouseLeave')}
|
onMouseLeave={dispatcher('mediaControlsMouseLeave')}>
|
||||||
>
|
|
||||||
{elements}
|
{elements}
|
||||||
{renderCastOptions(state)}
|
{renderCastOptions(state)}
|
||||||
{renderSubtitleOptions(state)}
|
{renderSubtitleOptions(state)}
|
||||||
@@ -757,14 +734,14 @@ function formatTime (time, total) {
|
|||||||
return '0:00'
|
return '0:00'
|
||||||
}
|
}
|
||||||
|
|
||||||
const totalHours = Math.floor(total / 3600)
|
let totalHours = Math.floor(total / 3600)
|
||||||
const totalMinutes = Math.floor(total / 60)
|
let totalMinutes = Math.floor(total % 3600 / 60)
|
||||||
const hours = Math.floor(time / 3600)
|
let hours = Math.floor(time / 3600)
|
||||||
let minutes = Math.floor(time % 3600 / 60)
|
let minutes = Math.floor(time % 3600 / 60)
|
||||||
if (totalMinutes > 9 && minutes < 10) {
|
if (totalMinutes > 9) {
|
||||||
minutes = '0' + minutes
|
minutes = zeroFill(2, minutes)
|
||||||
}
|
}
|
||||||
const seconds = `0${Math.floor(time % 60)}`.slice(-2)
|
let seconds = zeroFill(2, Math.floor(time % 60))
|
||||||
|
|
||||||
return (totalHours > 0 ? hours + ':' : '') + minutes + ':' + seconds
|
return (totalHours > 0 ? hours + ':' : '') + minutes + ':' + seconds
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
|
const path = require('path')
|
||||||
const React = require('react')
|
const React = require('react')
|
||||||
const PropTypes = require('prop-types')
|
|
||||||
|
|
||||||
const colors = require('material-ui/styles/colors')
|
const colors = require('material-ui/styles/colors')
|
||||||
const Checkbox = require('material-ui/Checkbox').default
|
const Checkbox = require('material-ui/Checkbox').default
|
||||||
@@ -25,9 +25,6 @@ class PreferencesPage extends React.Component {
|
|||||||
|
|
||||||
this.handleStartupChange =
|
this.handleStartupChange =
|
||||||
this.handleStartupChange.bind(this)
|
this.handleStartupChange.bind(this)
|
||||||
|
|
||||||
this.handleSoundNotificationsChange =
|
|
||||||
this.handleSoundNotificationsChange.bind(this)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
downloadPathSelector () {
|
downloadPathSelector () {
|
||||||
@@ -36,12 +33,11 @@ class PreferencesPage extends React.Component {
|
|||||||
<PathSelector
|
<PathSelector
|
||||||
dialog={{
|
dialog={{
|
||||||
title: 'Select download directory',
|
title: 'Select download directory',
|
||||||
properties: ['openDirectory']
|
properties: [ 'openDirectory' ]
|
||||||
}}
|
}}
|
||||||
onChange={this.handleDownloadPathChange}
|
onChange={this.handleDownloadPathChange}
|
||||||
title='Download location'
|
title='Download location'
|
||||||
value={this.props.state.saved.prefs.downloadPath}
|
value={this.props.state.saved.prefs.downloadPath} />
|
||||||
/>
|
|
||||||
</Preference>
|
</Preference>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -56,9 +52,8 @@ class PreferencesPage extends React.Component {
|
|||||||
<Checkbox
|
<Checkbox
|
||||||
className='control'
|
className='control'
|
||||||
checked={!this.props.state.saved.prefs.openExternalPlayer}
|
checked={!this.props.state.saved.prefs.openExternalPlayer}
|
||||||
label='Play torrent media files using WebTorrent'
|
label={'Play torrent media files using WebTorrent'}
|
||||||
onCheck={this.handleOpenExternalPlayerChange}
|
onCheck={this.handleOpenExternalPlayerChange} />
|
||||||
/>
|
|
||||||
</Preference>
|
</Preference>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -73,7 +68,7 @@ class PreferencesPage extends React.Component {
|
|||||||
<Checkbox
|
<Checkbox
|
||||||
className='control'
|
className='control'
|
||||||
checked={this.props.state.saved.prefs.highestPlaybackPriority}
|
checked={this.props.state.saved.prefs.highestPlaybackPriority}
|
||||||
label='Highest Playback Priority'
|
label={'Highest Playback Priority'}
|
||||||
onCheck={this.handleHighestPlaybackPriorityChange}
|
onCheck={this.handleHighestPlaybackPriorityChange}
|
||||||
/>
|
/>
|
||||||
<p>Pauses all active torrents to allow playback to use all of the available bandwidth.</p>
|
<p>Pauses all active torrents to allow playback to use all of the available bandwidth.</p>
|
||||||
@@ -99,12 +94,12 @@ class PreferencesPage extends React.Component {
|
|||||||
<PathSelector
|
<PathSelector
|
||||||
dialog={{
|
dialog={{
|
||||||
title: 'Select media player app',
|
title: 'Select media player app',
|
||||||
properties: ['openFile']
|
properties: [ 'openFile' ]
|
||||||
}}
|
}}
|
||||||
|
displayValue={playerName}
|
||||||
onChange={this.handleExternalPlayerPathChange}
|
onChange={this.handleExternalPlayerPathChange}
|
||||||
title='External player'
|
title='External player'
|
||||||
value={playerPath}
|
value={playerPath ? path.dirname(playerPath) : null} />
|
||||||
/>
|
|
||||||
</Preference>
|
</Preference>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -119,7 +114,7 @@ class PreferencesPage extends React.Component {
|
|||||||
<Checkbox
|
<Checkbox
|
||||||
className='control'
|
className='control'
|
||||||
checked={this.props.state.saved.prefs.autoAddTorrents}
|
checked={this.props.state.saved.prefs.autoAddTorrents}
|
||||||
label='Watch for new .torrent files and add them immediately'
|
label={'Watch for new .torrent files and add them immediately'}
|
||||||
onCheck={(e, value) => { this.handleAutoAddTorrentsChange(e, value) }}
|
onCheck={(e, value) => { this.handleAutoAddTorrentsChange(e, value) }}
|
||||||
/>
|
/>
|
||||||
</Preference>
|
</Preference>
|
||||||
@@ -137,11 +132,11 @@ class PreferencesPage extends React.Component {
|
|||||||
dispatch('updatePreferences', 'autoAddTorrents', isChecked)
|
dispatch('updatePreferences', 'autoAddTorrents', isChecked)
|
||||||
|
|
||||||
if (isChecked) {
|
if (isChecked) {
|
||||||
dispatch('startFolderWatcher')
|
dispatch('startFolderWatcher', null)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
dispatch('stopFolderWatcher')
|
dispatch('stopFolderWatcher', null)
|
||||||
}
|
}
|
||||||
|
|
||||||
torrentsFolderPathSelector () {
|
torrentsFolderPathSelector () {
|
||||||
@@ -152,17 +147,17 @@ class PreferencesPage extends React.Component {
|
|||||||
<PathSelector
|
<PathSelector
|
||||||
dialog={{
|
dialog={{
|
||||||
title: 'Select folder to watch for new torrents',
|
title: 'Select folder to watch for new torrents',
|
||||||
properties: ['openDirectory']
|
properties: [ 'openDirectory' ]
|
||||||
}}
|
}}
|
||||||
onChange={this.handleTorrentsFolderPathChange}
|
displayValue={torrentsFolderPath || ''}
|
||||||
|
onChange={this.handletorrentsFolderPathChange}
|
||||||
title='Folder to watch'
|
title='Folder to watch'
|
||||||
value={torrentsFolderPath}
|
value={torrentsFolderPath ? path.dirname(torrentsFolderPath) : null} />
|
||||||
/>
|
|
||||||
</Preference>
|
</Preference>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
handleTorrentsFolderPathChange (filePath) {
|
handletorrentsFolderPathChange (filePath) {
|
||||||
dispatch('updatePreferences', 'torrentsFolderPath', filePath)
|
dispatch('updatePreferences', 'torrentsFolderPath', filePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -181,8 +176,7 @@ class PreferencesPage extends React.Component {
|
|||||||
<RaisedButton
|
<RaisedButton
|
||||||
className='control'
|
className='control'
|
||||||
onClick={this.handleSetDefaultApp}
|
onClick={this.handleSetDefaultApp}
|
||||||
label='Make WebTorrent the default'
|
label='Make WebTorrent the default' />
|
||||||
/>
|
|
||||||
</Preference>
|
</Preference>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -191,40 +185,25 @@ class PreferencesPage extends React.Component {
|
|||||||
dispatch('updatePreferences', 'startup', isChecked)
|
dispatch('updatePreferences', 'startup', isChecked)
|
||||||
}
|
}
|
||||||
|
|
||||||
setStartupCheckbox () {
|
setStartupSection () {
|
||||||
if (config.IS_PORTABLE) {
|
if (config.IS_PORTABLE) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Preference>
|
<PreferencesSection title='Startup'>
|
||||||
<Checkbox
|
<Preference>
|
||||||
className='control'
|
<Checkbox
|
||||||
checked={this.props.state.saved.prefs.startup}
|
className='control'
|
||||||
label='Open WebTorrent on startup'
|
checked={this.props.state.saved.prefs.startup}
|
||||||
onCheck={this.handleStartupChange}
|
label={'Open WebTorrent on startup.'}
|
||||||
/>
|
onCheck={this.handleStartupChange}
|
||||||
</Preference>
|
/>
|
||||||
|
</Preference>
|
||||||
|
</PreferencesSection>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
soundNotificationsCheckbox () {
|
|
||||||
return (
|
|
||||||
<Preference>
|
|
||||||
<Checkbox
|
|
||||||
className='control'
|
|
||||||
checked={this.props.state.saved.prefs.soundNotifications}
|
|
||||||
label='Enable sounds'
|
|
||||||
onCheck={this.handleSoundNotificationsChange}
|
|
||||||
/>
|
|
||||||
</Preference>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
handleSoundNotificationsChange (e, isChecked) {
|
|
||||||
dispatch('updatePreferences', 'soundNotifications', isChecked)
|
|
||||||
}
|
|
||||||
|
|
||||||
handleSetDefaultApp () {
|
handleSetDefaultApp () {
|
||||||
dispatch('updatePreferences', 'isFileHandler', true)
|
dispatch('updatePreferences', 'isFileHandler', true)
|
||||||
}
|
}
|
||||||
@@ -250,10 +229,7 @@ class PreferencesPage extends React.Component {
|
|||||||
<PreferencesSection title='Default torrent app'>
|
<PreferencesSection title='Default torrent app'>
|
||||||
{this.setDefaultAppButton()}
|
{this.setDefaultAppButton()}
|
||||||
</PreferencesSection>
|
</PreferencesSection>
|
||||||
<PreferencesSection title='General'>
|
{this.setStartupSection()}
|
||||||
{this.setStartupCheckbox()}
|
|
||||||
{this.soundNotificationsCheckbox()}
|
|
||||||
</PreferencesSection>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -262,7 +238,7 @@ class PreferencesPage extends React.Component {
|
|||||||
class PreferencesSection extends React.Component {
|
class PreferencesSection extends React.Component {
|
||||||
static get propTypes () {
|
static get propTypes () {
|
||||||
return {
|
return {
|
||||||
title: PropTypes.string
|
title: React.PropTypes.string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ const LinearProgress = require('material-ui/LinearProgress').default
|
|||||||
|
|
||||||
const TorrentSummary = require('../lib/torrent-summary')
|
const TorrentSummary = require('../lib/torrent-summary')
|
||||||
const TorrentPlayer = require('../lib/torrent-player')
|
const TorrentPlayer = require('../lib/torrent-player')
|
||||||
const { dispatcher } = require('../lib/dispatcher')
|
const { dispatcher, dispatch } = require('../lib/dispatcher')
|
||||||
|
|
||||||
module.exports = class TorrentList extends React.Component {
|
module.exports = class TorrentList extends React.Component {
|
||||||
render () {
|
render () {
|
||||||
@@ -67,8 +67,7 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
style={style}
|
style={style}
|
||||||
className={classes.join(' ')}
|
className={classes.join(' ')}
|
||||||
onContextMenu={infoHash && dispatcher('openTorrentContextMenu', infoHash)}
|
onContextMenu={infoHash && dispatcher('openTorrentContextMenu', infoHash)}
|
||||||
onClick={infoHash && dispatcher('toggleSelectTorrent', infoHash)}
|
onClick={infoHash && dispatcher('toggleSelectTorrent', infoHash)}>
|
||||||
>
|
|
||||||
{this.renderTorrentMetadata(torrentSummary)}
|
{this.renderTorrentMetadata(torrentSummary)}
|
||||||
{infoHash ? this.renderTorrentButtons(torrentSummary) : null}
|
{infoHash ? this.renderTorrentButtons(torrentSummary) : null}
|
||||||
{isSelected ? this.renderTorrentDetails(torrentSummary) : null}
|
{isSelected ? this.renderTorrentDetails(torrentSummary) : null}
|
||||||
@@ -131,8 +130,7 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
}}
|
}}
|
||||||
checked={isActive}
|
checked={isActive}
|
||||||
onClick={stopPropagation}
|
onClick={stopPropagation}
|
||||||
onCheck={dispatcher('toggleTorrent', infoHash)}
|
onCheck={dispatcher('toggleTorrent', infoHash)} />
|
||||||
/>
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -149,7 +147,7 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<div key='progress-bar' style={styles.wrapper}>
|
<div style={styles.wrapper}>
|
||||||
<LinearProgress style={styles.progress} mode='determinate' value={progress} />
|
<LinearProgress style={styles.progress} mode='determinate' value={progress} />
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
@@ -202,7 +200,7 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
const minutesStr = (hours || minutes) ? minutes + 'm' : ''
|
const minutesStr = (hours || minutes) ? minutes + 'm' : ''
|
||||||
const secondsStr = seconds + 's'
|
const secondsStr = seconds + 's'
|
||||||
|
|
||||||
return (<span key='eta'>{hoursStr} {minutesStr} {secondsStr} remaining</span>)
|
return (<span>{hoursStr} {minutesStr} {secondsStr} remaining</span>)
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderTorrentStatus () {
|
function renderTorrentStatus () {
|
||||||
@@ -234,9 +232,8 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
<i
|
<i
|
||||||
key='play-button'
|
key='play-button'
|
||||||
title='Start streaming'
|
title='Start streaming'
|
||||||
className='icon play'
|
className={'icon play'}
|
||||||
onClick={dispatcher('playFile', infoHash)}
|
onClick={dispatcher('playFile', infoHash)}>
|
||||||
>
|
|
||||||
play_circle_outline
|
play_circle_outline
|
||||||
</i>
|
</i>
|
||||||
)
|
)
|
||||||
@@ -249,8 +246,7 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
key='delete-button'
|
key='delete-button'
|
||||||
className='icon delete'
|
className='icon delete'
|
||||||
title='Remove torrent'
|
title='Remove torrent'
|
||||||
onClick={dispatcher('confirmDeleteTorrent', infoHash, false)}
|
onClick={dispatcher('confirmDeleteTorrent', infoHash, false)}>
|
||||||
>
|
|
||||||
close
|
close
|
||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
@@ -359,10 +355,8 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
<td className={'col-size ' + rowClass}>
|
<td className={'col-size ' + rowClass}>
|
||||||
{prettyBytes(file.length)}
|
{prettyBytes(file.length)}
|
||||||
</td>
|
</td>
|
||||||
<td
|
<td className='col-select'
|
||||||
className='col-select'
|
onClick={dispatcher('toggleTorrentFile', infoHash, index)}>
|
||||||
onClick={dispatcher('toggleTorrentFile', infoHash, index)}
|
|
||||||
>
|
|
||||||
<i className='icon deselect-file'>{isSelected ? 'close' : 'add'}</i>
|
<i className='icon deselect-file'>{isSelected ? 'close' : 'add'}</i>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
@@ -376,16 +370,16 @@ module.exports = class TorrentList extends React.Component {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div key='radial-progress' className={'radial-progress ' + cssClass}>
|
<div key='radial-progress' className={'radial-progress ' + cssClass}>
|
||||||
<div className='circle'>
|
<div key='circle' className='circle'>
|
||||||
<div className='mask full' style={transformFill}>
|
<div key='mask-full' className='mask full' style={transformFill}>
|
||||||
<div className='fill' style={transformFill} />
|
<div key='fill' className='fill' style={transformFill} />
|
||||||
</div>
|
</div>
|
||||||
<div className='mask half'>
|
<div key='mask-half' className='mask half'>
|
||||||
<div className='fill' style={transformFill} />
|
<div key='fill' className='fill' style={transformFill} />
|
||||||
<div className='fill fix' style={transformFix} />
|
<div key='fill-fix' className='fill fix' style={transformFix} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className='inset' />
|
<div key='inset' className='inset' />
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -399,11 +393,21 @@ function getErrorMessage (torrentSummary) {
|
|||||||
const err = torrentSummary.error
|
const err = torrentSummary.error
|
||||||
if (err === 'path-missing') {
|
if (err === 'path-missing') {
|
||||||
return (
|
return (
|
||||||
<span key='path-missing'>
|
<span>
|
||||||
Path missing.<br />
|
Torrent data missing.<br />
|
||||||
Fix and restart the app, or delete the torrent.
|
Fix or click to
|
||||||
|
<a onClick={(e) => {
|
||||||
|
e.stopPropagation()
|
||||||
|
restartTorrent(torrentSummary)
|
||||||
|
}}>re-download torrent</a>.
|
||||||
</span>
|
</span>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
return 'Error'
|
return 'Error'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function restartTorrent (summary) {
|
||||||
|
delete summary.path
|
||||||
|
delete summary.error
|
||||||
|
dispatch('startTorrentingSummary', summary.torrentKey)
|
||||||
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ const mm = require('music-metadata')
|
|||||||
const networkAddress = require('network-address')
|
const networkAddress = require('network-address')
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const WebTorrent = require('webtorrent')
|
const WebTorrent = require('webtorrent')
|
||||||
|
const zeroFill = require('zero-fill')
|
||||||
|
|
||||||
const crashReporter = require('../crash-reporter')
|
const crashReporter = require('../crash-reporter')
|
||||||
const config = require('../config')
|
const config = require('../config')
|
||||||
@@ -40,9 +41,10 @@ const VERSION = require('../../package.json').version
|
|||||||
* '0.16.1' -> '0016'
|
* '0.16.1' -> '0016'
|
||||||
* '1.2.5' -> '0102'
|
* '1.2.5' -> '0102'
|
||||||
*/
|
*/
|
||||||
const VERSION_STR = VERSION
|
const VERSION_STR = VERSION.match(/([0-9]+)/g)
|
||||||
.replace(/\d*./g, v => `0${v % 100}`.slice(-2))
|
.slice(0, 2)
|
||||||
.slice(0, 4)
|
.map((v) => zeroFill(2, v))
|
||||||
|
.join('')
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Version prefix string (used in peer ID). WebTorrent uses the Azureus-style
|
* Version prefix string (used in peer ID). WebTorrent uses the Azureus-style
|
||||||
@@ -343,14 +345,12 @@ function getAudioMetadata (infoHash, index) {
|
|||||||
const metadata = { title: file.name }
|
const metadata = { title: file.name }
|
||||||
ipc.send('wt-audio-metadata', infoHash, index, metadata)
|
ipc.send('wt-audio-metadata', infoHash, index, metadata)
|
||||||
|
|
||||||
const options = {
|
const options = { native: false,
|
||||||
native: false,
|
|
||||||
skipCovers: true,
|
skipCovers: true,
|
||||||
fileSize: file.length,
|
fileSize: file.length,
|
||||||
observer: event => {
|
observer: event => {
|
||||||
ipc.send('wt-audio-metadata', infoHash, index, event.metadata)
|
ipc.send('wt-audio-metadata', infoHash, index, event.metadata)
|
||||||
}
|
} }
|
||||||
}
|
|
||||||
const onMetaData = file.done
|
const onMetaData = file.done
|
||||||
// If completed; use direct file access
|
// If completed; use direct file access
|
||||||
? mm.parseFile(path.join(torrent.path, file.path), options)
|
? mm.parseFile(path.join(torrent.path, file.path), options)
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ Version=1.0
|
|||||||
GenericName=BitTorrent Client
|
GenericName=BitTorrent Client
|
||||||
X-GNOME-FullName=$APP_NAME
|
X-GNOME-FullName=$APP_NAME
|
||||||
Comment=Download and share files over BitTorrent
|
Comment=Download and share files over BitTorrent
|
||||||
|
Encoding=UTF-8
|
||||||
Type=Application
|
Type=Application
|
||||||
Icon=webtorrent-desktop
|
Icon=webtorrent-desktop
|
||||||
Terminal=false
|
Terminal=false
|
||||||
@@ -19,11 +20,14 @@ Actions=CreateNewTorrent;OpenTorrentFile;OpenTorrentAddress;
|
|||||||
[Desktop Action CreateNewTorrent]
|
[Desktop Action CreateNewTorrent]
|
||||||
Name=Create New Torrent...
|
Name=Create New Torrent...
|
||||||
Exec=$EXEC_PATH -n
|
Exec=$EXEC_PATH -n
|
||||||
|
Path=$APP_PATH
|
||||||
|
|
||||||
[Desktop Action OpenTorrentFile]
|
[Desktop Action OpenTorrentFile]
|
||||||
Name=Open Torrent File...
|
Name=Open Torrent File...
|
||||||
Exec=$EXEC_PATH -o
|
Exec=$EXEC_PATH -o
|
||||||
|
Path=$APP_PATH
|
||||||
|
|
||||||
[Desktop Action OpenTorrentAddress]
|
[Desktop Action OpenTorrentAddress]
|
||||||
Name=Open Torrent Address...
|
Name=Open Torrent Address...
|
||||||
Exec=$EXEC_PATH -u
|
Exec=$EXEC_PATH -u
|
||||||
|
Path=$APP_PATH
|
||||||
|
|||||||
@@ -28,15 +28,10 @@ module.exports = {
|
|||||||
// Returns a promise that resolves to a Spectron Application once the app has loaded.
|
// Returns a promise that resolves to a Spectron Application once the app has loaded.
|
||||||
// Takes a Tape test. Makes some basic assertions to verify that the app loaded correctly.
|
// Takes a Tape test. Makes some basic assertions to verify that the app loaded correctly.
|
||||||
function createApp (t) {
|
function createApp (t) {
|
||||||
const userDataDir = process.platform === 'win32'
|
|
||||||
? path.join('C:\\Windows\\Temp', 'WebTorrentTest')
|
|
||||||
: path.join('/tmp', 'WebTorrentTest')
|
|
||||||
|
|
||||||
return new Application({
|
return new Application({
|
||||||
path: path.join(__dirname, '..', 'node_modules', '.bin',
|
path: path.join(__dirname, '..', 'node_modules', '.bin',
|
||||||
'electron' + (process.platform === 'win32' ? '.cmd' : '')),
|
'electron' + (process.platform === 'win32' ? '.cmd' : '')),
|
||||||
args: ['-r', path.join(__dirname, 'mocks.js'), path.join(__dirname, '..')],
|
args: ['-r', path.join(__dirname, 'mocks.js'), path.join(__dirname, '..')],
|
||||||
chromeDriverArgs: [`--user-data-dir=${userDataDir}`],
|
|
||||||
env: { NODE_ENV: 'test' },
|
env: { NODE_ENV: 'test' },
|
||||||
waitTimeout: 10e3
|
waitTimeout: 10e3
|
||||||
})
|
})
|
||||||
@@ -58,7 +53,7 @@ function waitForLoad (app, t, opts) {
|
|||||||
}).then(function () {
|
}).then(function () {
|
||||||
return app.webContents.getTitle()
|
return app.webContents.getTitle()
|
||||||
}).then(function (title) {
|
}).then(function (title) {
|
||||||
// Note the window title is WebTorrent, this is the HTML <title>
|
// Note the window title is WebTorrent (BETA), this is the HTML <title>
|
||||||
t.equal(title, 'Main Window', 'html title')
|
t.equal(title, 'Main Window', 'html title')
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -83,12 +78,6 @@ function endTest (app, t, err) {
|
|||||||
// Otherwise, create the reference screenshot: test/screenshots/<platform>/<name>.png
|
// Otherwise, create the reference screenshot: test/screenshots/<platform>/<name>.png
|
||||||
function screenshotCreateOrCompare (app, t, name) {
|
function screenshotCreateOrCompare (app, t, name) {
|
||||||
const ssDir = path.join(__dirname, 'screenshots', process.platform)
|
const ssDir = path.join(__dirname, 'screenshots', process.platform)
|
||||||
|
|
||||||
// check that path exists otherwise create it
|
|
||||||
if (!fs.existsSync(ssDir)) {
|
|
||||||
fs.mkdirSync(ssDir)
|
|
||||||
}
|
|
||||||
|
|
||||||
const ssPath = path.join(ssDir, name + '.png')
|
const ssPath = path.join(ssDir, name + '.png')
|
||||||
let ssBuf
|
let ssBuf
|
||||||
|
|
||||||
@@ -216,7 +205,7 @@ function compareTorrentFile (t, pathActual, fieldsExpected) {
|
|||||||
function extractImportantFields (parsedTorrent) {
|
function extractImportantFields (parsedTorrent) {
|
||||||
const { infoHash, name, announce, urlList, comment } = parsedTorrent
|
const { infoHash, name, announce, urlList, comment } = parsedTorrent
|
||||||
const priv = parsedTorrent.private // private is a reserved word in JS
|
const priv = parsedTorrent.private // private is a reserved word in JS
|
||||||
return { infoHash, name, announce, urlList, comment, private: priv }
|
return { infoHash, name, announce, urlList, comment, 'private': priv }
|
||||||
}
|
}
|
||||||
|
|
||||||
function copy (pathFrom, pathTo) {
|
function copy (pathFrom, pathTo) {
|
||||||
|
|||||||
Reference in New Issue
Block a user