Files
yottob/templates/watch.html
Ryan Chen be76f0a610 Add comprehensive deletion functionality and scheduled cleanup
Features:
- Delete entire channels with all videos and downloaded files
- Delete individual video files while keeping database entries
- Scheduled automatic cleanup of videos older than 7 days
- Proper cascading deletes with file cleanup

Channel Deletion:
- New DELETE endpoint at /api/channels/<id>
- Removes channel, all video entries, and downloaded files
- User ownership verification
- Returns count of deleted files
- UI button on channels page with detailed confirmation dialog

Video File Deletion:
- New DELETE endpoint at /api/videos/<id>/file
- Celery async task to remove file from disk
- Resets download status to pending (allows re-download)
- UI button on watch page for completed videos
- Confirmation dialog with clear warnings

Scheduled Cleanup:
- Celery beat configuration for periodic tasks
- cleanup_old_videos task runs daily at midnight
- Automatically deletes videos completed more than 7 days ago
- Removes files and resets database status
- scheduled_tasks.py for beat schedule configuration
- verify_schedule.py helper to check task scheduling

UI Improvements:
- Added .btn-danger CSS class (black/white theme)
- Delete buttons with loading states
- Detailed confirmation dialogs warning about permanent deletion
- Dashboard now filters to show only completed videos

Bug Fixes:
- Fixed navbar alignment issues
- Added proper error handling for file deletion

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-26 20:55:43 -05:00

132 lines
4.9 KiB
HTML

{% extends "base.html" %}
{% block title %}{{ video.title }} - YottoB{% endblock %}
{% block content %}
<div class="watch-page">
<div class="video-player-container">
{% if video.download_status.value == 'completed' and video.download_path %}
<video controls class="video-player">
<source src="/api/video/stream/{{ video.id }}" type="video/mp4">
Your browser does not support the video tag.
</video>
{% elif video.download_status.value == 'downloading' %}
<div class="video-placeholder">
<div class="placeholder-content">
<h3>Video is downloading...</h3>
<p>Please check back in a few minutes</p>
<button class="btn btn-secondary" onclick="location.reload()">Refresh</button>
</div>
</div>
{% elif video.download_status.value == 'failed' %}
<div class="video-placeholder error">
<div class="placeholder-content">
<h3>Download failed</h3>
<p>There was an error downloading this video</p>
<button class="btn btn-primary" onclick="retryDownload()">Retry Download</button>
</div>
</div>
{% else %}
<div class="video-placeholder">
<div class="placeholder-content">
{% if video.thumbnail_url %}
<img src="{{ video.thumbnail_url }}" alt="{{ video.title }}" class="placeholder-thumbnail">
{% endif %}
<h3>Video not downloaded yet</h3>
<button class="btn btn-primary" onclick="startDownload()">Download Video</button>
</div>
</div>
{% endif %}
</div>
<div class="video-details">
<h1 class="video-title">{{ video.title }}</h1>
<div class="video-metadata">
<div class="channel-info">
<h3>{{ video.channel.title }}</h3>
<a href="{{ video.channel.rss_url }}" target="_blank" class="channel-link">View Channel</a>
</div>
<div class="video-stats">
<span class="publish-date">Published: {{ video.published_at.strftime('%B %d, %Y') }}</span>
{% if video.download_status.value == 'completed' and video.download_completed_at %}
<span class="download-date">Downloaded: {{ video.download_completed_at.strftime('%B %d, %Y') }}</span>
{% endif %}
</div>
</div>
{% if video.description %}
<div class="video-description">
<h4>Description</h4>
<p>{{ video.description }}</p>
</div>
{% endif %}
<div class="video-links">
<a href="{{ video.video_url }}" target="_blank" class="btn btn-link">Watch on YouTube</a>
{% if video.download_status.value == 'completed' and video.download_path %}
<a href="/api/video/stream/{{ video.id }}?download=1" class="btn btn-secondary" download>Download MP4</a>
<button class="btn btn-danger" onclick="deleteFromDisk()">Delete from Disk</button>
{% endif %}
</div>
</div>
</div>
<div class="back-link">
<a href="/">&larr; Back to all videos</a>
</div>
{% endblock %}
{% block scripts %}
<script>
function startDownload() {
if (!confirm('Start downloading this video?')) return;
fetch('/api/download/{{ video.id }}', {
method: 'POST'
})
.then(response => response.json())
.then(data => {
if (data.status === 'success') {
alert('Download started! This page will refresh automatically.');
setTimeout(() => location.reload(), 2000);
} else {
alert('Failed to start download: ' + data.message);
}
})
.catch(error => {
alert('Error: ' + error);
});
}
function retryDownload() {
startDownload();
}
function deleteFromDisk() {
if (!confirm('Are you sure you want to delete this video from disk? This cannot be undone.')) return;
fetch('/api/videos/{{ video.id }}/file', {
method: 'DELETE'
})
.then(response => response.json())
.then(data => {
if (data.status === 'success') {
alert('Video deleted from disk! This page will refresh automatically.');
setTimeout(() => location.reload(), 2000);
} else {
alert('Failed to delete video: ' + data.message);
}
})
.catch(error => {
alert('Error: ' + error);
});
}
// Auto-refresh if video is downloading
{% if video.download_status.value == 'downloading' %}
setTimeout(() => location.reload(), 10000); // Refresh every 10 seconds
{% endif %}
</script>
{% endblock %}