feat: 支持清除重置,方便分页视频刷新 (#596)

This commit is contained in:
ᴀᴍᴛᴏᴀᴇʀ
2026-01-11 15:03:31 +08:00
committed by GitHub
parent 5944298f10
commit 26514f7174
7 changed files with 219 additions and 33 deletions

View File

@@ -33,6 +33,12 @@ pub struct ResetVideoResponse {
pub pages: Vec<PageInfo>,
}
#[derive(Serialize)]
pub struct ClearAndResetVideoStatusResponse {
pub warning: Option<String>,
pub video: VideoInfo,
}
#[derive(Serialize)]
pub struct ResetFilteredVideosResponse {
pub resetted: bool,

View File

@@ -1,12 +1,14 @@
use std::collections::HashSet;
use anyhow::Result;
use anyhow::{Context, Result};
use axum::extract::{Extension, Path, Query};
use axum::routing::{get, post};
use axum::{Json, Router};
use bili_sync_entity::*;
use sea_orm::ActiveValue::Set;
use sea_orm::{
ColumnTrait, DatabaseConnection, EntityTrait, PaginatorTrait, QueryFilter, QueryOrder, TransactionTrait,
ActiveModelTrait, ColumnTrait, DatabaseConnection, EntityTrait, IntoActiveModel, PaginatorTrait, QueryFilter,
QueryOrder, TransactionTrait, TryIntoModel,
};
use crate::api::error::InnerApiError;
@@ -16,8 +18,9 @@ use crate::api::request::{
UpdateVideoStatusRequest, VideosRequest,
};
use crate::api::response::{
PageInfo, ResetFilteredVideosResponse, ResetVideoResponse, SimplePageInfo, SimpleVideoInfo,
UpdateFilteredVideoStatusResponse, UpdateVideoStatusResponse, VideoInfo, VideoResponse, VideosResponse,
ClearAndResetVideoStatusResponse, PageInfo, ResetFilteredVideosResponse, ResetVideoResponse, SimplePageInfo,
SimpleVideoInfo, UpdateFilteredVideoStatusResponse, UpdateVideoStatusResponse, VideoInfo, VideoResponse,
VideosResponse,
};
use crate::api::wrapper::{ApiError, ApiResponse, ValidatedJson};
use crate::utils::status::{PageStatus, VideoStatus};
@@ -26,6 +29,10 @@ pub(super) fn router() -> Router {
Router::new()
.route("/videos", get(get_videos))
.route("/videos/{id}", get(get_video))
.route(
"/videos/{id}/clear-and-reset-status",
post(clear_and_reset_video_status),
)
.route("/videos/{id}/reset-status", post(reset_video_status))
.route("/videos/{id}/update-status", post(update_video_status))
.route("/videos/reset-status", post(reset_filtered_video_status))
@@ -152,6 +159,43 @@ pub async fn reset_video_status(
}))
}
pub async fn clear_and_reset_video_status(
Path(id): Path<i32>,
Extension(db): Extension<DatabaseConnection>,
) -> Result<ApiResponse<ClearAndResetVideoStatusResponse>, ApiError> {
let video_info = video::Entity::find_by_id(id).one(&db).await?;
let Some(video_info) = video_info else {
return Err(InnerApiError::NotFound(id).into());
};
let txn = db.begin().await?;
let mut video_info = video_info.into_active_model();
video_info.single_page = Set(None);
video_info.download_status = Set(0);
let video_info = video_info.update(&txn).await?;
page::Entity::delete_many()
.filter(page::Column::VideoId.eq(id))
.exec(&txn)
.await?;
txn.commit().await?;
let video_info = video_info.try_into_model()?;
let warning = tokio::fs::remove_dir_all(&video_info.path)
.await
.context(format!("删除本地路径「{}」失败", video_info.path))
.err()
.map(|e| format!("{:#}", e));
Ok(ApiResponse::ok(ClearAndResetVideoStatusResponse {
warning,
video: VideoInfo {
id: video_info.id,
bvid: video_info.bvid,
name: video_info.name,
upper_name: video_info.upper_name,
should_download: video_info.should_download,
download_status: video_info.download_status,
},
}))
}
pub async fn reset_filtered_video_status(
Extension(db): Extension<DatabaseConnection>,
Json(request): Json<ResetFilteredVideoStatusRequest>,

View File

@@ -5,6 +5,7 @@ import type {
VideosResponse,
VideoResponse,
ResetVideoResponse,
ClearAndResetVideoResponse,
ResetFilteredVideosResponse,
UpdateVideoStatusRequest,
UpdateVideoStatusResponse,
@@ -165,6 +166,10 @@ class ApiClient {
return this.post<ResetVideoResponse>(`/videos/${id}/reset-status`, request);
}
async clearAndResetVideoStatus(id: number): Promise<ApiResponse<ClearAndResetVideoResponse>> {
return this.post<ClearAndResetVideoResponse>(`/videos/${id}/clear-and-reset-status`);
}
async resetFilteredVideoStatus(
request: ResetFilteredVideoStatusRequest
): Promise<ApiResponse<ResetFilteredVideosResponse>> {
@@ -297,6 +302,7 @@ const api = {
getVideo: (id: number) => apiClient.getVideo(id),
resetVideoStatus: (id: number, request: ResetVideoStatusRequest) =>
apiClient.resetVideoStatus(id, request),
clearAndResetVideoStatus: (id: number) => apiClient.clearAndResetVideoStatus(id),
resetFilteredVideoStatus: (request: ResetFilteredVideoStatusRequest) =>
apiClient.resetFilteredVideoStatus(request),
updateVideoStatus: (id: number, request: UpdateVideoStatusRequest) =>

View File

@@ -9,6 +9,7 @@
import type { VideoInfo } from '$lib/types';
import RotateCcwIcon from '@lucide/svelte/icons/rotate-ccw';
import InfoIcon from '@lucide/svelte/icons/info';
import BrushCleaningIcon from '@lucide/svelte/icons/brush-cleaning';
import UserIcon from '@lucide/svelte/icons/user';
import SquareArrowOutUpRightIcon from '@lucide/svelte/icons/square-arrow-out-up-right';
import MoreHorizontalIcon from '@lucide/svelte/icons/more-horizontal';
@@ -24,8 +25,11 @@
export let taskNames: string[] = []; // 自定义任务名称
export let showProgress: boolean = true; // 是否显示进度信息
export let onReset: ((forceReset: boolean) => Promise<void>) | null = null; // 自定义重置函数
export let onClearAndReset: (() => Promise<void>) | null = null; // 自定义清空重置函数
export let resetDialogOpen = false; // 导出对话框状态,让父组件可以控制
export let clearAndResetDialogOpen = false; // 导出清空重置对话框状态
export let resetting = false;
export let clearAndResetting = false;
let forceReset = false;
@@ -98,6 +102,15 @@
forceReset = false;
}
async function handleClearAndReset() {
clearAndResetting = true;
if (onClearAndReset) {
await onClearAndReset();
}
clearAndResetting = false;
clearAndResetDialogOpen = false;
}
function handleViewDetail() {
goto(`/video/${video.id}`);
}
@@ -112,7 +125,7 @@
</script>
<Card class={cardClasses}>
<CardHeader class="flex-shrink-0 pb-3">
<CardHeader class="shrink-0 pb-3">
<div class="flex min-w-0 items-start justify-between gap-3">
<CardTitle
class="line-clamp-2 min-w-0 flex-1 cursor-default {mode === 'default'
@@ -196,6 +209,17 @@
{/snippet}
</DropdownMenu.Trigger>
<DropdownMenu.Content align="start" class="w-48">
<DropdownMenu.Item class="cursor-pointer" onclick={() => (resetDialogOpen = true)}>
<RotateCcwIcon class="mr-2 h-4 w-4" />
重置
</DropdownMenu.Item>
<DropdownMenu.Item
class="cursor-pointer"
onclick={() => (clearAndResetDialogOpen = true)}
>
<BrushCleaningIcon class="mr-2 h-4 w-4" />
清空重置
</DropdownMenu.Item>
<DropdownMenu.Item
class="cursor-pointer"
onclick={() =>
@@ -204,10 +228,6 @@
<SquareArrowOutUpRightIcon class="mr-2 h-4 w-4" />
在 B 站打开
</DropdownMenu.Item>
<DropdownMenu.Item class="cursor-pointer" onclick={() => (resetDialogOpen = true)}>
<RotateCcwIcon class="mr-2 h-4 w-4" />
重置下载状态
</DropdownMenu.Item>
</DropdownMenu.Content>
</DropdownMenu.Root>
</div>
@@ -261,3 +281,38 @@
</AlertDialog.Footer>
</AlertDialog.Content>
</AlertDialog.Root>
<!-- 清空重置确认对话框 -->
<AlertDialog.Root bind:open={clearAndResetDialogOpen}>
<AlertDialog.Content>
<AlertDialog.Header>
<AlertDialog.Title>清空重置视频</AlertDialog.Title>
<AlertDialog.Description>
确定要清空重置视频 <strong>"{displayTitle}"</strong> 吗?
<br />
<br />
此操作会:
<ul class="mt-2 ml-4 list-disc space-y-1">
<li>将视频状态重置为未开始</li>
<li>删除所有分页信息</li>
<li class="text-destructive font-medium">删除视频对应的文件夹</li>
</ul>
<br />
该功能可在多页视频变更后手动触发全量更新,执行后<span class="text-destructive font-medium"
>无法撤销</span
>
</AlertDialog.Description>
</AlertDialog.Header>
<AlertDialog.Footer>
<AlertDialog.Cancel>取消</AlertDialog.Cancel>
<AlertDialog.Action
onclick={handleClearAndReset}
disabled={clearAndResetting}
class="bg-destructive hover:bg-destructive/90"
>
{clearAndResetting ? '清空重置中...' : '确认清空重置'}
</AlertDialog.Action>
</AlertDialog.Footer>
</AlertDialog.Content>
</AlertDialog.Root>

View File

@@ -57,6 +57,11 @@ export interface ResetVideoResponse {
pages: PageInfo[];
}
export interface ClearAndResetVideoResponse {
warning?: string;
video: VideoInfo;
}
export interface ResetFilteredVideosResponse {
resetted: boolean;
resetted_videos_count: number;

View File

@@ -8,6 +8,7 @@
import type { ApiError, VideoResponse, UpdateVideoStatusRequest } from '$lib/types';
import RotateCcwIcon from '@lucide/svelte/icons/rotate-ccw';
import EditIcon from '@lucide/svelte/icons/edit';
import BrushCleaningIcon from '@lucide/svelte/icons/brush-cleaning';
import { setBreadcrumb } from '$lib/stores/breadcrumb';
import { appStateStore, ToQuery } from '$lib/stores/filter';
import VideoCard from '$lib/components/video-card.svelte';
@@ -19,6 +20,8 @@
let error: string | null = null;
let resetDialogOpen = false;
let resetting = false;
let clearAndResetDialogOpen = false;
let clearAndResetting = false;
let statusEditorOpen = false;
let statusEditorLoading = false;
@@ -87,6 +90,56 @@
statusEditorLoading = false;
}
}
async function handleReset(forceReset: boolean) {
if (!videoData) return;
try {
const result = await api.resetVideoStatus(videoData.video.id, { force: forceReset });
const data = result.data;
if (data.resetted) {
videoData = {
video: data.video,
pages: data.pages
};
toast.success('重置成功');
} else {
toast.info('重置无效', {
description: `视频「${data.video.name}」没有失败的状态,无需重置`
});
}
} catch (error) {
console.error('重置失败:', error);
toast.error('重置失败', {
description: (error as ApiError).message
});
}
}
async function handleClearAndReset() {
if (!videoData) return;
try {
const result = await api.clearAndResetVideoStatus(videoData.video.id);
const data = result.data;
videoData = {
video: data.video,
pages: []
};
if (data.warning) {
toast.warning('清空重置成功', {
description: data.warning
});
} else {
toast.success('清空重置成功', {
description: `视频「${data.video.name}」已清空重置`
});
}
} catch (error) {
console.error('清空重置失败:', error);
toast.error('清空重置失败', {
description: (error as ApiError).message
});
}
}
</script>
<svelte:head>
@@ -130,11 +183,21 @@
variant="outline"
class="shrink-0 cursor-pointer "
onclick={() => (resetDialogOpen = true)}
disabled={resetting}
disabled={resetting || clearAndResetting}
>
<RotateCcwIcon class="mr-2 h-4 w-4 {resetting ? 'animate-spin' : ''}" />
重置
</Button>
<Button
size="sm"
variant="outline"
class="shrink-0 cursor-pointer "
onclick={() => (clearAndResetDialogOpen = true)}
disabled={resetting || clearAndResetting}
>
<BrushCleaningIcon class="mr-2 h-4 w-4 {clearAndResetting ? 'animate-spin' : ''}" />
清空重置
</Button>
<Button
size="sm"
variant="outline"
@@ -164,28 +227,10 @@
taskNames={['视频封面', '视频信息', 'UP 主头像', 'UP 主信息', '分页下载']}
bind:resetDialogOpen
bind:resetting
onReset={async (forceReset: boolean) => {
try {
const result = await api.resetVideoStatus(videoData!.video.id, { force: forceReset });
const data = result.data;
if (data.resetted) {
videoData = {
video: data.video,
pages: data.pages
};
toast.success('重置成功');
} else {
toast.info('重置无效', {
description: `视频「${data.video.name}」没有失败的状态,无需重置`
});
}
} catch (error) {
console.error('重置失败:', error);
toast.error('重置失败', {
description: (error as ApiError).message
});
}
}}
bind:clearAndResetDialogOpen
bind:clearAndResetting
onReset={handleReset}
onClearAndReset={handleClearAndReset}
/>
</div>
</section>
@@ -226,7 +271,6 @@
<div class="py-12 text-center">
<div class="space-y-2">
<p class="text-muted-foreground">暂无分 P 数据</p>
<p class="text-muted-foreground text-sm">该视频可能为单 P 视频</p>
</div>
</div>
{/if}

View File

@@ -131,6 +131,29 @@
}
}
async function handleClearAndResetVideo(id: number) {
try {
const result = await api.clearAndResetVideoStatus(id);
const data = result.data;
if (data.warning) {
toast.warning('清空重置成功', {
description: data.warning
});
} else {
toast.success('清空重置成功', {
description: `视频「${data.video.name}」已清空重置`
});
}
const { query, currentPage, videoSource } = $appStateStore;
await loadVideos(query, currentPage, videoSource);
} catch (error) {
console.error('清空重置失败:', error);
toast.error('清空重置失败', {
description: (error as ApiError).message
});
}
}
async function handleResetAllVideos() {
resettingAll = true;
try {
@@ -332,6 +355,9 @@
onReset={async (forceReset: boolean) => {
await handleResetVideo(video.id, forceReset);
}}
onClearAndReset={async () => {
await handleClearAndResetVideo(video.id);
}}
/>
{/each}
</div>