mirror of
https://github.com/alexta69/metube.git
synced 2025-04-04 04:37:39 +03:00
Merge branch 'master' into master
This commit is contained in:
commit
e281d61773
3 changed files with 80 additions and 57 deletions
|
@ -12,6 +12,7 @@ import json
|
||||||
import pathlib
|
import pathlib
|
||||||
|
|
||||||
from ytdl import DownloadQueueNotifier, DownloadQueue
|
from ytdl import DownloadQueueNotifier, DownloadQueue
|
||||||
|
from yt_dlp.version import __version__ as yt_dlp_version
|
||||||
|
|
||||||
log = logging.getLogger('main')
|
log = logging.getLogger('main')
|
||||||
|
|
||||||
|
@ -246,6 +247,10 @@ def robots(request):
|
||||||
)
|
)
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
@routes.get(config.URL_PREFIX + 'version')
|
||||||
|
def version(request):
|
||||||
|
return web.json_response({"version": yt_dlp_version})
|
||||||
|
|
||||||
if config.URL_PREFIX != '/':
|
if config.URL_PREFIX != '/':
|
||||||
@routes.get('/')
|
@routes.get('/')
|
||||||
def index_redirect_root(request):
|
def index_redirect_root(request):
|
||||||
|
|
|
@ -267,5 +267,7 @@
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="footer text-center px-2">
|
||||||
|
<small *ngIf="versionInfo">{{ versionInfo }}</small>
|
||||||
|
</div>
|
||||||
</main><!-- /.container -->
|
</main><!-- /.container -->
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import { Component, ViewChild, ElementRef, AfterViewInit } from '@angular/core';
|
import { Component, ViewChild, ElementRef, AfterViewInit } from '@angular/core';
|
||||||
|
import { HttpClient } from '@angular/common/http';
|
||||||
import { faTrashAlt, faCheckCircle, faTimesCircle, IconDefinition } from '@fortawesome/free-regular-svg-icons';
|
import { faTrashAlt, faCheckCircle, faTimesCircle, IconDefinition } from '@fortawesome/free-regular-svg-icons';
|
||||||
import { faRedoAlt, faSun, faMoon, faCircleHalfStroke, faCheck, faExternalLinkAlt, faDownload } from '@fortawesome/free-solid-svg-icons';
|
import { faRedoAlt, faSun, faMoon, faCircleHalfStroke, faCheck, faExternalLinkAlt, faDownload } from '@fortawesome/free-solid-svg-icons';
|
||||||
import { CookieService } from 'ngx-cookie-service';
|
import { CookieService } from 'ngx-cookie-service';
|
||||||
|
@ -36,6 +37,7 @@ export class AppComponent implements AfterViewInit {
|
||||||
batchImportStatus = '';
|
batchImportStatus = '';
|
||||||
importInProgress = false;
|
importInProgress = false;
|
||||||
cancelImportFlag = false;
|
cancelImportFlag = false;
|
||||||
|
versionInfo: string | null = null;
|
||||||
|
|
||||||
@ViewChild('queueMasterCheckbox') queueMasterCheckbox: MasterCheckboxComponent;
|
@ViewChild('queueMasterCheckbox') queueMasterCheckbox: MasterCheckboxComponent;
|
||||||
@ViewChild('queueDelSelected') queueDelSelected: ElementRef;
|
@ViewChild('queueDelSelected') queueDelSelected: ElementRef;
|
||||||
|
@ -58,7 +60,7 @@ export class AppComponent implements AfterViewInit {
|
||||||
faDownload = faDownload;
|
faDownload = faDownload;
|
||||||
faExternalLinkAlt = faExternalLinkAlt;
|
faExternalLinkAlt = faExternalLinkAlt;
|
||||||
|
|
||||||
constructor(public downloads: DownloadsService, private cookieService: CookieService) {
|
constructor(public downloads: DownloadsService, private cookieService: CookieService, private http: HttpClient) {
|
||||||
this.format = cookieService.get('metube_format') || 'any';
|
this.format = cookieService.get('metube_format') || 'any';
|
||||||
// Needs to be set or qualities won't automatically be set
|
// Needs to be set or qualities won't automatically be set
|
||||||
this.setQualities()
|
this.setQualities()
|
||||||
|
@ -97,6 +99,7 @@ export class AppComponent implements AfterViewInit {
|
||||||
this.doneClearFailed.nativeElement.disabled = failed === 0;
|
this.doneClearFailed.nativeElement.disabled = failed === 0;
|
||||||
this.doneRetryFailed.nativeElement.disabled = failed === 0;
|
this.doneRetryFailed.nativeElement.disabled = failed === 0;
|
||||||
});
|
});
|
||||||
|
this.fetchVersionInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
// workaround to allow fetching of Map values in the order they were inserted
|
// workaround to allow fetching of Map values in the order they were inserted
|
||||||
|
@ -369,63 +372,76 @@ export class AppComponent implements AfterViewInit {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Export URLs based on filter: 'pending', 'completed', 'failed', or 'all'
|
// Export URLs based on filter: 'pending', 'completed', 'failed', or 'all'
|
||||||
exportBatchUrls(filter: 'pending' | 'completed' | 'failed' | 'all'): void {
|
exportBatchUrls(filter: 'pending' | 'completed' | 'failed' | 'all'): void {
|
||||||
let urls: string[];
|
let urls: string[];
|
||||||
if (filter === 'pending') {
|
if (filter === 'pending') {
|
||||||
urls = Array.from(this.downloads.queue.values()).map(dl => dl.url);
|
urls = Array.from(this.downloads.queue.values()).map(dl => dl.url);
|
||||||
} else if (filter === 'completed') {
|
} else if (filter === 'completed') {
|
||||||
// Only finished downloads in the "done" Map
|
// Only finished downloads in the "done" Map
|
||||||
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'finished').map(dl => dl.url);
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'finished').map(dl => dl.url);
|
||||||
} else if (filter === 'failed') {
|
} else if (filter === 'failed') {
|
||||||
// Only error downloads from the "done" Map
|
// Only error downloads from the "done" Map
|
||||||
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'error').map(dl => dl.url);
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'error').map(dl => dl.url);
|
||||||
} else {
|
} else {
|
||||||
// All: pending + both finished and error in done
|
// All: pending + both finished and error in done
|
||||||
urls = [
|
urls = [
|
||||||
...Array.from(this.downloads.queue.values()).map(dl => dl.url),
|
...Array.from(this.downloads.queue.values()).map(dl => dl.url),
|
||||||
...Array.from(this.downloads.done.values()).map(dl => dl.url)
|
...Array.from(this.downloads.done.values()).map(dl => dl.url)
|
||||||
];
|
];
|
||||||
|
}
|
||||||
|
if (!urls.length) {
|
||||||
|
alert('No URLs found for the selected filter.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const content = urls.join('\n');
|
||||||
|
const blob = new Blob([content], { type: 'text/plain' });
|
||||||
|
const downloadUrl = window.URL.createObjectURL(blob);
|
||||||
|
const a = document.createElement('a');
|
||||||
|
a.href = downloadUrl;
|
||||||
|
a.download = 'metube_urls.txt';
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a);
|
||||||
|
window.URL.revokeObjectURL(downloadUrl);
|
||||||
}
|
}
|
||||||
if (!urls.length) {
|
|
||||||
alert('No URLs found for the selected filter.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const content = urls.join('\n');
|
|
||||||
const blob = new Blob([content], { type: 'text/plain' });
|
|
||||||
const downloadUrl = window.URL.createObjectURL(blob);
|
|
||||||
const a = document.createElement('a');
|
|
||||||
a.href = downloadUrl;
|
|
||||||
a.download = 'metube_urls.txt';
|
|
||||||
document.body.appendChild(a);
|
|
||||||
a.click();
|
|
||||||
document.body.removeChild(a);
|
|
||||||
window.URL.revokeObjectURL(downloadUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Copy URLs to clipboard based on filter: 'pending', 'completed', 'failed', or 'all'
|
// Copy URLs to clipboard based on filter: 'pending', 'completed', 'failed', or 'all'
|
||||||
copyBatchUrls(filter: 'pending' | 'completed' | 'failed' | 'all'): void {
|
copyBatchUrls(filter: 'pending' | 'completed' | 'failed' | 'all'): void {
|
||||||
let urls: string[];
|
let urls: string[];
|
||||||
if (filter === 'pending') {
|
if (filter === 'pending') {
|
||||||
urls = Array.from(this.downloads.queue.values()).map(dl => dl.url);
|
urls = Array.from(this.downloads.queue.values()).map(dl => dl.url);
|
||||||
} else if (filter === 'completed') {
|
} else if (filter === 'completed') {
|
||||||
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'finished').map(dl => dl.url);
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'finished').map(dl => dl.url);
|
||||||
} else if (filter === 'failed') {
|
} else if (filter === 'failed') {
|
||||||
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'error').map(dl => dl.url);
|
urls = Array.from(this.downloads.done.values()).filter(dl => dl.status === 'error').map(dl => dl.url);
|
||||||
} else {
|
} else {
|
||||||
urls = [
|
urls = [
|
||||||
...Array.from(this.downloads.queue.values()).map(dl => dl.url),
|
...Array.from(this.downloads.queue.values()).map(dl => dl.url),
|
||||||
...Array.from(this.downloads.done.values()).map(dl => dl.url)
|
...Array.from(this.downloads.done.values()).map(dl => dl.url)
|
||||||
];
|
];
|
||||||
|
}
|
||||||
|
if (!urls.length) {
|
||||||
|
alert('No URLs found for the selected filter.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const content = urls.join('\n');
|
||||||
|
navigator.clipboard.writeText(content)
|
||||||
|
.then(() => alert('URLs copied to clipboard.'))
|
||||||
|
.catch(() => alert('Failed to copy URLs.'));
|
||||||
}
|
}
|
||||||
if (!urls.length) {
|
|
||||||
alert('No URLs found for the selected filter.');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const content = urls.join('\n');
|
|
||||||
navigator.clipboard.writeText(content)
|
|
||||||
.then(() => alert('URLs copied to clipboard.'))
|
|
||||||
.catch(() => alert('Failed to copy URLs.'));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
fetchVersionInfo(): void {
|
||||||
|
const baseUrl = `${window.location.origin}${window.location.pathname.replace(/\/[^\/]*$/, '/')}`;
|
||||||
|
const versionUrl = `${baseUrl}version`;
|
||||||
|
this.http.get<{ version: string}>(versionUrl)
|
||||||
|
.subscribe({
|
||||||
|
next: (data) => {
|
||||||
|
this.versionInfo = `yt-dlp version: ${data.version}`;
|
||||||
|
},
|
||||||
|
error: () => {
|
||||||
|
this.versionInfo = '';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue