mirror of
https://github.com/Tzahi12345/YoutubeDL-Material.git
synced 2026-03-18 18:50:58 +03:00
Compare commits
24 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
12c227badb | ||
|
|
181a9f842c | ||
|
|
b79d801c0f | ||
|
|
fc3691336d | ||
|
|
bcd879ebc8 | ||
|
|
b646db4828 | ||
|
|
426d52e359 | ||
|
|
17199dd9c0 | ||
|
|
c680c2827b | ||
|
|
2dbf8d31f7 | ||
|
|
a3753e557c | ||
|
|
ec80abdc8e | ||
|
|
1ef7d24c22 | ||
|
|
4b6f6996ae | ||
|
|
c930ee94c5 | ||
|
|
e88edbef5a | ||
|
|
ac13ed3359 | ||
|
|
faae0d44e6 | ||
|
|
7d8ec04ad6 | ||
|
|
8629e6ae9e | ||
|
|
6e311d46a6 | ||
|
|
006e983c14 | ||
|
|
5db3e06a81 | ||
|
|
2ced7b7f91 |
28
Dockerfile
Normal file
28
Dockerfile
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
FROM ubuntu:18.04
|
||||||
|
|
||||||
|
RUN apt-get update && apt-get install -y \
|
||||||
|
nodejs \
|
||||||
|
apache2 \
|
||||||
|
npm \
|
||||||
|
youtube-dl
|
||||||
|
|
||||||
|
# Change directory so that our commands run inside this new directory
|
||||||
|
WORKDIR /var/www/html
|
||||||
|
|
||||||
|
# Copy dependency definitions
|
||||||
|
COPY ./ /var/www/html/
|
||||||
|
|
||||||
|
# Change directory to backend
|
||||||
|
WORKDIR /var/www/html/backend
|
||||||
|
|
||||||
|
# Install dependencies on backend
|
||||||
|
RUN npm install
|
||||||
|
|
||||||
|
# Change back to original directory
|
||||||
|
WORKDIR /var/www/html
|
||||||
|
|
||||||
|
# Expose the port the app runs in
|
||||||
|
EXPOSE 80
|
||||||
|
|
||||||
|
# Run the specified command within the container.
|
||||||
|
CMD ./docker_wrapper.sh
|
||||||
22
README.md
22
README.md
@@ -1,6 +1,8 @@
|
|||||||
# YoutubeDL-Material
|
# YoutubeDL-Material
|
||||||
|
|
||||||
YoutubeDL-Material is a material design frontend for [youtube-dl](https://rg3.github.io/youtube-dl/). It's coded using [Angular 8](https://angular.io/) for the frontend, and [Nodejs](https://nodejs.org/) on the backend.
|
YoutubeDL-Material is a Material Design frontend for [youtube-dl](https://rg3.github.io/youtube-dl/). It's coded using [Angular 8](https://angular.io/) for the frontend, and [Node.js](https://nodejs.org/) on the backend.
|
||||||
|
|
||||||
|
Now with [Docker](#Docker) support!
|
||||||
|
|
||||||
## Getting Started
|
## Getting Started
|
||||||
|
|
||||||
@@ -20,10 +22,12 @@ Dark mode:
|
|||||||
|
|
||||||
### Prerequisites
|
### Prerequisites
|
||||||
|
|
||||||
You need to have a functioning web server for this to work. Also make sure you have these dependencies installed on your system: ffmpeg, nodejs, python. If you don't, run this command:
|
NOTE: If you would like to use Docker, you can go down to the [Docker](#Docker) section for a setup guide.
|
||||||
|
|
||||||
|
You need to have a functioning web server for this to work. Also make sure you have these dependencies installed on your system: nodejs and youtube-dl. If you don't, run this command:
|
||||||
|
|
||||||
```
|
```
|
||||||
sudo apt-get install ffmpeg nodejs python
|
sudo apt-get install nodejs youtube-dl
|
||||||
```
|
```
|
||||||
|
|
||||||
### Installing
|
### Installing
|
||||||
@@ -34,7 +38,7 @@ Drag all the files in `youtubedl-material` to a location accessible to a web ser
|
|||||||
|
|
||||||
Port forward `17442` if you're going to access YoutubeDL-Material from out of your network. This is an important step. Make sure the configuration reflects this appropriately.
|
Port forward `17442` if you're going to access YoutubeDL-Material from out of your network. This is an important step. Make sure the configuration reflects this appropriately.
|
||||||
|
|
||||||
Once the configuration is done, type `sudo nodejs app.js`. This will run the backend server. On your browser, navigate to your installation folder. Try putting in a youtube link to see if it works. If it does, viola! YoutubeDL-Material is now up and running.
|
Once the configuration is done, run `npm install` to install all the backend dependencies. Once that is finished, type `nodejs app.js`. This will run the backend server. On your browser, navigate to your installation folder. Try putting in a youtube link to see if it works. If it does, viola! YoutubeDL-Material is now up and running.
|
||||||
|
|
||||||
If you experience problems, know that it's usually caused by a configuration problem. The first thing you should do is check the console. To get there, right click anywhere on the page and click "Inspect element." Then on the menu that pops up, click console. Look at the error there, and try to investigate.
|
If you experience problems, know that it's usually caused by a configuration problem. The first thing you should do is check the console. To get there, right click anywhere on the page and click "Inspect element." Then on the menu that pops up, click console. Look at the error there, and try to investigate.
|
||||||
|
|
||||||
@@ -73,6 +77,16 @@ The frontend is now complete. The backend is much easier. Just go into the `yout
|
|||||||
|
|
||||||
Finally, port forward the port `17442` and point it to the server's IP address. Make sure the port is also allowed through the firewall.
|
Finally, port forward the port `17442` and point it to the server's IP address. Make sure the port is also allowed through the firewall.
|
||||||
|
|
||||||
|
## Docker
|
||||||
|
|
||||||
|
If you are looking to setup YoutubeDL-Material with Docker, this section is for you. And you're in luck! Docker setup is quite simple.
|
||||||
|
|
||||||
|
1. Run `curl https://github.com/Tzahi12345/YoutubeDL-Material/releases/latest/download/docker-compose.yml -o docker-compose.yml` to download the latest release of `docker-compose.yml`, or go to the [releases](https://github.com/Tzahi12345/YoutubeDL-Material/releases/) page to grab the version you'd like.
|
||||||
|
2. Modify the config items in the `environment` section of `docker-compose.yml` to your liking. Otherwise, the default options will work and point to `http://localhost:8998`. You can find an explanation of these configuration items in [Configuration](#Configuration) section.
|
||||||
|
3. Make sure the port in the `frontend_url` environment variable lines up with the port in the `ports` section.
|
||||||
|
4. Run `docker-compose pull`. This will download the official YoutubeDL-Material docker image.
|
||||||
|
5. Run `docker-compose up` to start it up. If successful, it should say "HTTP(S): Started on port 17442" or something similar. Make sure you can connect to the frontend, and if so, you are done!
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
Feel free to submit a pull request! I have no guidelines as of yet, so no need to worry about that.
|
Feel free to submit a pull request! I have no guidelines as of yet, so no need to worry about that.
|
||||||
|
|||||||
466
backend/app.js
466
backend/app.js
@@ -2,7 +2,6 @@ var async = require('async');
|
|||||||
var fs = require('fs');
|
var fs = require('fs');
|
||||||
var path = require('path');
|
var path = require('path');
|
||||||
var youtubedl = require('youtube-dl');
|
var youtubedl = require('youtube-dl');
|
||||||
var config = require('config');
|
|
||||||
var https = require('https');
|
var https = require('https');
|
||||||
var express = require("express");
|
var express = require("express");
|
||||||
var bodyParser = require("body-parser");
|
var bodyParser = require("body-parser");
|
||||||
@@ -10,6 +9,7 @@ var archiver = require('archiver');
|
|||||||
const low = require('lowdb')
|
const low = require('lowdb')
|
||||||
var URL = require('url').URL;
|
var URL = require('url').URL;
|
||||||
const shortid = require('shortid')
|
const shortid = require('shortid')
|
||||||
|
var config_api = require('./config.js');
|
||||||
|
|
||||||
var app = express();
|
var app = express();
|
||||||
|
|
||||||
@@ -18,65 +18,56 @@ const adapter = new FileSync('db.json');
|
|||||||
const db = low(adapter)
|
const db = low(adapter)
|
||||||
|
|
||||||
// Set some defaults
|
// Set some defaults
|
||||||
db.defaults({ playlists: {
|
db.defaults(
|
||||||
audio: [],
|
{
|
||||||
video: []
|
playlists: {
|
||||||
}}).write();
|
audio: [],
|
||||||
|
video: []
|
||||||
|
},
|
||||||
|
configWriteFlag: false
|
||||||
|
}).write();
|
||||||
|
|
||||||
|
// config values
|
||||||
|
var frontendUrl = null;
|
||||||
|
var backendUrl = null;
|
||||||
|
var backendPort = 17442;
|
||||||
|
var usingEncryption = null;
|
||||||
|
var basePath = null;
|
||||||
|
var audioFolderPath = null;
|
||||||
|
var videoFolderPath = null;
|
||||||
|
var downloadOnlyMode = null;
|
||||||
|
var useDefaultDownloadingAgent = null;
|
||||||
|
var customDownloadingAgent = null;
|
||||||
|
|
||||||
|
// other needed values
|
||||||
|
var options = null; // encryption options
|
||||||
|
var url_domain = null;
|
||||||
|
|
||||||
// check if debug mode
|
// check if debug mode
|
||||||
let debugMode = process.env.YTDL_MODE === 'debug';
|
let debugMode = process.env.YTDL_MODE === 'debug';
|
||||||
|
|
||||||
if (debugMode) console.log('YTDL-Material in debug mode!');
|
if (debugMode) console.log('YTDL-Material in debug mode!');
|
||||||
|
|
||||||
var frontendUrl = !debugMode ? config.get("YoutubeDLMaterial.Host.frontendurl") : 'http://localhost:4200';
|
|
||||||
var backendUrl = config.get("YoutubeDLMaterial.Host.backendurl")
|
|
||||||
var backendPort = 17442;
|
|
||||||
var usingEncryption = config.get("YoutubeDLMaterial.Encryption.use-encryption");
|
|
||||||
var basePath = config.get("YoutubeDLMaterial.Downloader.path-base");
|
|
||||||
var audioFolderPath = config.get("YoutubeDLMaterial.Downloader.path-audio");
|
|
||||||
var videoFolderPath = config.get("YoutubeDLMaterial.Downloader.path-video");
|
|
||||||
var downloadOnlyMode = config.get("YoutubeDLMaterial.Extra.download_only_mode")
|
|
||||||
var useDefaultDownloadingAgent = config.get("YoutubeDLMaterial.Advanced.use_default_downloading_agent");
|
|
||||||
var customDownloadingAgent = config.get("YoutubeDLMaterial.Advanced.custom_downloading_agent");
|
|
||||||
var validDownloadingAgents = [
|
var validDownloadingAgents = [
|
||||||
'aria2c'
|
'aria2c'
|
||||||
]
|
]
|
||||||
if (!useDefaultDownloadingAgent && validDownloadingAgents.indexOf(customDownloadingAgent) !== -1 ) {
|
|
||||||
console.log(`INFO: Using non-default downloading agent \'${customDownloadingAgent}\'`)
|
// don't overwrite config if it already happened.. NOT
|
||||||
|
// let alreadyWritten = db.get('configWriteFlag').value();
|
||||||
|
let writeConfigMode = process.env.write_ytdl_config;
|
||||||
|
var config = null;
|
||||||
|
|
||||||
|
if (writeConfigMode) {
|
||||||
|
setAndLoadConfig();
|
||||||
|
} else {
|
||||||
|
loadConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
var descriptors = {};
|
var descriptors = {};
|
||||||
|
|
||||||
|
|
||||||
if (usingEncryption)
|
|
||||||
{
|
|
||||||
|
|
||||||
var certFilePath = path.resolve(config.get("YoutubeDLMaterial.Encryption.cert-file-path"));
|
|
||||||
var keyFilePath = path.resolve(config.get("YoutubeDLMaterial.Encryption.key-file-path"));
|
|
||||||
|
|
||||||
var certKeyFile = fs.readFileSync(keyFilePath);
|
|
||||||
var certFile = fs.readFileSync(certFilePath);
|
|
||||||
|
|
||||||
var options = {
|
|
||||||
key: certKeyFile,
|
|
||||||
cert: certFile
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
app.use(bodyParser.urlencoded({ extended: false }));
|
app.use(bodyParser.urlencoded({ extended: false }));
|
||||||
app.use(bodyParser.json());
|
app.use(bodyParser.json());
|
||||||
|
|
||||||
var url_domain = new URL(frontendUrl);
|
|
||||||
|
|
||||||
app.use(function(req, res, next) {
|
|
||||||
res.header("Access-Control-Allow-Origin", url_domain.origin);
|
|
||||||
res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
|
|
||||||
next();
|
|
||||||
});
|
|
||||||
|
|
||||||
app.get('/using-encryption', function(req, res) {
|
app.get('/using-encryption', function(req, res) {
|
||||||
res.send(usingEncryption);
|
res.send(usingEncryption);
|
||||||
res.end("yes");
|
res.end("yes");
|
||||||
@@ -94,6 +85,109 @@ function File(id, title, thumbnailURL, isAudio, duration) {
|
|||||||
|
|
||||||
// actual functions
|
// actual functions
|
||||||
|
|
||||||
|
function startServer() {
|
||||||
|
if (usingEncryption)
|
||||||
|
{
|
||||||
|
https.createServer(options, app).listen(backendPort, function() {
|
||||||
|
console.log('HTTPS: Anchor set on 17442');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
app.listen(backendPort,function(){
|
||||||
|
console.log("HTTP: Started on PORT " + backendPort);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function setAndLoadConfig() {
|
||||||
|
await setConfigFromEnv();
|
||||||
|
await loadConfig();
|
||||||
|
// console.log(backendUrl);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function setConfigFromEnv() {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
let config_items = getEnvConfigItems();
|
||||||
|
let success = config_api.setConfigItems(config_items);
|
||||||
|
if (success) {
|
||||||
|
console.log('Config items set using ENV variables.');
|
||||||
|
setTimeout(() => resolve(true), 100);
|
||||||
|
} else {
|
||||||
|
console.log('ERROR: Failed to set config items using ENV variables.');
|
||||||
|
resolve(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadConfig() {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
// get config library
|
||||||
|
// config = require('config');
|
||||||
|
|
||||||
|
frontendUrl = !debugMode ? config_api.getConfigItem('ytdl_frontend_url') : 'http://localhost:4200';
|
||||||
|
backendUrl = config_api.getConfigItem('ytdl_backend_url')
|
||||||
|
backendPort = 17442;
|
||||||
|
usingEncryption = config_api.getConfigItem('ytdl_use_encryption');
|
||||||
|
basePath = config_api.getConfigItem('ytdl_base_path');
|
||||||
|
audioFolderPath = config_api.getConfigItem('ytdl_audio_folder_path');
|
||||||
|
videoFolderPath = config_api.getConfigItem('ytdl_video_folder_path');
|
||||||
|
downloadOnlyMode = config_api.getConfigItem('ytdl_download_only_mode');
|
||||||
|
useDefaultDownloadingAgent = config_api.getConfigItem('ytdl_use_default_downloading_agent');
|
||||||
|
customDownloadingAgent = config_api.getConfigItem('ytdl_custom_downloading_agent');
|
||||||
|
if (!useDefaultDownloadingAgent && validDownloadingAgents.indexOf(customDownloadingAgent) !== -1 ) {
|
||||||
|
console.log(`INFO: Using non-default downloading agent \'${customDownloadingAgent}\'`)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (usingEncryption)
|
||||||
|
{
|
||||||
|
var certFilePath = path.resolve(config_api.getConfigItem('ytdl_cert_file_path'));
|
||||||
|
var keyFilePath = path.resolve(config_api.getConfigItem('ytdl_key_file_path'));
|
||||||
|
|
||||||
|
var certKeyFile = fs.readFileSync(keyFilePath);
|
||||||
|
var certFile = fs.readFileSync(certFilePath);
|
||||||
|
|
||||||
|
options = {
|
||||||
|
key: certKeyFile,
|
||||||
|
cert: certFile
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
url_domain = new URL(frontendUrl);
|
||||||
|
|
||||||
|
// start the server here
|
||||||
|
startServer();
|
||||||
|
|
||||||
|
resolve(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOrigin() {
|
||||||
|
return url_domain.origin;
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets a list of config items that are stored as an environment variable
|
||||||
|
function getEnvConfigItems() {
|
||||||
|
let config_items = [];
|
||||||
|
|
||||||
|
let config_item_keys = Object.keys(config_api.CONFIG_ITEMS);
|
||||||
|
for (let i = 0; i < config_item_keys.length; i++) {
|
||||||
|
let key = config_item_keys[i];
|
||||||
|
if (process['env'][key]) {
|
||||||
|
const config_item = generateEnvVarConfigItem(key);
|
||||||
|
config_items.push(config_item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return config_items;
|
||||||
|
}
|
||||||
|
|
||||||
|
// gets value of a config item and stores it in an object
|
||||||
|
function generateEnvVarConfigItem(key) {
|
||||||
|
return {key: key, value: process['env'][key]};
|
||||||
|
}
|
||||||
|
|
||||||
function getThumbnailMp3(name)
|
function getThumbnailMp3(name)
|
||||||
{
|
{
|
||||||
var obj = getJSONMp3(name);
|
var obj = getJSONMp3(name);
|
||||||
@@ -319,6 +413,30 @@ async function deleteVideoFile(name) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function recFindByExt(base,ext,files,result)
|
||||||
|
{
|
||||||
|
files = files || fs.readdirSync(base)
|
||||||
|
result = result || []
|
||||||
|
|
||||||
|
files.forEach(
|
||||||
|
function (file) {
|
||||||
|
var newbase = path.join(base,file)
|
||||||
|
if ( fs.statSync(newbase).isDirectory() )
|
||||||
|
{
|
||||||
|
result = recFindByExt(newbase,ext,fs.readdirSync(newbase),result)
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ( file.substr(-1*(ext.length+1)) == '.' + ext )
|
||||||
|
{
|
||||||
|
result.push(newbase)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
function getAudioInfos(fileNames) {
|
function getAudioInfos(fileNames) {
|
||||||
let result = [];
|
let result = [];
|
||||||
for (let i = 0; i < fileNames.length; i++) {
|
for (let i = 0; i < fileNames.length; i++) {
|
||||||
@@ -382,31 +500,49 @@ async function getUrlInfos(urls) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
app.use(function(req, res, next) {
|
||||||
|
res.header("Access-Control-Allow-Origin", getOrigin());
|
||||||
|
res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
|
||||||
|
next();
|
||||||
|
});
|
||||||
|
|
||||||
app.post('/tomp3', function(req, res) {
|
app.post('/tomp3', function(req, res) {
|
||||||
var url = req.body.url;
|
var url = req.body.url;
|
||||||
var date = Date.now();
|
var date = Date.now();
|
||||||
var path = audioFolderPath;
|
|
||||||
var audiopath = '%(title)s';
|
var audiopath = '%(title)s';
|
||||||
|
|
||||||
var customQualityConfiguration = req.body.customQualityConfiguration;
|
var customQualityConfiguration = req.body.customQualityConfiguration;
|
||||||
var maxBitrate = req.body.maxBitrate;
|
var maxBitrate = req.body.maxBitrate;
|
||||||
|
var customArgs = req.body.customArgs;
|
||||||
|
var customOutput = req.body.customOutput;
|
||||||
|
|
||||||
let downloadConfig = ['-o', path + audiopath + ".mp3", '-x', '--audio-format', 'mp3', '--write-info-json', '--print-json']
|
|
||||||
|
let downloadConfig = null;
|
||||||
let qualityPath = '';
|
let qualityPath = '';
|
||||||
|
|
||||||
if (customQualityConfiguration) {
|
if (customArgs) {
|
||||||
qualityPath = `-f ${customQualityConfiguration}`;
|
downloadConfig = [customArgs];
|
||||||
} else if (maxBitrate) {
|
} else {
|
||||||
if (!maxBitrate || maxBitrate === '') maxBitrate = '0';
|
if (customOutput) {
|
||||||
qualityPath = `--audio-quality ${maxBitrate}`
|
downloadConfig = ['-o', audioFolderPath + customOutput + '.mp3', '-x', '--audio-format', 'mp3', '--write-info-json', '--print-json'];
|
||||||
}
|
} else {
|
||||||
|
downloadConfig = ['-o', audioFolderPath + audiopath + ".mp3", '-x', '--audio-format', 'mp3', '--write-info-json', '--print-json'];
|
||||||
|
}
|
||||||
|
|
||||||
if (qualityPath !== '') {
|
if (customQualityConfiguration) {
|
||||||
downloadConfig.splice(2, 0, qualityPath);
|
qualityPath = `-f ${customQualityConfiguration}`;
|
||||||
}
|
} else if (maxBitrate) {
|
||||||
|
if (!maxBitrate || maxBitrate === '') maxBitrate = '0';
|
||||||
|
qualityPath = `--audio-quality ${maxBitrate}`
|
||||||
|
}
|
||||||
|
|
||||||
if (!useDefaultDownloadingAgent && customDownloadingAgent === 'aria2c') {
|
if (qualityPath !== '') {
|
||||||
downloadConfig.splice(0, 0, '--external-downloader', 'aria2c');
|
downloadConfig.splice(2, 0, qualityPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!useDefaultDownloadingAgent && customDownloadingAgent === 'aria2c') {
|
||||||
|
downloadConfig.splice(0, 0, '--external-downloader', 'aria2c');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
youtubedl.exec(url, downloadConfig, {}, function(err, output) {
|
youtubedl.exec(url, downloadConfig, {}, function(err, output) {
|
||||||
@@ -434,12 +570,14 @@ app.post('/tomp3', function(req, res) {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
var file_name = output_json['_filename'].replace(/^.*[\\\/]/, '');
|
var file_name = output_json['_filename'].replace(/^.*[\\\/]/, '');
|
||||||
|
var file_path = output_json['_filename'].substring(audioFolderPath.length, output_json['_filename'].length);
|
||||||
|
var alternate_file_path = file_path.substring(0, file_path.length-4);
|
||||||
var alternate_file_name = file_name.substring(0, file_name.length-4);
|
var alternate_file_name = file_name.substring(0, file_name.length-4);
|
||||||
if (alternate_file_name) file_names.push(alternate_file_name);
|
if (alternate_file_path) file_names.push(alternate_file_path);
|
||||||
}
|
}
|
||||||
|
|
||||||
let is_playlist = file_names.length > 1;
|
let is_playlist = file_names.length > 1;
|
||||||
if (!is_playlist) audiopath = file_names[0];
|
// if (!is_playlist) audiopath = file_names[0];
|
||||||
|
|
||||||
var audiopathEncoded = encodeURIComponent(file_names[0]);
|
var audiopathEncoded = encodeURIComponent(file_names[0]);
|
||||||
res.send({
|
res.send({
|
||||||
@@ -455,22 +593,35 @@ app.post('/tomp4', function(req, res) {
|
|||||||
var date = Date.now();
|
var date = Date.now();
|
||||||
var path = videoFolderPath;
|
var path = videoFolderPath;
|
||||||
var videopath = '%(title)s';
|
var videopath = '%(title)s';
|
||||||
|
var customArgs = req.body.customArgs;
|
||||||
|
var customOutput = req.body.customOutput;
|
||||||
|
|
||||||
var selectedHeight = req.body.selectedHeight;
|
var selectedHeight = req.body.selectedHeight;
|
||||||
var customQualityConfiguration = req.body.customQualityConfiguration;
|
var customQualityConfiguration = req.body.customQualityConfiguration;
|
||||||
|
|
||||||
|
let downloadConfig = null;
|
||||||
let qualityPath = 'bestvideo[ext=mp4]+bestaudio[ext=m4a]/mp4';
|
let qualityPath = 'bestvideo[ext=mp4]+bestaudio[ext=m4a]/mp4';
|
||||||
|
|
||||||
if (customQualityConfiguration) {
|
if (customArgs) {
|
||||||
qualityPath = customQualityConfiguration;
|
downloadConfig = [customArgs];
|
||||||
} else if (selectedHeight && selectedHeight !== '') {
|
} else {
|
||||||
qualityPath = `bestvideo[height=${selectedHeight}]+bestaudio/best[height=${selectedHeight}]`;
|
if (customOutput) {
|
||||||
|
downloadConfig = ['-o', path + customOutput + ".mp4", '-f', qualityPath, '--write-info-json', '--print-json'];
|
||||||
|
} else {
|
||||||
|
downloadConfig = ['-o', path + videopath + ".mp4", '-f', qualityPath, '--write-info-json', '--print-json'];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (customQualityConfiguration) {
|
||||||
|
qualityPath = customQualityConfiguration;
|
||||||
|
} else if (selectedHeight && selectedHeight !== '') {
|
||||||
|
qualityPath = `bestvideo[height=${selectedHeight}]+bestaudio/best[height=${selectedHeight}]`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!useDefaultDownloadingAgent && customDownloadingAgent === 'aria2c') {
|
||||||
|
downloadConfig.splice(0, 0, '--external-downloader', 'aria2c');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let downloadConfig = ['-o', path + videopath + ".mp4", '-f', qualityPath, '--write-info-json', '--print-json']
|
|
||||||
if (!useDefaultDownloadingAgent && customDownloadingAgent === 'aria2c') {
|
|
||||||
downloadConfig.splice(0, 0, '--external-downloader', 'aria2c');
|
|
||||||
}
|
|
||||||
youtubedl.exec(url, downloadConfig, {}, function(err, output) {
|
youtubedl.exec(url, downloadConfig, {}, function(err, output) {
|
||||||
if (debugMode) {
|
if (debugMode) {
|
||||||
let new_date = Date.now();
|
let new_date = Date.now();
|
||||||
@@ -506,7 +657,9 @@ app.post('/tomp4', function(req, res) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
var alternate_file_name = file_name.substring(0, file_name.length-4);
|
var alternate_file_name = file_name.substring(0, file_name.length-4);
|
||||||
if (alternate_file_name) file_names.push(alternate_file_name);
|
var file_path = output_json['_filename'].substring(audioFolderPath.length, output_json['_filename'].length);
|
||||||
|
var alternate_file_path = file_path.substring(0, file_path.length-4);
|
||||||
|
if (alternate_file_name) file_names.push(alternate_file_path);
|
||||||
}
|
}
|
||||||
|
|
||||||
let is_playlist = file_names.length > 1;
|
let is_playlist = file_names.length > 1;
|
||||||
@@ -568,31 +721,25 @@ app.post('/fileStatusMp4', function(req, res) {
|
|||||||
app.post('/getMp3s', function(req, res) {
|
app.post('/getMp3s', function(req, res) {
|
||||||
var mp3s = [];
|
var mp3s = [];
|
||||||
var playlists = db.get('playlists.audio').value();
|
var playlists = db.get('playlists.audio').value();
|
||||||
var fullpath = audioFolderPath;
|
var files = recFindByExt(audioFolderPath, 'mp3'); // fs.readdirSync(audioFolderPath);
|
||||||
var files = fs.readdirSync(audioFolderPath);
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
let file = files[i];
|
||||||
|
var file_path = file.substring(audioFolderPath.length, file.length);
|
||||||
|
var id = file_path.substring(0, file_path.length-4);
|
||||||
|
var jsonobj = getJSONMp3(id);
|
||||||
|
if (!jsonobj) continue;
|
||||||
|
var title = jsonobj.title;
|
||||||
|
|
||||||
for (var i in files)
|
if (title.length > 14) // edits title if it's too long
|
||||||
{
|
|
||||||
var nameLength = path.basename(files[i]).length;
|
|
||||||
var ext = path.basename(files[i]).substring(nameLength-4, nameLength);
|
|
||||||
if (ext == ".mp3")
|
|
||||||
{
|
{
|
||||||
var jsonobj = getJSONMp3(path.basename(files[i]).substring(0, path.basename(files[i]).length-4));
|
title = title.substring(0,12) + "...";
|
||||||
if (!jsonobj) continue;
|
|
||||||
var id = path.basename(files[i]).substring(0, path.basename(files[i]).length-4);
|
|
||||||
var title = jsonobj.title;
|
|
||||||
|
|
||||||
if (title.length > 14) // edits title if it's too long
|
|
||||||
{
|
|
||||||
title = title.substring(0,12) + "...";
|
|
||||||
}
|
|
||||||
|
|
||||||
var thumbnail = jsonobj.thumbnail;
|
|
||||||
var duration = jsonobj.duration;
|
|
||||||
var isaudio = true;
|
|
||||||
var file = new File(id, title, thumbnail, isaudio, duration);
|
|
||||||
mp3s.push(file);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var thumbnail = jsonobj.thumbnail;
|
||||||
|
var duration = jsonobj.duration;
|
||||||
|
var isaudio = true;
|
||||||
|
var file_obj = new File(id, title, thumbnail, isaudio, duration);
|
||||||
|
mp3s.push(file_obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
res.send({
|
res.send({
|
||||||
@@ -607,30 +754,25 @@ app.post('/getMp4s', function(req, res) {
|
|||||||
var mp4s = [];
|
var mp4s = [];
|
||||||
var playlists = db.get('playlists.video').value();
|
var playlists = db.get('playlists.video').value();
|
||||||
var fullpath = videoFolderPath;
|
var fullpath = videoFolderPath;
|
||||||
var files = fs.readdirSync(videoFolderPath);
|
var files = recFindByExt(videoFolderPath, 'mp4');
|
||||||
|
for (let i = 0; i < files.length; i++) {
|
||||||
|
let file = files[i];
|
||||||
|
var file_path = file.substring(videoFolderPath.length, file.length);
|
||||||
|
var id = file_path.substring(0, file_path.length-4);
|
||||||
|
var jsonobj = getJSONMp4(id);
|
||||||
|
if (!jsonobj) continue;
|
||||||
|
var title = jsonobj.title;
|
||||||
|
|
||||||
for (var i in files)
|
if (title.length > 14) // edits title if it's too long
|
||||||
{
|
|
||||||
var nameLength = path.basename(files[i]).length;
|
|
||||||
var ext = path.basename(files[i]).substring(nameLength-4, nameLength);
|
|
||||||
if (ext == ".mp4")
|
|
||||||
{
|
{
|
||||||
var jsonobj = getJSONMp4(path.basename(files[i]).substring(0, path.basename(files[i]).length-4));
|
title = title.substring(0,12) + "...";
|
||||||
if (!jsonobj) continue;
|
|
||||||
var id = path.basename(files[i]).substring(0, path.basename(files[i]).length-4);
|
|
||||||
var title = jsonobj.title;
|
|
||||||
|
|
||||||
if (title.length > 14) // edits title if it's too long
|
|
||||||
{
|
|
||||||
title = title.substring(0,12) + "...";
|
|
||||||
}
|
|
||||||
|
|
||||||
var thumbnail = jsonobj.thumbnail;
|
|
||||||
var duration = jsonobj.duration;
|
|
||||||
var isaudio = false;
|
|
||||||
var file = new File(id, title, thumbnail, isaudio, duration);
|
|
||||||
mp4s.push(file);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var thumbnail = jsonobj.thumbnail;
|
||||||
|
var duration = jsonobj.duration;
|
||||||
|
var isaudio = false;
|
||||||
|
var file_obj = new File(id, title, thumbnail, isaudio, duration);
|
||||||
|
mp4s.push(file_obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
res.send({
|
res.send({
|
||||||
@@ -757,12 +899,16 @@ app.post('/downloadFile', async (req, res) => {
|
|||||||
let outputName = req.body.outputName;
|
let outputName = req.body.outputName;
|
||||||
let file = null;
|
let file = null;
|
||||||
if (!is_playlist) {
|
if (!is_playlist) {
|
||||||
|
fileNames = decodeURI(fileNames);
|
||||||
if (type === 'audio') {
|
if (type === 'audio') {
|
||||||
file = __dirname + '/' + 'audio/' + fileNames + '.mp3';
|
file = __dirname + '/' + audioFolderPath + fileNames + '.mp3';
|
||||||
} else if (type === 'video') {
|
} else if (type === 'video') {
|
||||||
file = __dirname + '/' + 'video/' + fileNames + '.mp4';
|
file = __dirname + '/' + videoFolderPath + fileNames + '.mp4';
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
for (let i = 0; i < fileNames.length; i++) {
|
||||||
|
fileNames[i] = decodeURI(fileNames[i]);
|
||||||
|
}
|
||||||
file = await createPlaylistZipFile(fileNames, type, outputName);
|
file = await createPlaylistZipFile(fileNames, type, outputName);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -782,46 +928,48 @@ app.post('/deleteFile', async (req, res) => {
|
|||||||
|
|
||||||
app.get('/video/:id', function(req , res){
|
app.get('/video/:id', function(req , res){
|
||||||
var head;
|
var head;
|
||||||
const path = "video/" + req.params.id + '.mp4';
|
let id = decodeURI(req.params.id);
|
||||||
const stat = fs.statSync(path)
|
const path = "video/" + id + '.mp4';
|
||||||
const fileSize = stat.size
|
const stat = fs.statSync(path)
|
||||||
const range = req.headers.range
|
const fileSize = stat.size
|
||||||
if (range) {
|
const range = req.headers.range
|
||||||
const parts = range.replace(/bytes=/, "").split("-")
|
if (range) {
|
||||||
const start = parseInt(parts[0], 10)
|
const parts = range.replace(/bytes=/, "").split("-")
|
||||||
const end = parts[1]
|
const start = parseInt(parts[0], 10)
|
||||||
? parseInt(parts[1], 10)
|
const end = parts[1]
|
||||||
: fileSize-1
|
? parseInt(parts[1], 10)
|
||||||
const chunksize = (end-start)+1
|
: fileSize-1
|
||||||
const file = fs.createReadStream(path, {start, end})
|
const chunksize = (end-start)+1
|
||||||
if (descriptors[req.params.id]) descriptors[req.params.id].push(file);
|
const file = fs.createReadStream(path, {start, end})
|
||||||
else descriptors[req.params.id] = [file];
|
if (descriptors[id]) descriptors[id].push(file);
|
||||||
file.on('close', function() {
|
else descriptors[id] = [file];
|
||||||
let index = descriptors[req.params.id].indexOf(file);
|
file.on('close', function() {
|
||||||
descriptors[req.params.id].splice(index, 1);
|
let index = descriptors[id].indexOf(file);
|
||||||
if (debugMode) console.log('Successfully closed stream and removed file reference.');
|
descriptors[id].splice(index, 1);
|
||||||
});
|
if (debugMode) console.log('Successfully closed stream and removed file reference.');
|
||||||
head = {
|
});
|
||||||
'Content-Range': `bytes ${start}-${end}/${fileSize}`,
|
head = {
|
||||||
'Accept-Ranges': 'bytes',
|
'Content-Range': `bytes ${start}-${end}/${fileSize}`,
|
||||||
'Content-Length': chunksize,
|
'Accept-Ranges': 'bytes',
|
||||||
'Content-Type': 'video/mp4',
|
'Content-Length': chunksize,
|
||||||
|
'Content-Type': 'video/mp4',
|
||||||
|
}
|
||||||
|
res.writeHead(206, head);
|
||||||
|
file.pipe(res);
|
||||||
|
} else {
|
||||||
|
head = {
|
||||||
|
'Content-Length': fileSize,
|
||||||
|
'Content-Type': 'video/mp4',
|
||||||
|
}
|
||||||
|
res.writeHead(200, head)
|
||||||
|
fs.createReadStream(path).pipe(res)
|
||||||
}
|
}
|
||||||
res.writeHead(206, head);
|
|
||||||
file.pipe(res);
|
|
||||||
} else {
|
|
||||||
head = {
|
|
||||||
'Content-Length': fileSize,
|
|
||||||
'Content-Type': 'video/mp4',
|
|
||||||
}
|
|
||||||
res.writeHead(200, head)
|
|
||||||
fs.createReadStream(path).pipe(res)
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
app.get('/audio/:id', function(req , res){
|
app.get('/audio/:id', function(req , res){
|
||||||
var head;
|
var head;
|
||||||
let path = "audio/" + req.params.id + '.mp3';
|
let id = decodeURI(req.params.id);
|
||||||
|
let path = "audio/" + id + '.mp3';
|
||||||
path = path.replace(/\"/g, '\'');
|
path = path.replace(/\"/g, '\'');
|
||||||
const stat = fs.statSync(path)
|
const stat = fs.statSync(path)
|
||||||
const fileSize = stat.size
|
const fileSize = stat.size
|
||||||
@@ -834,11 +982,11 @@ app.get('/audio/:id', function(req , res){
|
|||||||
: fileSize-1
|
: fileSize-1
|
||||||
const chunksize = (end-start)+1
|
const chunksize = (end-start)+1
|
||||||
const file = fs.createReadStream(path, {start, end});
|
const file = fs.createReadStream(path, {start, end});
|
||||||
if (descriptors[req.params.id]) descriptors[req.params.id].push(file);
|
if (descriptors[id]) descriptors[id].push(file);
|
||||||
else descriptors[req.params.id] = [file];
|
else descriptors[id] = [file];
|
||||||
file.on('close', function() {
|
file.on('close', function() {
|
||||||
let index = descriptors[req.params.id].indexOf(file);
|
let index = descriptors[id].indexOf(file);
|
||||||
descriptors[req.params.id].splice(index, 1);
|
descriptors[id].splice(index, 1);
|
||||||
if (debugMode) console.log('Successfully closed stream and removed file reference.');
|
if (debugMode) console.log('Successfully closed stream and removed file reference.');
|
||||||
});
|
});
|
||||||
head = {
|
head = {
|
||||||
@@ -879,19 +1027,3 @@ app.get('/audio/:id', function(req , res){
|
|||||||
success: !!result
|
success: !!result
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
if (usingEncryption)
|
|
||||||
{
|
|
||||||
https.createServer(options, app).listen(backendPort, function() {
|
|
||||||
console.log('HTTPS: Anchor set on 17442');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
app.listen(backendPort,function(){
|
|
||||||
console.log("HTTP: Started on PORT " + backendPort);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
112
backend/config.js
Normal file
112
backend/config.js
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
let CONFIG_ITEMS = require('./consts.js')['CONFIG_ITEMS'];
|
||||||
|
|
||||||
|
let configPath = 'config/default.json';
|
||||||
|
|
||||||
|
// https://stackoverflow.com/questions/6491463/accessing-nested-javascript-objects-with-string-key
|
||||||
|
Object.byString = function(o, s) {
|
||||||
|
s = s.replace(/\[(\w+)\]/g, '.$1'); // convert indexes to properties
|
||||||
|
s = s.replace(/^\./, ''); // strip a leading dot
|
||||||
|
var a = s.split('.');
|
||||||
|
for (var i = 0, n = a.length; i < n; ++i) {
|
||||||
|
var k = a[i];
|
||||||
|
if (k in o) {
|
||||||
|
o = o[k];
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return o;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getParentPath(path) {
|
||||||
|
let elements = path.split('.');
|
||||||
|
elements.splice(elements.length - 1, 1);
|
||||||
|
return elements.join('.');
|
||||||
|
}
|
||||||
|
|
||||||
|
function getElementNameInConfig(path) {
|
||||||
|
let elements = path.split('.');
|
||||||
|
return elements[elements.length - 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Gets config file and returns as a json
|
||||||
|
*/
|
||||||
|
function getConfigFile() {
|
||||||
|
let raw_data = fs.readFileSync(configPath);
|
||||||
|
try {
|
||||||
|
let parsed_data = JSON.parse(raw_data);
|
||||||
|
return parsed_data;
|
||||||
|
} catch(e) {
|
||||||
|
console.log('ERROR: Failed to get config file');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setConfigFile(config) {
|
||||||
|
try {
|
||||||
|
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
|
||||||
|
return true;
|
||||||
|
} catch(e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getConfigItem(key) {
|
||||||
|
let config_json = getConfigFile();
|
||||||
|
if (!CONFIG_ITEMS[key]) console.log('cannot find config with key ' + key);
|
||||||
|
let path = CONFIG_ITEMS[key]['path'];
|
||||||
|
return Object.byString(config_json, path);
|
||||||
|
};
|
||||||
|
|
||||||
|
function setConfigItem(key, value) {
|
||||||
|
let success = false;
|
||||||
|
let config_json = getConfigFile();
|
||||||
|
let path = CONFIG_ITEMS[key]['path'];
|
||||||
|
let parent_path = getParentPath(path);
|
||||||
|
let element_name = getElementNameInConfig(path);
|
||||||
|
|
||||||
|
let parent_object = Object.byString(config_json, parent_path);
|
||||||
|
if (value === 'false' || value === 'true') {
|
||||||
|
parent_object[element_name] = (value === 'true');
|
||||||
|
} else {
|
||||||
|
parent_object[element_name] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
success = setConfigFile(config_json);
|
||||||
|
|
||||||
|
return success;
|
||||||
|
};
|
||||||
|
|
||||||
|
function setConfigItems(items) {
|
||||||
|
let success = false;
|
||||||
|
let config_json = getConfigFile();
|
||||||
|
for (let i = 0; i < items.length; i++) {
|
||||||
|
let key = items[i].key;
|
||||||
|
let value = items[i].value;
|
||||||
|
|
||||||
|
// if boolean strings, set to booleans again
|
||||||
|
if (value === 'false' || value === 'true') {
|
||||||
|
value = (value === 'true');
|
||||||
|
}
|
||||||
|
|
||||||
|
let item_path = CONFIG_ITEMS[key]['path'];
|
||||||
|
let item_parent_path = getParentPath(item_path);
|
||||||
|
let item_element_name = getElementNameInConfig(item_path);
|
||||||
|
|
||||||
|
let item_parent_object = Object.byString(config_json, item_parent_path);
|
||||||
|
item_parent_object[item_element_name] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
success = setConfigFile(config_json);
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
getConfigItem: getConfigItem,
|
||||||
|
setConfigItem: setConfigItem,
|
||||||
|
setConfigItems: setConfigItems,
|
||||||
|
CONFIG_ITEMS: CONFIG_ITEMS
|
||||||
|
}
|
||||||
@@ -18,7 +18,8 @@
|
|||||||
"title_top": "Youtube Downloader",
|
"title_top": "Youtube Downloader",
|
||||||
"file_manager_enabled": true,
|
"file_manager_enabled": true,
|
||||||
"allow_quality_select": true,
|
"allow_quality_select": true,
|
||||||
"download_only_mode": false
|
"download_only_mode": false,
|
||||||
|
"allow_multi_download_mode": true
|
||||||
},
|
},
|
||||||
"API": {
|
"API": {
|
||||||
"use_youtube_API": false,
|
"use_youtube_API": false,
|
||||||
@@ -30,7 +31,8 @@
|
|||||||
},
|
},
|
||||||
"Advanced": {
|
"Advanced": {
|
||||||
"use_default_downloading_agent": true,
|
"use_default_downloading_agent": true,
|
||||||
"custom_downloading_agent": ""
|
"custom_downloading_agent": "",
|
||||||
|
"allow_advanced_download": false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,7 +18,8 @@
|
|||||||
"title_top": "Youtube Downloader",
|
"title_top": "Youtube Downloader",
|
||||||
"file_manager_enabled": true,
|
"file_manager_enabled": true,
|
||||||
"allow_quality_select": true,
|
"allow_quality_select": true,
|
||||||
"download_only_mode": false
|
"download_only_mode": false,
|
||||||
|
"allow_multi_download_mode": true
|
||||||
},
|
},
|
||||||
"API": {
|
"API": {
|
||||||
"use_youtube_API": false,
|
"use_youtube_API": false,
|
||||||
@@ -30,7 +31,8 @@
|
|||||||
},
|
},
|
||||||
"Advanced": {
|
"Advanced": {
|
||||||
"use_default_downloading_agent": true,
|
"use_default_downloading_agent": true,
|
||||||
"custom_downloading_agent": ""
|
"custom_downloading_agent": "",
|
||||||
|
"allow_advanced_download": false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
100
backend/consts.js
Normal file
100
backend/consts.js
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
var config = require('config');
|
||||||
|
|
||||||
|
let CONFIG_ITEMS = {
|
||||||
|
// Host
|
||||||
|
'ytdl_frontend_url': {
|
||||||
|
'key': 'ytdl_frontend_url',
|
||||||
|
'path': 'YoutubeDLMaterial.Host.frontendurl'
|
||||||
|
},
|
||||||
|
'ytdl_backend_url': {
|
||||||
|
'key': 'ytdl_backend_url',
|
||||||
|
'path': 'YoutubeDLMaterial.Host.backendurl'
|
||||||
|
},
|
||||||
|
|
||||||
|
// Encryption
|
||||||
|
'ytdl_use_encryption': {
|
||||||
|
'key': 'ytdl_use_encryption',
|
||||||
|
'path': 'YoutubeDLMaterial.Encryption.use-encryption'
|
||||||
|
},
|
||||||
|
'ytdl_cert_file_path': {
|
||||||
|
'key': 'ytdl_cert_file_path',
|
||||||
|
'path': 'YoutubeDLMaterial.Encryption.cert-file-path'
|
||||||
|
},
|
||||||
|
'ytdl_key_file_path': {
|
||||||
|
'key': 'ytdl_key_file_path',
|
||||||
|
'path': 'YoutubeDLMaterial.Encryption.key-file-path'
|
||||||
|
},
|
||||||
|
|
||||||
|
// Downloader
|
||||||
|
'ytdl_base_path': {
|
||||||
|
'key': 'ytdl_base_path',
|
||||||
|
'path': 'YoutubeDLMaterial.Downloader.path-base'
|
||||||
|
},
|
||||||
|
'ytdl_audio_folder_path': {
|
||||||
|
'key': 'ytdl_audio_folder_path',
|
||||||
|
'path': 'YoutubeDLMaterial.Downloader.path-audio'
|
||||||
|
},
|
||||||
|
'ytdl_video_folder_path': {
|
||||||
|
'key': 'ytdl_video_folder_path',
|
||||||
|
'path': 'YoutubeDLMaterial.Downloader.path-video'
|
||||||
|
},
|
||||||
|
|
||||||
|
// Extra
|
||||||
|
'ytdl_title_top': {
|
||||||
|
'key': 'ytdl_title_top',
|
||||||
|
'path': 'YoutubeDLMaterial.Extra.title_top'
|
||||||
|
},
|
||||||
|
'ytdl_file_manager_enabled': {
|
||||||
|
'key': 'ytdl_file_manager_enabled',
|
||||||
|
'path': 'YoutubeDLMaterial.Extra.file_manager_enabled'
|
||||||
|
},
|
||||||
|
'ytdl_allow_quality_select': {
|
||||||
|
'key': 'ytdl_allow_quality_select',
|
||||||
|
'path': 'YoutubeDLMaterial.Extra.allow_quality_select'
|
||||||
|
},
|
||||||
|
'ytdl_download_only_mode': {
|
||||||
|
'key': 'ytdl_download_only_mode',
|
||||||
|
'path': 'YoutubeDLMaterial.Extra.download_only_mode'
|
||||||
|
},
|
||||||
|
'ytdl_allow_multi_download_mode': {
|
||||||
|
'key': 'ytdl_allow_multi_download_mode',
|
||||||
|
'path': 'YoutubeDLMaterial.Extra.allow_multi_download_mode'
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
// API
|
||||||
|
'ytdl_use_youtube_api': {
|
||||||
|
'key': 'ytdl_use_youtube_api',
|
||||||
|
'path': 'YoutubeDLMaterial.API.use_youtube_API'
|
||||||
|
},
|
||||||
|
'ytdl_youtube_api_key': {
|
||||||
|
'key': 'ytdl_youtube_api_key',
|
||||||
|
'path': 'YoutubeDLMaterial.API.youtube_API_key'
|
||||||
|
},
|
||||||
|
|
||||||
|
// Themes
|
||||||
|
'ytdl_default_theme': {
|
||||||
|
'key': 'ytdl_default_theme',
|
||||||
|
'path': 'YoutubeDLMaterial.Themes.default_theme'
|
||||||
|
},
|
||||||
|
'ytdl_allow_theme_change': {
|
||||||
|
'key': 'ytdl_allow_theme_change',
|
||||||
|
'path': 'YoutubeDLMaterial.Themes.allow_theme_change'
|
||||||
|
},
|
||||||
|
|
||||||
|
// Advanced
|
||||||
|
'ytdl_use_default_downloading_agent': {
|
||||||
|
'key': 'ytdl_use_default_downloading_agent',
|
||||||
|
'path': 'YoutubeDLMaterial.Advanced.use_default_downloading_agent'
|
||||||
|
},
|
||||||
|
'ytdl_custom_downloading_agent': {
|
||||||
|
'key': 'ytdl_custom_downloading_agent',
|
||||||
|
'path': 'YoutubeDLMaterial.Advanced.custom_downloading_agent'
|
||||||
|
},
|
||||||
|
'ytdl_allow_advanced_download': {
|
||||||
|
'key': 'ytdl_allow_advanced_download',
|
||||||
|
'path': 'YoutubeDLMaterial.Advanced.allow_advanced_download'
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports.CONFIG_ITEMS = CONFIG_ITEMS;
|
||||||
1345
backend/package-lock.json
generated
Normal file
1345
backend/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -4,7 +4,8 @@
|
|||||||
"description": "backend for YoutubeDL-Material",
|
"description": "backend for YoutubeDL-Material",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"test": "echo \"Error: no test specified\" && exit 1"
|
"test": "echo \"Error: no test specified\" && exit 1",
|
||||||
|
"start": "node app.js"
|
||||||
},
|
},
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
@@ -24,6 +25,6 @@
|
|||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"lowdb": "^1.0.0",
|
"lowdb": "^1.0.0",
|
||||||
"shortid": "^2.2.15",
|
"shortid": "^2.2.15",
|
||||||
"youtube-dl": "^2.3.0"
|
"youtube-dl": "^3.0.2"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
35
docker-compose.yml
Normal file
35
docker-compose.yml
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
|
||||||
|
version: "2"
|
||||||
|
services:
|
||||||
|
ytdl_material:
|
||||||
|
build: .
|
||||||
|
environment:
|
||||||
|
# config items
|
||||||
|
ytdl_frontend_url: http://localhost:8998
|
||||||
|
ytdl_backend_url: http://localhost:17442/
|
||||||
|
ytdl_use_encryption: 'false'
|
||||||
|
ytdl_cert_file_path: /etc/letsencrypt/live/example.com/fullchain.pem
|
||||||
|
ytdl_key_file_path: /etc/letsencrypt/live/example.com/privkey.pem
|
||||||
|
ytdl_base_path: http://localhost:17442/
|
||||||
|
ytdl_audio_folder_path: audio/
|
||||||
|
ytdl_video_folder_path: video/
|
||||||
|
ytdl_title_top: Youtube Downloader
|
||||||
|
ytdl_file_manager_enabled: 'true'
|
||||||
|
ytdl_allow_quality_select: 'true'
|
||||||
|
ytdl_download_only_mode: 'false'
|
||||||
|
ytdl_allow_multi_download_mode: true
|
||||||
|
ytdl_use_youtube_api: 'false'
|
||||||
|
ytdl_youtube_api_key: 'false'
|
||||||
|
ytdl_default_theme: default
|
||||||
|
ytdl_allow_theme_change: 'true'
|
||||||
|
ytdl_use_default_downloading_agent: 'true'
|
||||||
|
ytdl_custom_downloading_agent: 'false'
|
||||||
|
ytdl_allow_advanced_download: 'false'
|
||||||
|
# do not touch this
|
||||||
|
write_ytdl_config: 'true'
|
||||||
|
ALLOW_CONFIG_MUTATIONS: 'true'
|
||||||
|
restart: always
|
||||||
|
ports:
|
||||||
|
- "17442:17442"
|
||||||
|
- "8998:80"
|
||||||
|
image: tzahi12345/youtubedl-material:3.1
|
||||||
39
docker_wrapper.sh
Normal file
39
docker_wrapper.sh
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
cd backend
|
||||||
|
|
||||||
|
# Start the first process
|
||||||
|
node app.js &
|
||||||
|
status=$?
|
||||||
|
if [ $status -ne 0 ]; then
|
||||||
|
echo "Failed to start my_first_process: $status"
|
||||||
|
exit $status
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Start the second process
|
||||||
|
apachectl -DFOREGROUND
|
||||||
|
status=$?
|
||||||
|
if [ $status -ne 0 ]; then
|
||||||
|
echo "Failed to start my_second_process: $status"
|
||||||
|
exit $status
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Naive check runs checks once a minute to see if either of the processes exited.
|
||||||
|
# This illustrates part of the heavy lifting you need to do if you want to run
|
||||||
|
# more than one service in a container. The container will exit with an error
|
||||||
|
# if it detects that either of the processes has exited.
|
||||||
|
# Otherwise it will loop forever, waking up every 60 seconds
|
||||||
|
|
||||||
|
while /bin/true; do
|
||||||
|
ps aux |grep node\ app.js # |grep -q -v grep
|
||||||
|
PROCESS_1_STATUS=$?
|
||||||
|
ps aux |grep apache2 # |grep -q -v grep
|
||||||
|
PROCESS_2_STATUS=$?
|
||||||
|
# If the greps above find anything, they will exit with 0 status
|
||||||
|
# If they are not both 0, then something is wrong
|
||||||
|
if [ $PROCESS_1_STATUS -ne 0 -o $PROCESS_2_STATUS -ne 0 ]; then
|
||||||
|
echo "One of the processes has already exited."
|
||||||
|
exit -1
|
||||||
|
fi
|
||||||
|
sleep 60
|
||||||
|
done
|
||||||
0
installer.py
Normal file
0
installer.py
Normal file
12768
package-lock.json
generated
Normal file
12768
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -11,6 +11,10 @@
|
|||||||
"e2e": "ng e2e",
|
"e2e": "ng e2e",
|
||||||
"electron": "ng build --base-href ./ && electron ."
|
"electron": "ng build --base-href ./ && electron ."
|
||||||
},
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": "12.3.1",
|
||||||
|
"npm": "6.10.3"
|
||||||
|
},
|
||||||
"private": true,
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@angular-devkit/core": "^8.3.12",
|
"@angular-devkit/core": "^8.3.12",
|
||||||
@@ -35,7 +39,8 @@
|
|||||||
"tslib": "^1.10.0",
|
"tslib": "^1.10.0",
|
||||||
"videogular2": "^7.0.1",
|
"videogular2": "^7.0.1",
|
||||||
"web-animations-js": "^2.3.2",
|
"web-animations-js": "^2.3.2",
|
||||||
"zone.js": "~0.9.1"
|
"zone.js": "~0.9.1",
|
||||||
|
"typescript": "~3.5.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@angular-devkit/build-angular": "^0.803.24",
|
"@angular-devkit/build-angular": "^0.803.24",
|
||||||
|
|||||||
@@ -25,11 +25,12 @@ import {VgBufferingModule} from 'videogular2/compiled/buffering';
|
|||||||
import { InputDialogComponent } from './input-dialog/input-dialog.component';
|
import { InputDialogComponent } from './input-dialog/input-dialog.component';
|
||||||
import { LazyLoadImageModule, IsVisibleProps } from 'ng-lazyload-image';
|
import { LazyLoadImageModule, IsVisibleProps } from 'ng-lazyload-image';
|
||||||
import { NgxContentLoadingModule } from 'ngx-content-loading';
|
import { NgxContentLoadingModule } from 'ngx-content-loading';
|
||||||
import { audioFilesMouseHovering, videoFilesMouseHovering } from './main/main.component';
|
import { audioFilesMouseHovering, videoFilesMouseHovering, audioFilesOpened, videoFilesOpened } from './main/main.component';
|
||||||
import { CreatePlaylistComponent } from './create-playlist/create-playlist.component';
|
import { CreatePlaylistComponent } from './create-playlist/create-playlist.component';
|
||||||
|
import { DownloadItemComponent } from './download-item/download-item.component';
|
||||||
|
|
||||||
export function isVisible({ event, element, scrollContainer, offset }: IsVisibleProps<any>) {
|
export function isVisible({ event, element, scrollContainer, offset }: IsVisibleProps<any>) {
|
||||||
return (element.id === 'video' ? videoFilesMouseHovering : audioFilesMouseHovering);
|
return (element.id === 'video' ? videoFilesMouseHovering || videoFilesOpened : audioFilesMouseHovering || audioFilesOpened);
|
||||||
}
|
}
|
||||||
|
|
||||||
@NgModule({
|
@NgModule({
|
||||||
@@ -39,7 +40,8 @@ export function isVisible({ event, element, scrollContainer, offset }: IsVisible
|
|||||||
MainComponent,
|
MainComponent,
|
||||||
PlayerComponent,
|
PlayerComponent,
|
||||||
InputDialogComponent,
|
InputDialogComponent,
|
||||||
CreatePlaylistComponent
|
CreatePlaylistComponent,
|
||||||
|
DownloadItemComponent
|
||||||
],
|
],
|
||||||
imports: [
|
imports: [
|
||||||
BrowserModule,
|
BrowserModule,
|
||||||
|
|||||||
16
src/app/download-item/download-item.component.html
Normal file
16
src/app/download-item/download-item.component.html
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<div>
|
||||||
|
<mat-grid-list [rowHeight]="50" [cols]="24">
|
||||||
|
<mat-grid-tile [colspan]="2">
|
||||||
|
<h5 style="display: inline-block; margin-right: 5px; position: relative; top: 5px;">{{queueNumber}}.</h5>
|
||||||
|
</mat-grid-tile>
|
||||||
|
<mat-grid-tile [colspan]="6">
|
||||||
|
<div style="display: inline-block; text-align: center;">ID: {{url_id}}</div>
|
||||||
|
</mat-grid-tile>
|
||||||
|
<mat-grid-tile [colspan]="13">
|
||||||
|
<mat-progress-bar style="width: 80%" [value]="download.percent_complete" [mode]="(download.percent_complete === 0) ? 'indeterminate' : 'determinate'"></mat-progress-bar>
|
||||||
|
</mat-grid-tile>
|
||||||
|
<mat-grid-tile [colspan]="3">
|
||||||
|
<button (click)="cancelTheDownload()" mat-icon-button color="warn"><mat-icon fontSet="material-icons-outlined">cancel</mat-icon></button>
|
||||||
|
</mat-grid-tile>
|
||||||
|
</mat-grid-list>
|
||||||
|
</div>
|
||||||
0
src/app/download-item/download-item.component.scss
Normal file
0
src/app/download-item/download-item.component.scss
Normal file
25
src/app/download-item/download-item.component.spec.ts
Normal file
25
src/app/download-item/download-item.component.spec.ts
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
|
||||||
|
|
||||||
|
import { DownloadItemComponent } from './download-item.component';
|
||||||
|
|
||||||
|
describe('DownloadItemComponent', () => {
|
||||||
|
let component: DownloadItemComponent;
|
||||||
|
let fixture: ComponentFixture<DownloadItemComponent>;
|
||||||
|
|
||||||
|
beforeEach(async(() => {
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [ DownloadItemComponent ]
|
||||||
|
})
|
||||||
|
.compileComponents();
|
||||||
|
}));
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
fixture = TestBed.createComponent(DownloadItemComponent);
|
||||||
|
component = fixture.componentInstance;
|
||||||
|
fixture.detectChanges();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create', () => {
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
41
src/app/download-item/download-item.component.ts
Normal file
41
src/app/download-item/download-item.component.ts
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import { Component, OnInit, Output, EventEmitter, Input } from '@angular/core';
|
||||||
|
import { Download } from 'app/main/main.component';
|
||||||
|
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'app-download-item',
|
||||||
|
templateUrl: './download-item.component.html',
|
||||||
|
styleUrls: ['./download-item.component.scss']
|
||||||
|
})
|
||||||
|
export class DownloadItemComponent implements OnInit {
|
||||||
|
|
||||||
|
@Input() download: Download = {
|
||||||
|
uid: null,
|
||||||
|
type: 'audio',
|
||||||
|
percent_complete: 0,
|
||||||
|
url: 'http://youtube.com/watch?v=17848rufj',
|
||||||
|
downloading: true,
|
||||||
|
is_playlist: false
|
||||||
|
};
|
||||||
|
@Output() cancelDownload = new EventEmitter<Download>();
|
||||||
|
|
||||||
|
@Input() queueNumber = null;
|
||||||
|
|
||||||
|
url_id = null;
|
||||||
|
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
ngOnInit() {
|
||||||
|
if (this.download && this.download.url && this.download.url.includes('youtube')) {
|
||||||
|
const string_id = (this.download.is_playlist ? '?list=' : '?v=')
|
||||||
|
const index_offset = (this.download.is_playlist ? 6 : 3);
|
||||||
|
const end_index = this.download.url.indexOf(string_id) + index_offset;
|
||||||
|
this.url_id = this.download.url.substring(end_index, this.download.url.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cancelTheDownload() {
|
||||||
|
this.cancelDownload.emit(this.download);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
<mat-card class="example-card mat-elevation-z6">
|
<mat-card class="example-card mat-elevation-z6">
|
||||||
<button (click)="deleteFile()" class="deleteButton" mat-icon-button><mat-icon>delete_forever</mat-icon></button>
|
|
||||||
<div style="padding:5px">
|
<div style="padding:5px">
|
||||||
<b><a href="javascript:void(0)" (click)="!isPlaylist ? mainComponent.goToFile(name, isAudio) : mainComponent.goToPlaylist(name, type)">{{title}}</a></b>
|
<b><a href="javascript:void(0)" (click)="!isPlaylist ? mainComponent.goToFile(name, isAudio) : mainComponent.goToPlaylist(name, type)">{{title}}</a></b>
|
||||||
<br/>
|
<br/>
|
||||||
@@ -13,9 +12,6 @@
|
|||||||
</ngx-content-loading>
|
</ngx-content-loading>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<button (click)="deleteFile()" class="deleteButton" mat-icon-button><mat-icon>delete_forever</mat-icon></button>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
</mat-card>
|
</mat-card>
|
||||||
|
|||||||
@@ -116,3 +116,9 @@ mat-form-field.mat-form-field {
|
|||||||
.add-playlist-button {
|
.add-playlist-button {
|
||||||
float: right;
|
float: right;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.advanced-input {
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 20px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
@@ -49,18 +49,61 @@
|
|||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
<br/>
|
<br/>
|
||||||
<mat-checkbox (change)="videoModeChanged($event)" [(ngModel)]="audioOnly" style="float: left; margin-top: -12px">Only Audio</mat-checkbox>
|
<mat-checkbox [disabled]="current_download" (change)="videoModeChanged($event)" [(ngModel)]="audioOnly" style="float: left; margin-top: -12px">Only Audio</mat-checkbox>
|
||||||
|
<mat-checkbox *ngIf="allowMultiDownloadMode" [disabled]="current_download" (change)="multiDownloadModeChanged($event)" [(ngModel)]="multiDownloadMode" style="float: right; margin-top: -12px">Multi-download mode</mat-checkbox>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</mat-card-content>
|
</mat-card-content>
|
||||||
<mat-card-actions>
|
<mat-card-actions>
|
||||||
<button style="margin-left: 8px; margin-bottom: 8px" (click)="downloadClicked()" [disabled]="downloadingfile" type="submit" mat-stroked-button
|
<button style="margin-left: 8px; margin-bottom: 8px" (click)="downloadClicked()" [disabled]="downloadingfile" type="submit" mat-stroked-button
|
||||||
color="accent">Download</button>
|
color="accent">Download</button>
|
||||||
|
<button (click)="cancelDownload()" style="float: right" *ngIf="!!current_download" mat-stroked-button color="warn">Cancel</button>
|
||||||
</mat-card-actions>
|
</mat-card-actions>
|
||||||
</mat-card>
|
</mat-card>
|
||||||
</div>
|
</div>
|
||||||
|
<div *ngIf="false && allowAdvancedDownload" class="big demo-basic">
|
||||||
|
<form style="margin-left: 20px; margin-right: 20px;">
|
||||||
|
<mat-expansion-panel class="big">
|
||||||
|
<mat-expansion-panel-header>
|
||||||
|
<mat-panel-title>
|
||||||
|
Advanced
|
||||||
|
</mat-panel-title>
|
||||||
|
</mat-expansion-panel-header>
|
||||||
|
<div class="container" style="padding-bottom: 20px;">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col">
|
||||||
|
<mat-checkbox color="accent" [disabled]="current_download" (change)="customArgsEnabledChanged($event)" [(ngModel)]="customArgsEnabled" style="position: absolute; z-index: 999" [ngModelOptions]="{standalone: true}">Use custom args</mat-checkbox>
|
||||||
|
<mat-form-field color="accent" class="advanced-input">
|
||||||
|
<input [(ngModel)]="customArgs" [ngModelOptions]="{standalone: true}" [disabled]="!customArgsEnabled" matInput placeholder="Custom args">
|
||||||
|
<mat-hint>No need to include URL, just everything after.</mat-hint>
|
||||||
|
</mat-form-field>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
<mat-checkbox color="accent" [disabled]="current_download" (change)="customOutputEnabledChanged($event)" [(ngModel)]="customOutputEnabled" style="position: absolute; z-index: 999" [ngModelOptions]="{standalone: true}">Use custom output</mat-checkbox>
|
||||||
|
<mat-form-field color="accent" class="advanced-input">
|
||||||
|
<input [(ngModel)]="customOutput" [ngModelOptions]="{standalone: true}" [disabled]="!customOutputEnabled" matInput placeholder="Custom output">
|
||||||
|
<mat-hint><a target="_blank" href="https://github.com/ytdl-org/youtube-dl/blob/master/README.md#output-template">This link</a> will be helpful. Path is relative to the config download path.</mat-hint>
|
||||||
|
</mat-form-field>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</mat-expansion-panel>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
<div *ngIf="multiDownloadMode && downloads.length > 0 && !current_download" style="margin-top: 15px;" class="big demo-basic">
|
||||||
|
<mat-card id="card" style="margin-right: 20px; margin-left: 20px;">
|
||||||
|
<div class="container">
|
||||||
|
<div *ngFor="let download of downloads; let i = index;" class="row">
|
||||||
|
<ng-container *ngIf="current_download !== download">
|
||||||
|
<app-download-item style="width: 100%" [download]="download" [queueNumber]="i+1" (cancelDownload)="cancelDownload($event)"></app-download-item>
|
||||||
|
<mat-divider style="position: relative" *ngIf="i !== downloads.length - 1"></mat-divider>
|
||||||
|
</ng-container>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</mat-card>
|
||||||
|
</div>
|
||||||
<br/>
|
<br/>
|
||||||
<div class="centered big" id="bar_div" *ngIf="downloadingfile;else nofile">
|
<div class="centered big" id="bar_div" *ngIf="current_download && current_download.downloading; else nofile">
|
||||||
<div class="margined">
|
<div class="margined">
|
||||||
<div [ngClass]="(determinateProgress && percentDownloaded === 100)?'make-room-for-spinner':'equal-sizes'" style="display: inline-block; width: 100%; padding-left: 20px" *ngIf="determinateProgress;else indeterminateprogress">
|
<div [ngClass]="(determinateProgress && percentDownloaded === 100)?'make-room-for-spinner':'equal-sizes'" style="display: inline-block; width: 100%; padding-left: 20px" *ngIf="determinateProgress;else indeterminateprogress">
|
||||||
<mat-progress-bar mode="determinate" value="{{percentDownloaded}}"></mat-progress-bar>
|
<mat-progress-bar mode="determinate" value="{{percentDownloaded}}"></mat-progress-bar>
|
||||||
@@ -80,7 +123,7 @@
|
|||||||
</ng-template>
|
</ng-template>
|
||||||
<div style="margin: 20px" *ngIf="fileManagerEnabled">
|
<div style="margin: 20px" *ngIf="fileManagerEnabled">
|
||||||
<mat-accordion>
|
<mat-accordion>
|
||||||
<mat-expansion-panel (mouseleave)="accordionLeft('audio')" (mouseenter)="accordionEntered('audio')" class="big">
|
<mat-expansion-panel (opened)="accordionOpened('audio')" (closed)="accordionClosed('audio')" (mouseleave)="accordionLeft('audio')" (mouseenter)="accordionEntered('audio')" class="big">
|
||||||
<mat-expansion-panel-header>
|
<mat-expansion-panel-header>
|
||||||
<mat-panel-title>
|
<mat-panel-title>
|
||||||
Audio
|
Audio
|
||||||
@@ -115,7 +158,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
</mat-expansion-panel>
|
</mat-expansion-panel>
|
||||||
<mat-expansion-panel (mouseleave)="accordionLeft('video')" (mouseenter)="accordionEntered('video')" class="big">
|
<mat-expansion-panel (opened)="accordionOpened('video')" (closed)="accordionClosed('video')" (mouseleave)="accordionLeft('video')" (mouseenter)="accordionEntered('video')" class="big">
|
||||||
<mat-expansion-panel-header>
|
<mat-expansion-panel-header>
|
||||||
<mat-panel-title>
|
<mat-panel-title>
|
||||||
Video
|
Video
|
||||||
|
|||||||
@@ -18,9 +18,22 @@ import { YoutubeSearchService, Result } from '../youtube-search.service';
|
|||||||
import { Router } from '@angular/router';
|
import { Router } from '@angular/router';
|
||||||
import { CreatePlaylistComponent } from 'app/create-playlist/create-playlist.component';
|
import { CreatePlaylistComponent } from 'app/create-playlist/create-playlist.component';
|
||||||
import { Platform } from '@angular/cdk/platform';
|
import { Platform } from '@angular/cdk/platform';
|
||||||
|
import { v4 as uuid } from 'uuid';
|
||||||
|
|
||||||
export let audioFilesMouseHovering = false;
|
export let audioFilesMouseHovering = false;
|
||||||
export let videoFilesMouseHovering = false;
|
export let videoFilesMouseHovering = false;
|
||||||
|
export let audioFilesOpened = false;
|
||||||
|
export let videoFilesOpened = false;
|
||||||
|
|
||||||
|
export interface Download {
|
||||||
|
uid: string;
|
||||||
|
type: string;
|
||||||
|
url: string;
|
||||||
|
percent_complete: number;
|
||||||
|
downloading: boolean;
|
||||||
|
is_playlist: boolean;
|
||||||
|
fileNames?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-root',
|
selector: 'app-root',
|
||||||
@@ -33,6 +46,11 @@ export class MainComponent implements OnInit {
|
|||||||
determinateProgress = false;
|
determinateProgress = false;
|
||||||
downloadingfile = false;
|
downloadingfile = false;
|
||||||
audioOnly: boolean;
|
audioOnly: boolean;
|
||||||
|
multiDownloadMode = false;
|
||||||
|
customArgsEnabled = false;
|
||||||
|
customArgs = null;
|
||||||
|
customOutputEnabled = false;
|
||||||
|
customOutput = null;
|
||||||
urlError = false;
|
urlError = false;
|
||||||
path = '';
|
path = '';
|
||||||
url = '';
|
url = '';
|
||||||
@@ -43,9 +61,11 @@ export class MainComponent implements OnInit {
|
|||||||
fileManagerEnabled = false;
|
fileManagerEnabled = false;
|
||||||
allowQualitySelect = false;
|
allowQualitySelect = false;
|
||||||
downloadOnlyMode = false;
|
downloadOnlyMode = false;
|
||||||
|
allowMultiDownloadMode = false;
|
||||||
baseStreamPath;
|
baseStreamPath;
|
||||||
audioFolderPath;
|
audioFolderPath;
|
||||||
videoFolderPath;
|
videoFolderPath;
|
||||||
|
allowAdvancedDownload = false;
|
||||||
|
|
||||||
cachedAvailableFormats = {};
|
cachedAvailableFormats = {};
|
||||||
|
|
||||||
@@ -62,6 +82,8 @@ export class MainComponent implements OnInit {
|
|||||||
playlists = {'audio': [], 'video': []};
|
playlists = {'audio': [], 'video': []};
|
||||||
playlist_thumbnails = {};
|
playlist_thumbnails = {};
|
||||||
downloading_content = {'audio': {}, 'video': {}};
|
downloading_content = {'audio': {}, 'video': {}};
|
||||||
|
downloads: Download[] = [];
|
||||||
|
current_download: Download = null;
|
||||||
|
|
||||||
urlForm = new FormControl('', [Validators.required]);
|
urlForm = new FormControl('', [Validators.required]);
|
||||||
|
|
||||||
@@ -166,16 +188,25 @@ export class MainComponent implements OnInit {
|
|||||||
last_valid_url = '';
|
last_valid_url = '';
|
||||||
last_url_check = 0;
|
last_url_check = 0;
|
||||||
|
|
||||||
|
test_download: Download = {
|
||||||
|
uid: null,
|
||||||
|
type: 'audio',
|
||||||
|
percent_complete: 0,
|
||||||
|
url: 'http://youtube.com/watch?v=17848rufj',
|
||||||
|
downloading: true,
|
||||||
|
is_playlist: false
|
||||||
|
};
|
||||||
|
|
||||||
constructor(private postsService: PostsService, private youtubeSearch: YoutubeSearchService, public snackBar: MatSnackBar,
|
constructor(private postsService: PostsService, private youtubeSearch: YoutubeSearchService, public snackBar: MatSnackBar,
|
||||||
private router: Router, public dialog: MatDialog, private platform: Platform) {
|
private router: Router, public dialog: MatDialog, private platform: Platform) {
|
||||||
this.audioOnly = false;
|
this.audioOnly = false;
|
||||||
|
|
||||||
|
|
||||||
// loading config
|
// loading config
|
||||||
this.postsService.loadNavItems().subscribe(result => { // loads settings
|
this.postsService.loadNavItems().subscribe(result => { // loads settings
|
||||||
const backendUrl = result['YoutubeDLMaterial']['Host']['backendurl'];
|
const backendUrl = result['YoutubeDLMaterial']['Host']['backendurl'];
|
||||||
this.fileManagerEnabled = result['YoutubeDLMaterial']['Extra']['file_manager_enabled'];
|
this.fileManagerEnabled = result['YoutubeDLMaterial']['Extra']['file_manager_enabled'];
|
||||||
this.downloadOnlyMode = result['YoutubeDLMaterial']['Extra']['download_only_mode'];
|
this.downloadOnlyMode = result['YoutubeDLMaterial']['Extra']['download_only_mode'];
|
||||||
|
this.allowMultiDownloadMode = result['YoutubeDLMaterial']['Extra']['allow_multi_download_mode'];
|
||||||
this.baseStreamPath = result['YoutubeDLMaterial']['Downloader']['path-base'];
|
this.baseStreamPath = result['YoutubeDLMaterial']['Downloader']['path-base'];
|
||||||
this.audioFolderPath = result['YoutubeDLMaterial']['Downloader']['path-audio'];
|
this.audioFolderPath = result['YoutubeDLMaterial']['Downloader']['path-audio'];
|
||||||
this.videoFolderPath = result['YoutubeDLMaterial']['Downloader']['path-video'];
|
this.videoFolderPath = result['YoutubeDLMaterial']['Downloader']['path-video'];
|
||||||
@@ -183,6 +214,7 @@ export class MainComponent implements OnInit {
|
|||||||
result['YoutubeDLMaterial']['API']['youtube_API_key'];
|
result['YoutubeDLMaterial']['API']['youtube_API_key'];
|
||||||
this.youtubeAPIKey = this.youtubeSearchEnabled ? result['YoutubeDLMaterial']['API']['youtube_API_key'] : null;
|
this.youtubeAPIKey = this.youtubeSearchEnabled ? result['YoutubeDLMaterial']['API']['youtube_API_key'] : null;
|
||||||
this.allowQualitySelect = result['YoutubeDLMaterial']['Extra']['allow_quality_select'];
|
this.allowQualitySelect = result['YoutubeDLMaterial']['Extra']['allow_quality_select'];
|
||||||
|
this.allowAdvancedDownload = result['YoutubeDLMaterial']['Advanced']['allow_advanced_download'];
|
||||||
|
|
||||||
this.postsService.path = backendUrl;
|
this.postsService.path = backendUrl;
|
||||||
this.postsService.startPath = backendUrl;
|
this.postsService.startPath = backendUrl;
|
||||||
@@ -197,6 +229,18 @@ export class MainComponent implements OnInit {
|
|||||||
this.youtubeSearch.initializeAPI(this.youtubeAPIKey);
|
this.youtubeSearch.initializeAPI(this.youtubeAPIKey);
|
||||||
this.attachToInput();
|
this.attachToInput();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// set final cache items
|
||||||
|
if (this.allowAdvancedDownload) {
|
||||||
|
if (localStorage.getItem('customArgsEnabled') !== null) {
|
||||||
|
this.customArgsEnabled = localStorage.getItem('customArgsEnabled') === 'true';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (localStorage.getItem('customOutputEnabled') !== null) {
|
||||||
|
this.customOutputEnabled = localStorage.getItem('customOutputEnabled') === 'true';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}, error => {
|
}, error => {
|
||||||
console.log(error);
|
console.log(error);
|
||||||
});
|
});
|
||||||
@@ -337,61 +381,89 @@ export class MainComponent implements OnInit {
|
|||||||
if (localStorage.getItem('audioOnly') !== null) {
|
if (localStorage.getItem('audioOnly') !== null) {
|
||||||
this.audioOnly = localStorage.getItem('audioOnly') === 'true';
|
this.audioOnly = localStorage.getItem('audioOnly') === 'true';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (localStorage.getItem('multiDownloadMode') !== null) {
|
||||||
|
this.multiDownloadMode = localStorage.getItem('multiDownloadMode') === 'true';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// download helpers
|
// download helpers
|
||||||
|
|
||||||
downloadHelperMp3(name, is_playlist = false, forceView = false) {
|
downloadHelperMp3(name, is_playlist = false, forceView = false, new_download = null) {
|
||||||
this.downloadingfile = false;
|
this.downloadingfile = false;
|
||||||
|
|
||||||
// if download only mode, just download the file. no redirect
|
if (new_download && this.current_download !== new_download) {
|
||||||
if (forceView === false && this.downloadOnlyMode && !this.iOS) {
|
// console.log('mismatched downloads');
|
||||||
if (is_playlist) {
|
} else if (!this.multiDownloadMode) {
|
||||||
const zipName = name[0].split(' ')[0] + name[1].split(' ')[0];
|
// if download only mode, just download the file. no redirect
|
||||||
this.downloadPlaylist(name, 'audio', zipName);
|
if (forceView === false && this.downloadOnlyMode && !this.iOS) {
|
||||||
|
if (is_playlist) {
|
||||||
|
const zipName = name[0].split(' ')[0] + name[1].split(' ')[0];
|
||||||
|
this.downloadPlaylist(name, 'audio', zipName);
|
||||||
|
} else {
|
||||||
|
this.downloadAudioFile(decodeURI(name));
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this.downloadAudioFile(decodeURI(name));
|
if (is_playlist) {
|
||||||
}
|
this.router.navigate(['/player', {fileNames: name.join('|nvr|'), type: 'audio'}]);
|
||||||
} else {
|
// window.location.href = this.baseStreamPath + this.audioFolderPath + name[0] + '.mp3';
|
||||||
if (is_playlist) {
|
} else {
|
||||||
this.router.navigate(['/player', {fileNames: name.join('|nvr|'), type: 'audio'}]);
|
this.router.navigate(['/player', {fileNames: name, type: 'audio'}]);
|
||||||
// window.location.href = this.baseStreamPath + this.audioFolderPath + name[0] + '.mp3';
|
// window.location.href = this.baseStreamPath + this.audioFolderPath + name + '.mp3';
|
||||||
} else {
|
}
|
||||||
this.router.navigate(['/player', {fileNames: name, type: 'audio'}]);
|
|
||||||
// window.location.href = this.baseStreamPath + this.audioFolderPath + name + '.mp3';
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// remove download from current downloads
|
||||||
|
this.removeDownloadFromCurrentDownloads(new_download);
|
||||||
|
|
||||||
// reloads mp3s
|
// reloads mp3s
|
||||||
if (this.fileManagerEnabled) {
|
if (this.fileManagerEnabled) {
|
||||||
this.getMp3s();
|
this.getMp3s();
|
||||||
|
setTimeout(() => {
|
||||||
|
this.audioFileCards.forEach(filecard => {
|
||||||
|
filecard.onHoverResponse();
|
||||||
|
});
|
||||||
|
}, 200);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
downloadHelperMp4(name, is_playlist = false, forceView = false) {
|
downloadHelperMp4(name, is_playlist = false, forceView = false, new_download = null) {
|
||||||
this.downloadingfile = false;
|
this.downloadingfile = false;
|
||||||
|
|
||||||
// if download only mode, just download the file. no redirect
|
if (new_download && this.current_download !== new_download) {
|
||||||
if (forceView === false && this.downloadOnlyMode) {
|
// console.log('mismatched downloads');
|
||||||
if (is_playlist) {
|
} else if (!this.multiDownloadMode) {
|
||||||
const zipName = name[0].split(' ')[0] + name[1].split(' ')[0];
|
// if download only mode, just download the file. no redirect
|
||||||
this.downloadPlaylist(name, 'video', zipName);
|
if (forceView === false && this.downloadOnlyMode) {
|
||||||
|
if (is_playlist) {
|
||||||
|
const zipName = name[0].split(' ')[0] + name[1].split(' ')[0];
|
||||||
|
this.downloadPlaylist(name, 'video', zipName);
|
||||||
|
} else {
|
||||||
|
this.downloadVideoFile(decodeURI(name));
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this.downloadVideoFile(decodeURI(name));
|
if (is_playlist) {
|
||||||
}
|
this.router.navigate(['/player', {fileNames: name.join('|nvr|'), type: 'video'}]);
|
||||||
} else {
|
// window.location.href = this.baseStreamPath + this.videoFolderPath + name[0] + '.mp4';
|
||||||
if (is_playlist) {
|
} else {
|
||||||
this.router.navigate(['/player', {fileNames: name.join('|nvr|'), type: 'video'}]);
|
this.router.navigate(['/player', {fileNames: name, type: 'video'}]);
|
||||||
// window.location.href = this.baseStreamPath + this.videoFolderPath + name[0] + '.mp4';
|
// window.location.href = this.baseStreamPath + this.videoFolderPath + name + '.mp4';
|
||||||
} else {
|
}
|
||||||
this.router.navigate(['/player', {fileNames: name, type: 'video'}]);
|
|
||||||
// window.location.href = this.baseStreamPath + this.videoFolderPath + name + '.mp4';
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// remove download from current downloads
|
||||||
|
this.removeDownloadFromCurrentDownloads(new_download);
|
||||||
|
|
||||||
// reloads mp4s
|
// reloads mp4s
|
||||||
if (this.fileManagerEnabled) {
|
if (this.fileManagerEnabled) {
|
||||||
this.getMp4s();
|
this.getMp4s();
|
||||||
|
setTimeout(() => {
|
||||||
|
this.videoFileCards.forEach(filecard => {
|
||||||
|
filecard.onHoverResponse();
|
||||||
|
});
|
||||||
|
}, 200);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -402,6 +474,17 @@ export class MainComponent implements OnInit {
|
|||||||
this.path = '';
|
this.path = '';
|
||||||
|
|
||||||
if (this.audioOnly) {
|
if (this.audioOnly) {
|
||||||
|
// create download object
|
||||||
|
const new_download: Download = {
|
||||||
|
uid: uuid(),
|
||||||
|
type: 'audio',
|
||||||
|
percent_complete: 0,
|
||||||
|
url: this.url,
|
||||||
|
downloading: true,
|
||||||
|
is_playlist: this.url.includes('playlist')
|
||||||
|
};
|
||||||
|
this.downloads.push(new_download);
|
||||||
|
if (!this.current_download && !this.multiDownloadMode) { this.current_download = new_download };
|
||||||
this.downloadingfile = true;
|
this.downloadingfile = true;
|
||||||
|
|
||||||
let customQualityConfiguration = null;
|
let customQualityConfiguration = null;
|
||||||
@@ -412,18 +495,40 @@ export class MainComponent implements OnInit {
|
|||||||
customQualityConfiguration = audio_formats[this.selectedQuality]['format_id'];
|
customQualityConfiguration = audio_formats[this.selectedQuality]['format_id'];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const customArgs = (this.customArgsEnabled ? this.customArgs : null);
|
||||||
|
const customOutput = (this.customOutputEnabled ? this.customOutput : null);
|
||||||
|
|
||||||
this.postsService.makeMP3(this.url, (this.selectedQuality === '' ? null : this.selectedQuality),
|
this.postsService.makeMP3(this.url, (this.selectedQuality === '' ? null : this.selectedQuality),
|
||||||
customQualityConfiguration).subscribe(posts => {
|
customQualityConfiguration, customArgs, customOutput).subscribe(posts => {
|
||||||
|
// update download object
|
||||||
|
new_download.downloading = false;
|
||||||
|
new_download.percent_complete = 100;
|
||||||
|
|
||||||
const is_playlist = !!(posts['file_names']);
|
const is_playlist = !!(posts['file_names']);
|
||||||
this.path = is_playlist ? posts['file_names'] : posts['audiopathEncoded'];
|
this.path = is_playlist ? posts['file_names'] : posts['audiopathEncoded'];
|
||||||
|
|
||||||
if (this.path !== '-1') {
|
if (this.path !== '-1') {
|
||||||
this.downloadHelperMp3(this.path, is_playlist);
|
this.downloadHelperMp3(this.path, is_playlist, false, new_download);
|
||||||
}
|
}
|
||||||
}, error => { // can't access server
|
}, error => { // can't access server
|
||||||
this.downloadingfile = false;
|
this.downloadingfile = false;
|
||||||
this.openSnackBar('Download failed!', 'OK.');
|
this.openSnackBar('Download failed!', 'OK.');
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
|
// create download object
|
||||||
|
const new_download: Download = {
|
||||||
|
uid: uuid(),
|
||||||
|
type: 'video',
|
||||||
|
percent_complete: 0,
|
||||||
|
url: this.url,
|
||||||
|
downloading: true,
|
||||||
|
is_playlist: this.url.includes('playlist')
|
||||||
|
};
|
||||||
|
this.downloads.push(new_download);
|
||||||
|
if (!this.current_download && !this.multiDownloadMode) { this.current_download = new_download };
|
||||||
|
this.downloadingfile = true;
|
||||||
|
|
||||||
let customQualityConfiguration = null;
|
let customQualityConfiguration = null;
|
||||||
const cachedFormatsExists = this.cachedAvailableFormats[this.url] && this.cachedAvailableFormats[this.url]['formats'];
|
const cachedFormatsExists = this.cachedAvailableFormats[this.url] && this.cachedAvailableFormats[this.url]['formats'];
|
||||||
if (cachedFormatsExists) {
|
if (cachedFormatsExists) {
|
||||||
@@ -433,24 +538,67 @@ export class MainComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.downloadingfile = true;
|
const customArgs = (this.customArgsEnabled ? this.customArgs : null);
|
||||||
|
const customOutput = (this.customOutputEnabled ? this.customOutput : null);
|
||||||
|
|
||||||
this.postsService.makeMP4(this.url, (this.selectedQuality === '' ? null : this.selectedQuality),
|
this.postsService.makeMP4(this.url, (this.selectedQuality === '' ? null : this.selectedQuality),
|
||||||
customQualityConfiguration).subscribe(posts => {
|
customQualityConfiguration, customArgs, customOutput).subscribe(posts => {
|
||||||
|
// update download object
|
||||||
|
new_download.downloading = false;
|
||||||
|
new_download.percent_complete = 100;
|
||||||
|
|
||||||
const is_playlist = !!(posts['file_names']);
|
const is_playlist = !!(posts['file_names']);
|
||||||
this.path = is_playlist ? posts['file_names'] : posts['videopathEncoded'];
|
this.path = is_playlist ? posts['file_names'] : posts['videopathEncoded'];
|
||||||
|
|
||||||
if (this.path !== '-1') {
|
if (this.path !== '-1') {
|
||||||
this.downloadHelperMp4(this.path, is_playlist);
|
this.downloadHelperMp4(this.path, is_playlist, false, new_download);
|
||||||
}
|
}
|
||||||
}, error => { // can't access server
|
}, error => { // can't access server
|
||||||
this.downloadingfile = false;
|
this.downloadingfile = false;
|
||||||
this.openSnackBar('Download failed!', 'OK.');
|
this.openSnackBar('Download failed!', 'OK.');
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.multiDownloadMode) {
|
||||||
|
this.url = '';
|
||||||
|
this.downloadingfile = false;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this.urlError = true;
|
this.urlError = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// download canceled handler
|
||||||
|
cancelDownload(download_to_cancel = null) {
|
||||||
|
// if one is provided, cancel that one. otherwise, remove the current one
|
||||||
|
if (download_to_cancel) {
|
||||||
|
this.removeDownloadFromCurrentDownloads(download_to_cancel)
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.downloadingfile = false;
|
||||||
|
this.current_download.downloading = false;
|
||||||
|
this.current_download = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDownloadByUID(uid) {
|
||||||
|
const index = this.downloads.findIndex(download => download.uid === uid);
|
||||||
|
if (index !== -1) {
|
||||||
|
return this.downloads[index];
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
removeDownloadFromCurrentDownloads(download_to_remove) {
|
||||||
|
const index = this.downloads.indexOf(download_to_remove);
|
||||||
|
if (index !== -1) {
|
||||||
|
this.downloads.splice(index, 1);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
downloadAudioFile(name) {
|
downloadAudioFile(name) {
|
||||||
this.downloading_content['audio'][name] = true;
|
this.downloading_content['audio'][name] = true;
|
||||||
this.postsService.downloadFileFromServer(name, 'audio').subscribe(res => {
|
this.postsService.downloadFileFromServer(name, 'audio').subscribe(res => {
|
||||||
@@ -617,6 +765,24 @@ export class MainComponent implements OnInit {
|
|||||||
localStorage.setItem('audioOnly', new_val.checked.toString());
|
localStorage.setItem('audioOnly', new_val.checked.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
multiDownloadModeChanged(new_val) {
|
||||||
|
localStorage.setItem('multiDownloadMode', new_val.checked.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
customArgsEnabledChanged(new_val) {
|
||||||
|
localStorage.setItem('customArgsEnabled', new_val.checked.toString());
|
||||||
|
if (new_val.checked === true && this.customOutputEnabled) {
|
||||||
|
this.customOutputEnabled = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
customOutputEnabledChanged(new_val) {
|
||||||
|
localStorage.setItem('customOutputEnabled', new_val.checked.toString());
|
||||||
|
if (new_val.checked === true && this.customArgsEnabled) {
|
||||||
|
this.customArgsEnabled = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
getAudioAndVideoFormats(formats): any[] {
|
getAudioAndVideoFormats(formats): any[] {
|
||||||
const audio_formats = {};
|
const audio_formats = {};
|
||||||
const video_formats = {};
|
const video_formats = {};
|
||||||
@@ -700,6 +866,22 @@ export class MainComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
accordionOpened(type) {
|
||||||
|
if (type === 'audio') {
|
||||||
|
audioFilesOpened = true;
|
||||||
|
} else if (type === 'video') {
|
||||||
|
videoFilesOpened = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
accordionClosed(type) {
|
||||||
|
if (type === 'audio') {
|
||||||
|
audioFilesOpened = false;
|
||||||
|
} else if (type === 'video') {
|
||||||
|
videoFilesOpened = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// creating a playlist
|
// creating a playlist
|
||||||
openCreatePlaylistDialog(type) {
|
openCreatePlaylistDialog(type) {
|
||||||
const dialogRef = this.dialog.open(CreatePlaylistComponent, {
|
const dialogRef = this.dialog.open(CreatePlaylistComponent, {
|
||||||
|
|||||||
@@ -76,7 +76,7 @@ export class PlayerComponent implements OnInit {
|
|||||||
for (let i = 0; i < this.fileNames.length; i++) {
|
for (let i = 0; i < this.fileNames.length; i++) {
|
||||||
const fileName = this.fileNames[i];
|
const fileName = this.fileNames[i];
|
||||||
const baseLocation = (this.type === 'audio') ? this.audioFolderPath : this.videoFolderPath;
|
const baseLocation = (this.type === 'audio') ? this.audioFolderPath : this.videoFolderPath;
|
||||||
const fullLocation = this.baseStreamPath + baseLocation + fileName; // + (this.type === 'audio' ? '.mp3' : '.mp4');
|
const fullLocation = this.baseStreamPath + baseLocation + encodeURI(fileName); // + (this.type === 'audio' ? '.mp3' : '.mp4');
|
||||||
const mediaObject: IMedia = {
|
const mediaObject: IMedia = {
|
||||||
title: fileName,
|
title: fileName,
|
||||||
src: fullLocation,
|
src: fullLocation,
|
||||||
|
|||||||
@@ -43,16 +43,22 @@ export class PostsService {
|
|||||||
return this.http.get(this.startPath + 'audiofolder');
|
return this.http.get(this.startPath + 'audiofolder');
|
||||||
}
|
}
|
||||||
|
|
||||||
makeMP3(url: string, selectedQuality: string, customQualityConfiguration: string) {
|
// tslint:disable-next-line: max-line-length
|
||||||
|
makeMP3(url: string, selectedQuality: string, customQualityConfiguration: string, customArgs: string = null, customOutput: string = null) {
|
||||||
return this.http.post(this.path + 'tomp3', {url: url,
|
return this.http.post(this.path + 'tomp3', {url: url,
|
||||||
maxBitrate: selectedQuality,
|
maxBitrate: selectedQuality,
|
||||||
customQualityConfiguration: customQualityConfiguration});
|
customQualityConfiguration: customQualityConfiguration,
|
||||||
|
customArgs: customArgs,
|
||||||
|
customOutput: customOutput});
|
||||||
}
|
}
|
||||||
|
|
||||||
makeMP4(url: string, selectedQuality: string, customQualityConfiguration: string) {
|
// tslint:disable-next-line: max-line-length
|
||||||
|
makeMP4(url: string, selectedQuality: string, customQualityConfiguration: string, customArgs: string = null, customOutput: string = null) {
|
||||||
return this.http.post(this.path + 'tomp4', {url: url,
|
return this.http.post(this.path + 'tomp4', {url: url,
|
||||||
selectedHeight: selectedQuality,
|
selectedHeight: selectedQuality,
|
||||||
customQualityConfiguration: customQualityConfiguration});
|
customQualityConfiguration: customQualityConfiguration,
|
||||||
|
customArgs: customArgs,
|
||||||
|
customOutput: customOutput});
|
||||||
}
|
}
|
||||||
|
|
||||||
getFileStatusMp3(name: string) {
|
getFileStatusMp3(name: string) {
|
||||||
|
|||||||
@@ -18,7 +18,8 @@
|
|||||||
"title_top": "Youtube Downloader",
|
"title_top": "Youtube Downloader",
|
||||||
"file_manager_enabled": true,
|
"file_manager_enabled": true,
|
||||||
"allow_quality_select": true,
|
"allow_quality_select": true,
|
||||||
"download_only_mode": false
|
"download_only_mode": false,
|
||||||
|
"allow_multi_download_mode": true
|
||||||
},
|
},
|
||||||
"API": {
|
"API": {
|
||||||
"use_youtube_API": false,
|
"use_youtube_API": false,
|
||||||
@@ -30,7 +31,8 @@
|
|||||||
},
|
},
|
||||||
"Advanced": {
|
"Advanced": {
|
||||||
"use_default_downloading_agent": true,
|
"use_default_downloading_agent": true,
|
||||||
"custom_downloading_agent": ""
|
"custom_downloading_agent": "",
|
||||||
|
"allow_advanced_download": true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
<link href="https://fonts.googleapis.com/css?family=Roboto&display=swap" rel="stylesheet">
|
<link href="https://fonts.googleapis.com/css?family=Roboto&display=swap" rel="stylesheet">
|
||||||
<link rel="icon" type="image/x-icon" href="favicon.ico">
|
<link rel="icon" type="image/x-icon" href="favicon.ico">
|
||||||
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css" integrity="sha384-Gn5384xqQ1aoWXA+058RXPxPg6fy4IWvTNh0E263XmFcJlSAwiGgFAW/dAiS6JXm" crossorigin="anonymous">
|
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css" integrity="sha384-Gn5384xqQ1aoWXA+058RXPxPg6fy4IWvTNh0E263XmFcJlSAwiGgFAW/dAiS6JXm" crossorigin="anonymous">
|
||||||
<link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
|
<link href="https://fonts.googleapis.com/icon?family=Material+Icons|Material+Icons+Outlined" rel="stylesheet"/>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<app-root></app-root>
|
<app-root></app-root>
|
||||||
|
|||||||
Reference in New Issue
Block a user