403 lines
19 KiB
HTML
403 lines
19 KiB
HTML
<nav class="navbar navbar-expand-md navbar-dark">
|
|
<div class="container-fluid">
|
|
<a class="navbar-brand d-flex align-items-center" href="#">
|
|
<img src="assets/icons/android-chrome-192x192.png" alt="MeTube Logo" height="32" class="me-2">
|
|
MeTube
|
|
</a>
|
|
<div class="download-metrics">
|
|
<div class="metric" *ngIf="activeDownloads > 0">
|
|
<fa-icon [icon]="faDownload" class="text-primary"></fa-icon>
|
|
<span>{{activeDownloads}} downloading</span>
|
|
</div>
|
|
<div class="metric" *ngIf="queuedDownloads > 0">
|
|
<fa-icon [icon]="faClock" class="text-warning"></fa-icon>
|
|
<span>{{queuedDownloads}} queued</span>
|
|
</div>
|
|
<div class="metric" *ngIf="completedDownloads > 0">
|
|
<fa-icon [icon]="faCheck" class="text-success"></fa-icon>
|
|
<span>{{completedDownloads}} completed</span>
|
|
</div>
|
|
<div class="metric" *ngIf="failedDownloads > 0">
|
|
<fa-icon [icon]="faTimesCircle" class="text-danger"></fa-icon>
|
|
<span>{{failedDownloads}} failed</span>
|
|
</div>
|
|
<div class="metric" *ngIf="(totalSpeed | speed) !== ''">
|
|
<fa-icon [icon]="faTachometerAlt" class="text-info"></fa-icon>
|
|
<span>{{totalSpeed | speed }}</span>
|
|
</div>
|
|
</div>
|
|
<!--
|
|
<button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarsDefault" aria-controls="navbarsDefault" aria-expanded="false" aria-label="Toggle navigation">
|
|
<span class="navbar-toggler-icon"></span>
|
|
</button>
|
|
<div class="collapse navbar-collapse" id="navbarsDefault">
|
|
<ul class="navbar-nav mr-auto">
|
|
<li class="nav-item active">
|
|
<a class="nav-link" href="#">Home <span class="sr-only">(current)</span></a>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
-->
|
|
<div class="navbar-nav ms-auto">
|
|
<div class="nav-item dropdown">
|
|
<button class="btn btn-link nav-link py-2 px-0 px-sm-2 dropdown-toggle d-flex align-items-center"
|
|
id="theme-select"
|
|
type="button"
|
|
aria-expanded="false"
|
|
data-bs-toggle="dropdown"
|
|
data-bs-display="static">
|
|
<fa-icon [icon]="activeTheme.icon"></fa-icon>
|
|
</button>
|
|
<ul class="dropdown-menu dropdown-menu-end position-absolute" aria-labelledby="theme-select">
|
|
<li *ngFor="let theme of themes">
|
|
<button type="button" class="dropdown-item d-flex align-items-center" [ngClass]="{'active' : activeTheme == theme}" (click)="themeChanged(theme)">
|
|
<span class="me-2 opacity-50">
|
|
<fa-icon [icon]="theme.icon"></fa-icon>
|
|
</span>
|
|
{{ theme.displayName }}
|
|
<span class="ms-auto" [ngClass]="{'d-none' : activeTheme != theme}">
|
|
<fa-icon [icon]="faCheck"></fa-icon>
|
|
</span>
|
|
</button>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</nav>
|
|
|
|
<main role="main" class="container container-xl">
|
|
<form #f="ngForm">
|
|
<div class="container add-url-box">
|
|
<!-- Main URL Input with Download Button -->
|
|
<div class="row mb-4">
|
|
<div class="col">
|
|
<div class="input-group input-group-lg shadow-sm">
|
|
<input type="text"
|
|
autocomplete="off"
|
|
spellcheck="false"
|
|
class="form-control form-control-lg"
|
|
placeholder="Enter video or playlist URL"
|
|
name="addUrl"
|
|
[(ngModel)]="addUrl"
|
|
[disabled]="addInProgress || downloads.loading">
|
|
<button class="btn btn-primary btn-lg px-4"
|
|
type="submit"
|
|
(click)="addDownload()"
|
|
[disabled]="addInProgress || downloads.loading">
|
|
<span class="spinner-border spinner-border-sm" role="status" id="add-spinner" *ngIf="addInProgress"></span>
|
|
{{ addInProgress ? "Adding..." : "Download" }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Options Row -->
|
|
<div class="row mb-3 g-3">
|
|
<div class="col-md-4">
|
|
<div class="input-group">
|
|
<span class="input-group-text">Quality</span>
|
|
<select class="form-select"
|
|
name="quality"
|
|
[(ngModel)]="quality"
|
|
(change)="qualityChanged()"
|
|
[disabled]="addInProgress || downloads.loading">
|
|
<option *ngFor="let q of qualities" [ngValue]="q.id">{{ q.text }}</option>
|
|
</select>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-4">
|
|
<div class="input-group">
|
|
<span class="input-group-text">Format</span>
|
|
<select class="form-select"
|
|
name="format"
|
|
[(ngModel)]="format"
|
|
(change)="formatChanged()"
|
|
[disabled]="addInProgress || downloads.loading">
|
|
<option *ngFor="let f of formats" [ngValue]="f.id">{{ f.text }}</option>
|
|
</select>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-4">
|
|
<button type="button"
|
|
class="btn btn-outline-secondary w-100 h-100"
|
|
(click)="toggleAdvanced()">
|
|
Advanced Options
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Advanced Options Panel -->
|
|
<div class="row">
|
|
<div class="col-12">
|
|
<div class="collapse show" id="advancedOptions" [ngbCollapse]="!isAdvancedOpen">
|
|
<div class="card card-body">
|
|
<!-- Advanced Settings -->
|
|
<div class="row g-3 mb-2">
|
|
<div class="col-md-6">
|
|
<div class="input-group">
|
|
<span class="input-group-text">Auto Start</span>
|
|
<select class="form-select"
|
|
name="autoStart"
|
|
[(ngModel)]="autoStart"
|
|
(change)="autoStartChanged()"
|
|
[disabled]="addInProgress || downloads.loading"
|
|
ngbTooltip="Automatically start downloads when added">
|
|
<option [ngValue]="true">Yes</option>
|
|
<option [ngValue]="false">No</option>
|
|
</select>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-6">
|
|
<div class="input-group">
|
|
<span class="input-group-text">Download Folder</span>
|
|
<ng-select [items]="customDirs$ | async"
|
|
placeholder="Default"
|
|
[addTag]="allowCustomDir.bind(this)"
|
|
addTagText="Create directory"
|
|
bindLabel="folder"
|
|
[(ngModel)]="folder"
|
|
[disabled]="addInProgress || downloads.loading"
|
|
[virtualScroll]="true"
|
|
[clearable]="true"
|
|
[loading]="downloads.loading"
|
|
[searchable]="true"
|
|
[closeOnSelect]="true"
|
|
ngbTooltip="Choose where to save downloads. Type to create a new folder.">
|
|
</ng-select>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-6">
|
|
<div class="input-group">
|
|
<span class="input-group-text">Custom Name Prefix</span>
|
|
<input type="text"
|
|
class="form-control"
|
|
placeholder="Default"
|
|
name="customNamePrefix"
|
|
[(ngModel)]="customNamePrefix"
|
|
[disabled]="addInProgress || downloads.loading"
|
|
ngbTooltip="Add a prefix to downloaded filenames">
|
|
</div>
|
|
</div>
|
|
<div class="col-md-6">
|
|
<div class="input-group">
|
|
<span class="input-group-text">Items Limit</span>
|
|
<input type="number"
|
|
min="0"
|
|
class="form-control"
|
|
placeholder="Default"
|
|
name="playlistItemLimit"
|
|
(keydown)="isNumber($event)"
|
|
[(ngModel)]="playlistItemLimit"
|
|
[disabled]="addInProgress || downloads.loading"
|
|
ngbTooltip="Maximum number of items to download from a playlist (0 = no limit)">
|
|
</div>
|
|
</div>
|
|
<div class="col-12">
|
|
<div class="form-check form-switch">
|
|
<input class="form-check-input"
|
|
type="checkbox"
|
|
role="switch"
|
|
name="playlistStrictMode"
|
|
[(ngModel)]="playlistStrictMode"
|
|
[disabled]="addInProgress || downloads.loading"
|
|
ngbTooltip="Only download playlists when URL explicitly points to a playlist">
|
|
<label class="form-check-label">Strict Playlist Mode</label>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Advanced Actions -->
|
|
<div class="row">
|
|
<div class="col-12">
|
|
<hr class="my-3">
|
|
<div class="row g-2">
|
|
<div class="col-md-4">
|
|
<button type="button"
|
|
class="btn btn-secondary w-100"
|
|
(click)="openBatchImportModal()">
|
|
<fa-icon [icon]="faFileImport" class="me-2"></fa-icon>
|
|
Import URLs
|
|
</button>
|
|
</div>
|
|
<div class="col-md-4">
|
|
<button type="button"
|
|
class="btn btn-secondary w-100"
|
|
(click)="exportBatchUrls('all')">
|
|
<fa-icon [icon]="faFileExport" class="me-2"></fa-icon>
|
|
Export URLs
|
|
</button>
|
|
</div>
|
|
<div class="col-md-4">
|
|
<button type="button"
|
|
class="btn btn-secondary w-100"
|
|
(click)="copyBatchUrls('all')">
|
|
<fa-icon [icon]="faCopy" class="me-2"></fa-icon>
|
|
Copy URLs
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</form>
|
|
|
|
<!-- Batch Import Modal -->
|
|
<div class="modal fade" tabindex="-1" role="dialog" [ngClass]="{'show': batchImportModalOpen}" [ngStyle]="{'display': batchImportModalOpen ? 'block' : 'none'}">
|
|
<div class="modal-dialog" role="document">
|
|
<div class="modal-content">
|
|
<div class="modal-header">
|
|
<h5 class="modal-title">Batch Import URLs</h5>
|
|
<button type="button" class="btn-close" aria-label="Close" (click)="closeBatchImportModal()"></button>
|
|
</div>
|
|
<div class="modal-body">
|
|
<textarea [(ngModel)]="batchImportText" class="form-control" rows="6"
|
|
placeholder="Paste one video URL per line"></textarea>
|
|
<div class="mt-2">
|
|
<small *ngIf="batchImportStatus">{{ batchImportStatus }}</small>
|
|
</div>
|
|
</div>
|
|
<div class="modal-footer">
|
|
<button type="button" class="btn btn-danger me-auto" *ngIf="importInProgress" (click)="cancelBatchImport()">
|
|
Cancel Import
|
|
</button>
|
|
<button type="button" class="btn btn-secondary" (click)="closeBatchImportModal()">Close</button>
|
|
<button type="button" class="btn btn-primary" (click)="startBatchImport()" [disabled]="importInProgress">
|
|
Import URLs
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
|
|
<div *ngIf="downloads.loading" class="alert alert-info" role="alert">
|
|
Connecting to server...
|
|
</div>
|
|
<div class="metube-section-header">Downloading</div>
|
|
<div class="px-2 py-3 border-bottom">
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #queueDelSelected (click)="delSelectedDownloads('queue')"><fa-icon [icon]="faTrashAlt"></fa-icon> Cancel selected</button>
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #queueDownloadSelected (click)="startSelectedDownloads('queue')"><fa-icon [icon]="faDownload"></fa-icon> Download selected</button>
|
|
</div>
|
|
<div class="overflow-auto">
|
|
<table class="table">
|
|
<thead>
|
|
<tr>
|
|
<th scope="col" style="width: 1rem;">
|
|
<app-master-checkbox #queueMasterCheckbox [id]="'queue'" [list]="downloads.queue" (changed)="queueSelectionChanged($event)"></app-master-checkbox>
|
|
</th>
|
|
<th scope="col">Video</th>
|
|
<th scope="col" style="width: 8rem;">Speed</th>
|
|
<th scope="col" style="width: 7rem;">ETA</th>
|
|
<th scope="col" style="width: 6rem;"></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr *ngFor="let download of downloads.queue | keyvalue: asIsOrder; trackBy: identifyDownloadRow" [class.disabled]='download.value.deleting'>
|
|
<td>
|
|
<app-slave-checkbox [id]="download.key" [master]="queueMasterCheckbox" [checkable]="download.value"></app-slave-checkbox>
|
|
</td>
|
|
<td title="{{ download.value.filename }}">
|
|
<div class="d-flex flex-column flex-sm-row align-items-center row-gap-2 column-gap-3">
|
|
<div>{{ download.value.title }}</div>
|
|
<ngb-progressbar height="1.5rem" [showValue]="download.value.status != 'preparing'" [striped]="download.value.status == 'preparing'" [animated]="download.value.status == 'preparing'" type="success" [value]="download.value.status == 'preparing' ? 100 : download.value.percent | number:'1.0-0'" class="download-progressbar"></ngb-progressbar>
|
|
</div>
|
|
</td>
|
|
<td>{{ download.value.speed | speed }}</td>
|
|
<td>{{ download.value.eta | eta }}</td>
|
|
<td>
|
|
<div class="d-flex">
|
|
<button *ngIf="download.value.status === 'pending'" type="button" class="btn btn-link" (click)="downloadItemByKey(download.key)"><fa-icon [icon]="faDownload"></fa-icon></button>
|
|
<button type="button" class="btn btn-link" (click)="delDownload('queue', download.key)"><fa-icon [icon]="faTrashAlt"></fa-icon></button>
|
|
<a href="{{download.value.url}}" target="_blank" class="btn btn-link"><fa-icon [icon]="faExternalLinkAlt"></fa-icon></a>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<div class="metube-section-header">Completed</div>
|
|
<div class="px-2 py-3 border-bottom">
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #doneDelSelected (click)="delSelectedDownloads('done')"><fa-icon [icon]="faTrashAlt"></fa-icon> Clear selected</button>
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #doneClearCompleted (click)="clearCompletedDownloads()"><fa-icon [icon]="faCheckCircle"></fa-icon> Clear completed</button>
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #doneClearFailed (click)="clearFailedDownloads()"><fa-icon [icon]="faTimesCircle"></fa-icon> Clear failed</button>
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #doneRetryFailed (click)="retryFailedDownloads()"><fa-icon [icon]="faRedoAlt"></fa-icon> Retry failed</button>
|
|
<button type="button" class="btn btn-link text-decoration-none px-0 me-4" disabled #doneDownloadSelected (click)="downloadSelectedFiles()"><fa-icon [icon]="faDownload"></fa-icon> Download Selected</button>
|
|
</div>
|
|
<div class="overflow-auto">
|
|
<table class="table">
|
|
<thead>
|
|
<tr>
|
|
<th scope="col" style="width: 1rem;">
|
|
<app-master-checkbox #doneMasterCheckbox [id]="'done'" [list]="downloads.done" (changed)="doneSelectionChanged($event)"></app-master-checkbox>
|
|
</th>
|
|
<th scope="col">Video</th>
|
|
<th scope="col">File Size</th>
|
|
<th scope="col" style="width: 8rem;"></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr *ngFor="let download of downloads.done | keyvalue: asIsOrder; trackBy: identifyDownloadRow" [class.disabled]='download.value.deleting'>
|
|
<td>
|
|
<app-slave-checkbox [id]="download.key" [master]="doneMasterCheckbox" [checkable]="download.value"></app-slave-checkbox>
|
|
</td>
|
|
<td>
|
|
<div style="display: inline-block; width: 1.5rem;">
|
|
<fa-icon *ngIf="download.value.status == 'finished'" [icon]="faCheckCircle" class="text-success"></fa-icon>
|
|
<fa-icon *ngIf="download.value.status == 'error'" [icon]="faTimesCircle" class="text-danger"></fa-icon>
|
|
</div>
|
|
<span ngbTooltip="{{download.value.msg}} | {{download.value.error}}"><a *ngIf="!!download.value.filename; else noDownloadLink" href="{{buildDownloadLink(download.value)}}" target="_blank">{{ download.value.title }}</a></span>
|
|
<ng-template #noDownloadLink>
|
|
{{download.value.title}}
|
|
<span *ngIf="download.value.msg"><br>{{download.value.msg}}</span>
|
|
<span *ngIf="download.value.error"><br>Error: {{download.value.error}}</span>
|
|
</ng-template>
|
|
</td>
|
|
<td>
|
|
<span *ngIf="download.value.size">{{ download.value.size | fileSize }}</span>
|
|
</td>
|
|
<td>
|
|
<div class="d-flex">
|
|
<button *ngIf="download.value.status == 'error'" type="button" class="btn btn-link" (click)="retryDownload(download.key, download.value)"><fa-icon [icon]="faRedoAlt"></fa-icon></button>
|
|
<a *ngIf="download.value.filename" href="{{buildDownloadLink(download.value)}}" download class="btn btn-link"><fa-icon [icon]="faDownload"></fa-icon></a>
|
|
<a href="{{download.value.url}}" target="_blank" class="btn btn-link"><fa-icon [icon]="faExternalLinkAlt"></fa-icon></a>
|
|
<button type="button" class="btn btn-link" (click)="delDownload('done', download.key)"><fa-icon [icon]="faTrashAlt"></fa-icon></button>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</main><!-- /.container -->
|
|
|
|
<footer class="footer navbar-dark bg-dark py-3 mt-5">
|
|
<div class="container text-center">
|
|
<div class="footer-content" *ngIf="ytDlpVersion && metubeVersion">
|
|
<div class="version-item">
|
|
<span class="version-label">yt-dlp</span>
|
|
<span class="version-value">{{ytDlpVersion}}</span>
|
|
</div>
|
|
<div class="version-separator"></div>
|
|
<div class="version-item">
|
|
<span class="version-label">MeTube</span>
|
|
<span class="version-value">{{metubeVersion}}</span>
|
|
</div>
|
|
<div class="version-separator"></div>
|
|
<div class="version-item" *ngIf="ytDlpOptionsUpdateTime">
|
|
<span class="version-label">yt-dlp-options</span>
|
|
<span class="version-value">{{ytDlpOptionsUpdateTime}}</span>
|
|
</div>
|
|
<div class="version-separator" *ngIf="ytDlpOptionsUpdateTime"></div>
|
|
<a href="https://github.com/alexta69/metube" target="_blank" class="github-link">
|
|
<fa-icon [icon]="faGithub"></fa-icon>
|
|
<span>GitHub</span>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</footer>
|