Add batch selection support for subscriptions and history

This commit is contained in:
jxxghp
2026-06-28 13:47:13 +08:00
parent ed28832484
commit e3fa0b9dae
8 changed files with 504 additions and 112 deletions

View File

@@ -31,6 +31,7 @@ export interface DynamicButtonMenuItem {
icon?: string
color?: string
permission?: UserPermissionKey
disabled?: boolean
action: () => void
}

View File

@@ -187,6 +187,7 @@ const legacyDynamicMenuTitleKeyMap: Record<string, string> = {
'components.pluginMarketSetting.title': 'dialog.pluginMarketSetting.title',
}
// 解析动态按钮菜单项标题,兼容旧版直接传入 i18n key 的写法。
function resolveDynamicMenuItemTitle(item: DynamicButtonMenuItem) {
if (item.titleKey) {
return t(item.titleKey, item.titleParams as any)
@@ -202,14 +203,16 @@ function resolveDynamicMenuItemTitle(item: DynamicButtonMenuItem) {
return looksLikeI18nKey ? t(normalizedTitleKey, item.titleParams as any) : item.title
}
// 处理页面注册的动态按钮主操作点击。
function handleDynamicButtonClick() {
if (!dynamicButton.value || !hasItemPermission(dynamicButton.value, userPermissions.value)) return
dynamicButton.value.action()
}
// 处理页面注册的动态按钮菜单项点击。
function handleDynamicMenuItemClick(item: DynamicButtonMenuItem) {
if (!hasItemPermission(item, userPermissions.value)) return
if (item.disabled || !hasItemPermission(item, userPermissions.value)) return
item.action()
}
@@ -292,6 +295,7 @@ function handleDynamicMenuItemClick(item: DynamicButtonMenuItem) {
v-for="(item, index) in visibleDynamicButtonMenuItems"
:key="item.titleKey || item.title || index"
:base-color="item.color"
:disabled="item.disabled"
@click="handleDynamicMenuItemClick(item)"
>
<template #prepend>

View File

@@ -1105,6 +1105,9 @@ export default {
},
selectedCount: 'Selected {count}/{total} items',
noSelectedItems: 'Please select subscriptions to operate',
batchSelectAll: 'Select All Subscriptions',
batchDeselectAll: 'Deselect All Subscriptions',
exitBatchMode: 'Exit Batch Mode',
batchEnable: 'Batch Enable',
batchPause: 'Batch Pause',
batchDelete: 'Batch Delete',
@@ -3290,6 +3293,7 @@ export default {
loading: 'Loading...',
pageSize: 'Items Per Page',
pageInfo: '{begin} - {end} / {total}',
selectedCount: 'Selected {count}/{total} items',
aiRedoDisabled: 'Please enable the AI assistant in system settings first',
aiRedoQueued: 'Assistant organize task submitted: {title}',
aiRedoFailed: 'Failed to submit assistant organize task',
@@ -3297,6 +3301,11 @@ export default {
aiRedo: 'Assistant Organize',
aiRedoPending: 'Assistant Organizing...',
batchAiRedo: 'Assistant Batch Organize',
batchSelect: 'Batch Select',
exitBatchSelect: 'Exit Batch Select',
exitBatchMode: 'Exit Batch Mode',
selectAll: 'Select All',
deselectAll: 'Deselect All',
redo: 'Reorganize',
delete: 'Delete',
batchRedo: 'Batch Reorganize',

View File

@@ -1100,6 +1100,9 @@ export default {
},
selectedCount: '已选择 {count}/{total} 项',
noSelectedItems: '请先选择要操作的订阅',
batchSelectAll: '选择全部订阅',
batchDeselectAll: '取消全选订阅',
exitBatchMode: '退出批量操作',
batchEnable: '批量启用',
batchPause: '批量暂停',
batchDelete: '批量删除',
@@ -3233,6 +3236,7 @@ export default {
loading: '加载中...',
pageSize: '每页条数',
pageInfo: '{begin} - {end} / {total}',
selectedCount: '已选择 {count}/{total} 项',
aiRedoDisabled: '请先在系统设置中启用 AI 智能助手',
aiRedoQueued: '已提交智能助手整理任务:{title}',
aiRedoFailed: '提交智能助手整理任务失败',
@@ -3240,6 +3244,11 @@ export default {
aiRedo: '智能助手整理',
aiRedoPending: '智能助手整理中...',
batchAiRedo: '智能助手批量整理',
batchSelect: '批量选择',
exitBatchSelect: '退出批量选择',
exitBatchMode: '退出批量操作',
selectAll: '全部选中',
deselectAll: '取消全选',
redo: '重新整理',
delete: '删除',
batchRedo: '批量重新整理',

View File

@@ -1100,6 +1100,9 @@ export default {
},
selectedCount: '已選擇 {count}/{total} 項',
noSelectedItems: '請先選擇要操作的訂閱',
batchSelectAll: '選擇全部訂閱',
batchDeselectAll: '取消全選訂閱',
exitBatchMode: '退出批量操作',
batchEnable: '批量啟用',
batchPause: '批量暫停',
batchDelete: '批量刪除',
@@ -3232,6 +3235,7 @@ export default {
loading: '加載中...',
pageSize: '每頁條數',
pageInfo: '{begin} - {end} / {total}',
selectedCount: '已選擇 {count}/{total} 項',
aiRedoDisabled: '請先在系統設置中啟用 AI 智能助手',
aiRedoQueued: '已提交智能助手整理任務:{title}',
aiRedoFailed: '提交智能助手整理任務失敗',
@@ -3239,6 +3243,11 @@ export default {
aiRedo: '智能助手整理',
aiRedoPending: '智能助手整理中...',
batchAiRedo: '智能助手批量整理',
batchSelect: '批量選擇',
exitBatchSelect: '退出批量選擇',
exitBatchMode: '退出批量操作',
selectAll: '全部選中',
deselectAll: '取消全選',
redo: '重新整理',
delete: '刪除',
batchRedo: '批量重新整理',

View File

@@ -3,7 +3,7 @@ import { debounce } from 'lodash-es'
import SubscribeListView from '@/views/subscribe/SubscribeListView.vue'
import { useI18n } from 'vue-i18n'
import { useDynamicHeaderTab } from '@/composables/useDynamicHeaderTab'
import { useDynamicButton } from '@/composables/useDynamicButton'
import { useDynamicButton, type DynamicButtonMenuItem } from '@/composables/useDynamicButton'
import { usePWA } from '@/composables/usePWA'
import { useUserStore } from '@/stores'
import { openSharedDialog } from '@/composables/useSharedDialog'
@@ -31,6 +31,21 @@ const subId = ref(route.query.id as string)
const activeTab = ref((route.query.tab as string) || '')
const subscribeListViewRef = ref<InstanceType<typeof SubscribeListView> | null>(null)
// 订阅批量模式状态快照,来源于订阅列表组件。
interface SubscribeBatchState {
enabled: boolean
selectedCount: number
totalCount: number
allSelected: boolean
}
const subscribeBatchState = ref<SubscribeBatchState>({
enabled: false,
selectedCount: 0,
totalCount: 0,
allSelected: false,
})
// 获取标签页
const subscribeTabs = computed(() => {
if (subType === '电影') {
@@ -221,6 +236,52 @@ function openShareStatisticsDialog() {
openSharedDialog(SubscribeShareStatisticsDialog, {}, {}, { closeOn: ['close'] })
}
// 订阅列表批量状态变化响应,用于驱动移动端 Footer 和桌面 FAB 操作按钮。
function handleSubscribeBatchStateChange(state: SubscribeBatchState) {
subscribeBatchState.value = state
}
// 重置父页面保存的订阅批量操作状态。
function resetSubscribeBatchState() {
subscribeBatchState.value = {
enabled: false,
selectedCount: 0,
totalCount: 0,
allSelected: false,
}
}
// 进入订阅批量操作模式。
function enterSubscribeBatchMode() {
subscribeListViewRef.value?.enterBatchMode()
}
// 退出订阅批量操作模式。
function exitSubscribeBatchMode() {
resetSubscribeBatchState()
subscribeListViewRef.value?.exitBatchMode()
}
// 切换当前订阅列表全选状态。
function toggleSubscribeBatchSelectAll() {
subscribeListViewRef.value?.toggleSelectAll()
}
// 批量启用已选订阅。
function batchEnableSelectedSubscribes() {
subscribeListViewRef.value?.batchEnableSubscribes()
}
// 批量暂停已选订阅。
function batchPauseSelectedSubscribes() {
subscribeListViewRef.value?.batchPauseSubscribes()
}
// 批量删除已选订阅。
function batchDeleteSelectedSubscribes() {
subscribeListViewRef.value?.batchDeleteSubscribes()
}
// 切换订阅拖拽排序模式,进入时固定使用自定义排序。
function toggleSubscribeSortMode() {
if (!subscribeSortMode.value) {
@@ -241,6 +302,10 @@ watch(activeTab, newTab => {
if (newTab !== 'share') {
searchShareDialog.value = false
}
if (newTab !== 'mysub' && subscribeBatchState.value.enabled) {
exitSubscribeBatchMode()
}
})
watch(subscribeSortBy, newSortBy => {
@@ -251,17 +316,68 @@ onUnmounted(() => {
shareKeywordUpdater.cancel()
})
const subscribeDynamicMenuItems = computed(() => {
const subscribeDynamicMenuItems = computed<DynamicButtonMenuItem[] | undefined>(() => {
if (!appMode.value) return undefined
if (activeTab.value === 'mysub') {
const items: Array<{
titleKey: string
titleParams?: Record<string, unknown>
icon: string
permission: 'admin'
action: () => void
}> = []
if (subscribeBatchState.value.enabled) {
const hasSelectedSubscribes = subscribeBatchState.value.selectedCount > 0
return [
{
titleKey: 'subscribe.selectedCount',
titleParams: {
count: subscribeBatchState.value.selectedCount,
total: subscribeBatchState.value.totalCount,
},
icon: 'mdi-checkbox-multiple-marked-outline',
permission: 'subscribe',
disabled: true,
action: () => {},
},
{
titleKey: subscribeBatchState.value.allSelected
? 'subscribe.batchDeselectAll'
: 'subscribe.batchSelectAll',
icon: subscribeBatchState.value.allSelected ? 'mdi-checkbox-blank-outline' : 'mdi-checkbox-multiple-marked',
permission: 'subscribe',
disabled: subscribeBatchState.value.totalCount === 0,
action: toggleSubscribeBatchSelectAll,
},
{
titleKey: 'subscribe.batchEnable',
icon: 'mdi-play',
color: 'success',
permission: 'subscribe',
disabled: !hasSelectedSubscribes,
action: batchEnableSelectedSubscribes,
},
{
titleKey: 'subscribe.batchPause',
icon: 'mdi-pause',
color: 'info',
permission: 'subscribe',
disabled: !hasSelectedSubscribes,
action: batchPauseSelectedSubscribes,
},
{
titleKey: 'subscribe.batchDelete',
icon: 'mdi-delete',
color: 'error',
permission: 'subscribe',
disabled: !hasSelectedSubscribes,
action: batchDeleteSelectedSubscribes,
},
{
titleKey: 'subscribe.exitBatchMode',
icon: 'mdi-close',
permission: 'subscribe',
action: exitSubscribeBatchMode,
},
]
}
const items: DynamicButtonMenuItem[] = []
if (showSubscribeHistoryAction.value) {
items.push({
@@ -287,12 +403,18 @@ const subscribeDynamicMenuItems = computed(() => {
})
const subscribeDynamicIcon = computed(() => {
if (subscribeBatchState.value.enabled) return 'mdi-checkbox-multiple-marked-outline'
if (showShareStatisticsAction.value) return 'mdi-chart-line'
if (showSubscribeHistoryAction.value) return 'mdi-history'
return 'mdi-clipboard-edit-outline'
})
function handleSubscribeDynamicAction() {
if (subscribeBatchState.value.enabled) {
exitSubscribeBatchMode()
return
}
if (showShareStatisticsAction.value) {
openShareStatisticsDialog()
return
@@ -313,7 +435,9 @@ useDynamicButton({
onClick: handleSubscribeDynamicAction,
menuItems: subscribeDynamicMenuItems,
permission: 'subscribe',
show: computed(() => appMode.value && (showDefaultRuleAction.value || showShareStatisticsAction.value)),
show: computed(
() => appMode.value && (subscribeBatchState.value.enabled || showDefaultRuleAction.value || showShareStatisticsAction.value),
),
})
// 使用动态标签页
@@ -348,13 +472,16 @@ registerHeaderTab({
{
icon: 'mdi-checkbox-multiple-marked-outline',
variant: 'text',
color: 'gray',
color: computed(() => (subscribeBatchState.value.enabled ? 'primary' : 'gray')),
class: 'settings-icon-button',
permission: 'subscribe',
action: () => {
// 触发批量管理模式
const event = new CustomEvent('toggle-batch-mode')
window.dispatchEvent(event)
if (subscribeBatchState.value.enabled) {
exitSubscribeBatchMode()
return
}
enterSubscribeBatchMode()
},
show: computed(() => activeTab.value === 'mysub'),
},
@@ -399,6 +526,7 @@ onMounted(() => {
:active="activeTab === 'mysub'"
@update:sort-mode="subscribeSortMode = $event"
@update:sort-by="subscribeSortBy = $event"
@batch-state-change="handleSubscribeBatchStateChange"
/>
</div>
</transition>
@@ -513,7 +641,56 @@ onMounted(() => {
<Teleport to="body" v-if="!appMode && route.path.startsWith(`/subscribe/${subType === '电影' ? 'movie' : 'tv'}`)">
<div class="compact-fab-stack">
<VFab
v-if="showSubscribeHistoryAction"
v-if="subscribeBatchState.enabled"
icon="mdi-close"
color="secondary"
variant="tonal"
appear
class="compact-fab compact-fab--secondary"
@click="exitSubscribeBatchMode"
/>
<VFab
v-if="subscribeBatchState.enabled"
:icon="subscribeBatchState.allSelected ? 'mdi-checkbox-blank-outline' : 'mdi-checkbox-multiple-marked'"
color="primary"
variant="tonal"
appear
class="compact-fab compact-fab--secondary"
:disabled="subscribeBatchState.totalCount === 0"
@click="toggleSubscribeBatchSelectAll"
/>
<VFab
v-if="subscribeBatchState.enabled"
icon="mdi-delete"
color="error"
variant="tonal"
appear
class="compact-fab compact-fab--secondary"
:disabled="subscribeBatchState.selectedCount === 0"
@click="batchDeleteSelectedSubscribes"
/>
<VFab
v-if="subscribeBatchState.enabled"
icon="mdi-pause"
color="info"
variant="tonal"
appear
class="compact-fab compact-fab--secondary"
:disabled="subscribeBatchState.selectedCount === 0"
@click="batchPauseSelectedSubscribes"
/>
<VFab
v-if="subscribeBatchState.enabled"
icon="mdi-play"
color="success"
variant="tonal"
appear
class="compact-fab compact-fab--secondary"
:disabled="subscribeBatchState.selectedCount === 0"
@click="batchEnableSelectedSubscribes"
/>
<VFab
v-if="!subscribeBatchState.enabled && showSubscribeHistoryAction"
icon="mdi-history"
color="info"
variant="tonal"
@@ -522,7 +699,7 @@ onMounted(() => {
@click="openSubscribeHistoryDialog"
/>
<VFab
v-if="showDefaultRuleAction"
v-if="!subscribeBatchState.enabled && showDefaultRuleAction"
icon="mdi-clipboard-edit-outline"
color="primary"
appear
@@ -530,7 +707,7 @@ onMounted(() => {
@click="openDefaultRuleDialog"
/>
<VFab
v-if="showShareStatisticsAction"
v-if="!subscribeBatchState.enabled && showShareStatisticsAction"
icon="mdi-chart-line"
color="primary"
appear

View File

@@ -78,6 +78,9 @@ const redoTargetStorage = ref<string>()
// 已选中的数据
const selected = ref<TransferHistory[]>([])
// 移动端批量选择模式
const mobileBatchMode = ref(false)
// 从季集字符串中提取可排序的数字。
const getNum = (s?: string) => (s ? parseInt(s.replace(/[^0-9]/g, ''), 10) || 0 : 0)
@@ -504,6 +507,7 @@ function resetMobileHistory() {
totalItems.value = 0
mobileExpandedPathIds.value = []
selected.value = []
mobileBatchMode.value = false
mobileInfiniteKey.value++
}
@@ -699,6 +703,9 @@ async function removeBatch(deleteSrc: boolean, deleteDest: boolean) {
}
// 清空选中项
selected.value = []
if (isMobile.value) {
mobileBatchMode.value = false
}
// 隐藏进度条
closeProgressDialog()
// 重新获取数据
@@ -742,6 +749,9 @@ async function transferDone() {
// 清空当前操作记录
currentHistory.value = undefined
selected.value = []
if (isMobile.value) {
mobileBatchMode.value = false
}
// 刷新
await refreshDataAfterOperation()
}
@@ -886,6 +896,9 @@ async function triggerBatchAiRedo() {
}
startAiRedoProgressBatch(acceptedIds, progressKey)
selected.value = selected.value.filter(item => !acceptedIds.includes(item.id))
if (isMobile.value && selected.value.length === 0) {
mobileBatchMode.value = false
}
progressStarted = true
} catch (error) {
console.error(error)
@@ -984,6 +997,19 @@ function ensurePageSize(value: any, defaultValue: number = 50) {
// 已选历史记录 ID 集合,供移动端卡片和分组选择状态复用。
const selectedIdSet = computed(() => new Set(selected.value.map(item => item.id)))
// 移动端当前已加载记录数量,用于批量菜单展示选择进度。
const mobileBatchTotalCount = computed(() => mobileDataList.value.length)
// 移动端当前已加载记录中的已选数量。
const mobileBatchSelectedCount = computed(() => {
return mobileDataList.value.filter(item => selectedIdSet.value.has(item.id)).length
})
// 移动端当前已加载记录是否已全部选中。
const isAllMobileHistorySelected = computed(() => {
return mobileBatchTotalCount.value > 0 && mobileBatchSelectedCount.value === mobileBatchTotalCount.value
})
// 拼接移动端展示用的季集文本。
function getHistoryEpisodeText(item: TransferHistory) {
return `${item.seasons || ''}${item.episodes || ''}`
@@ -1085,6 +1111,50 @@ function toggleHistorySelection(item: TransferHistory, checked: boolean | null)
updateHistorySelection([item], checked)
}
// 切换移动端批量选择模式,退出时清空移动端选择状态。
function toggleMobileBatchMode() {
if (mobileBatchMode.value) {
exitMobileBatchMode()
return
}
selected.value = []
mobileBatchMode.value = true
}
// 退出移动端批量选择模式并清空选择状态。
function exitMobileBatchMode() {
mobileBatchMode.value = false
selected.value = []
}
// 批量模式下点击移动端记录卡片时切换该记录的选择状态。
function handleMobileRecordClick(item: TransferHistory) {
if (!mobileBatchMode.value) return
toggleHistorySelection(item, !isHistorySelected(item))
}
// 移动端路径点击在批量模式下转为选择记录,普通模式下展开路径。
function handleMobilePathClick(item: TransferHistory) {
if (mobileBatchMode.value) {
handleMobileRecordClick(item)
return
}
toggleMobilePathExpanded(item)
}
// 选中移动端当前已加载的全部历史记录。
function selectAllMobileHistory() {
updateHistorySelection(mobileDataList.value, true)
}
// 取消移动端历史记录的全部选择。
function deselectAllMobileHistory() {
updateHistorySelection(mobileDataList.value, false)
}
// 按标题分组后的选中数量统计,键为标题,值为对应分组的选中数
const selectedCountsGroupedByTitle = computed(() => {
return selected.value.reduce(
@@ -1103,14 +1173,95 @@ const toggleGroupSelection = (checked: boolean | null, items: readonly any[]) =>
updateHistorySelection(values, checked)
}
const historyDynamicIcon = computed(() => (selected.value.length > 0 ? 'mdi-chevron-up' : 'mdi-timer-sand-paused'))
const historyDynamicIcon = computed(() => 'mdi-timer-sand-paused')
const historyDynamicMenuItems = computed(() => {
if (!appMode.value) return undefined
if (mobileBatchMode.value) {
const hasSelectedHistory = mobileBatchSelectedCount.value > 0
const items: DynamicButtonMenuItem[] = [
{
titleKey: 'transferHistory.selectedCount',
titleParams: {
count: mobileBatchSelectedCount.value,
total: mobileBatchTotalCount.value,
},
icon: 'mdi-checkbox-multiple-marked-outline',
permission: 'manage',
disabled: true,
action: () => {},
},
{
titleKey: isAllMobileHistorySelected.value
? 'transferHistory.actions.deselectAll'
: 'transferHistory.actions.selectAll',
icon: isAllMobileHistorySelected.value ? 'mdi-checkbox-blank-outline' : 'mdi-checkbox-multiple-marked',
permission: 'manage',
disabled: mobileBatchTotalCount.value === 0,
action: () => {
if (isAllMobileHistorySelected.value) {
deselectAllMobileHistory()
return
}
selectAllMobileHistory()
},
},
]
if (!hasRunningAiRedo.value) {
items.push(
{
titleKey: 'transferHistory.actions.batchAiRedo',
icon: 'mdi-robot-outline',
color: 'info',
permission: 'manage',
disabled: !hasSelectedHistory,
action: () => {
triggerBatchAiRedo()
},
},
{
titleKey: 'transferHistory.actions.batchRedo',
icon: 'mdi-redo-variant',
color: 'success',
permission: 'manage',
disabled: !hasSelectedHistory,
action: () => {
retransferBatch()
},
},
{
titleKey: 'transferHistory.actions.batchDelete',
icon: 'mdi-trash-can-outline',
color: 'error',
permission: 'manage',
disabled: !hasSelectedHistory,
action: () => {
removeHistoryBatch()
},
},
)
}
items.push({
titleKey: 'transferHistory.actions.exitBatchMode',
icon: 'mdi-close',
permission: 'manage',
action: exitMobileBatchMode,
})
return items
}
if (selected.value.length === 0) return undefined
const items: DynamicButtonMenuItem[] = [
{
titleKey: 'dialog.transferQueue.title',
icon: 'mdi-timer-sand-paused',
color: 'primary',
permission: 'manage',
action: openTransferQueueDialog,
},
@@ -1121,6 +1272,7 @@ const historyDynamicMenuItems = computed(() => {
{
titleKey: 'transferHistory.actions.batchAiRedo',
icon: 'mdi-robot-outline',
color: 'info',
permission: 'manage',
action: () => {
triggerBatchAiRedo()
@@ -1129,6 +1281,7 @@ const historyDynamicMenuItems = computed(() => {
{
titleKey: 'transferHistory.actions.batchRedo',
icon: 'mdi-redo-variant',
color: 'success',
permission: 'manage',
action: () => {
retransferBatch()
@@ -1442,7 +1595,23 @@ onUnmounted(() => {
<section v-else class="transfer-history-mobile-page">
<div class="transfer-history-mobile-titlebar">
<VPageContentTitle :title="t('navItems.mediaOrganize')" class="transfer-history-mobile-title my-0" style="margin-block: 0" />
<VPageContentTitle
:title="t('navItems.mediaOrganize')"
class="transfer-history-mobile-title my-0"
style="margin-block: 0"
/>
<VBtn
v-if="canManage"
icon="mdi-checkbox-multiple-marked-outline"
:color="mobileBatchMode ? 'primary' : 'gray'"
:aria-label="
mobileBatchMode ? t('transferHistory.actions.exitBatchSelect') : t('transferHistory.actions.batchSelect')
"
:title="mobileBatchMode ? t('transferHistory.actions.exitBatchSelect') : t('transferHistory.actions.batchSelect')"
variant="text"
class="settings-icon-button transfer-history-mobile-titlebar__batch"
@click="toggleMobileBatchMode"
/>
</div>
<VCombobox
@@ -1491,9 +1660,11 @@ onUnmounted(() => {
:ref="itemRef"
class="transfer-history-mobile-record"
:class="{
'transfer-history-mobile-record--selected': isHistorySelected(item),
'transfer-history-mobile-record--batch': mobileBatchMode,
'transfer-history-mobile-record--selected': mobileBatchMode && isHistorySelected(item),
'transfer-history-mobile-record--failed': !item.status,
}"
@click="handleMobileRecordClick(item)"
>
<header class="transfer-history-mobile-record__header">
<VAvatar class="transfer-history-mobile-record__avatar" size="40">
@@ -1513,7 +1684,17 @@ onUnmounted(() => {
{{ getHistoryStatusText(item) }}
</VChip>
<IconBtn class="transfer-history-mobile-record__menu" size="small">
<VCheckbox
v-if="mobileBatchMode"
class="transfer-history-mobile-record__checkbox"
:model-value="isHistorySelected(item)"
density="compact"
hide-details
@click.stop
@update:model-value="checked => toggleHistorySelection(item, checked)"
/>
<IconBtn v-else class="transfer-history-mobile-record__menu" size="small">
<VIcon icon="mdi-dots-vertical" />
<VMenu activator="parent" close-on-content-click>
<VList>
@@ -1547,7 +1728,7 @@ onUnmounted(() => {
type="button"
class="transfer-history-mobile-record__paths"
:class="{ 'transfer-history-mobile-record__paths--expanded': isMobilePathExpanded(item) }"
@click="toggleMobilePathExpanded(item)"
@click.stop="handleMobilePathClick(item)"
>
<div class="transfer-history-mobile-record__path-row">
<span class="transfer-history-mobile-record__storage">
@@ -1582,7 +1763,7 @@ onUnmounted(() => {
</section>
<!-- 非 app 模式下的 FAB 按钮 -->
<Teleport to="body" v-if="!appMode && route.path === '/history' && isDesktop">
<Teleport to="body" v-if="!appMode && route.path === '/history'">
<div v-if="isRefreshed && canManage" class="compact-fab-stack compact-fab-stack--history">
<VFab
v-if="selected.length > 0 && !hasRunningAiRedo"
@@ -1649,12 +1830,19 @@ onUnmounted(() => {
.transfer-history-mobile-titlebar {
display: flex;
align-items: center;
gap: 0.5rem;
justify-content: space-between;
}
.transfer-history-mobile-title {
flex: 1;
min-inline-size: 0;
}
.transfer-history-mobile-titlebar__batch {
flex: 0 0 auto;
}
.transfer-history-mobile-title :deep(h2) {
font-size: 1.875rem;
line-height: 1.15;
@@ -1714,6 +1902,15 @@ onUnmounted(() => {
box-shadow: var(--app-card-rest-shadow);
}
.transfer-history-mobile-record--batch {
cursor: pointer;
}
.transfer-history-mobile-record--selected {
outline: 2px solid rgb(var(--v-theme-primary));
outline-offset: -2px;
}
.transfer-history-mobile-record + .transfer-history-mobile-record {
margin-block-start: 0.875rem;
}
@@ -1770,6 +1967,17 @@ onUnmounted(() => {
justify-self: end;
}
.transfer-history-mobile-record__checkbox {
align-self: start;
justify-self: end;
margin-block-start: -0.35rem;
margin-inline-end: -0.35rem;
}
.transfer-history-mobile-record__checkbox :deep(.v-selection-control) {
min-block-size: 2rem;
}
.transfer-history-mobile-record__meta {
display: flex;
align-items: center;

View File

@@ -53,10 +53,19 @@ const props = defineProps({
const emit = defineEmits<{
'update:sortMode': [value: boolean]
'update:sortBy': [value: SubscribeSortBy]
'batch-state-change': [state: SubscribeBatchState]
}>()
type SubscribeSortBy = 'custom' | 'last_update' | 'date' | 'lack_episode'
// 订阅批量模式状态快照,供父页面渲染外部批量操作按钮。
interface SubscribeBatchState {
enabled: boolean
selectedCount: number
totalCount: number
allSelected: boolean
}
// 是否刷新过
let isRefreshed = ref(false)
@@ -79,6 +88,9 @@ const selectedSubscribes = ref<number[]>([])
const normalizedKeyword = computed(() => props.keyword?.trim().toLowerCase() || '')
const selectedSubscribesSet = computed(() => new Set(selectedSubscribes.value))
const hasCustomOrder = computed(() => orderConfig.value.length > 0)
const isAllSubscribesSelected = computed(
() => displayList.value.length > 0 && selectedSubscribes.value.length === displayList.value.length,
)
// 归一化订阅排序方式,电影订阅不使用缺失集数排序。
const normalizedSortBy = computed<SubscribeSortBy | ''>(() => {
@@ -248,6 +260,12 @@ watch(
{ immediate: true },
)
watch(
[isBatchMode, () => selectedSubscribes.value.length, () => displayList.value.length, isAllSubscribesSelected],
emitBatchStateChange,
{ immediate: true },
)
// 加载顺序
async function loadSubscribeOrderConfig() {
try {
@@ -313,25 +331,47 @@ function openHistoryDialog() {
)
}
// 批量管理相关函数
// 切换批量模式
function toggleBatchMode() {
isBatchMode.value = !isBatchMode.value
if (!isBatchMode.value) {
selectedSubscribes.value = []
}
// 向父组件同步批量操作状态,供 Footer/FAB 动态按钮渲染。
function emitBatchStateChange() {
emit('batch-state-change', {
enabled: isBatchMode.value,
selectedCount: selectedSubscribes.value.length,
totalCount: displayList.value.length,
allSelected: isAllSubscribesSelected.value,
})
}
// 全选/取消全选
// 进入批量模式。
function enterBatchMode() {
isBatchMode.value = true
}
// 退出批量模式并清空已选择的订阅。
function exitBatchMode() {
isBatchMode.value = false
selectedSubscribes.value = []
}
// 切换批量模式。
function toggleBatchMode() {
if (isBatchMode.value) {
exitBatchMode()
return
}
enterBatchMode()
}
// 全选或取消全选当前显示的订阅。
function toggleSelectAll() {
if (selectedSubscribes.value.length === displayList.value.length) {
if (isAllSubscribesSelected.value) {
selectedSubscribes.value = []
} else {
selectedSubscribes.value = displayList.value.map(item => item.id)
}
}
// 选择单个订阅
// 切换单个订阅的选中状态。
function toggleSelectSubscribe(id: number) {
const index = selectedSubscribes.value.indexOf(id)
if (index > -1) {
@@ -341,7 +381,7 @@ function toggleSelectSubscribe(id: number) {
}
}
// 批量删除订阅
// 批量删除已选中的订阅
async function batchDeleteSubscribes() {
if (selectedSubscribes.value.length === 0) {
$toast.warning(t('subscribe.noSelectedItems'))
@@ -370,11 +410,8 @@ async function batchDeleteSubscribes() {
$toast.error(t('subscribe.batchDeleteFailed', { count: failedCount }))
}
// 刷新数据
await fetchData()
// 退出批量模式
isBatchMode.value = false
selectedSubscribes.value = []
exitBatchMode()
} catch (error) {
console.error(error)
$toast.error(t('subscribe.batchDeleteError'))
@@ -383,7 +420,7 @@ async function batchDeleteSubscribes() {
}
}
// 批量启用订阅
// 批量启用已选中的订阅
async function batchEnableSubscribes() {
if (selectedSubscribes.value.length === 0) {
$toast.warning(t('subscribe.noSelectedItems'))
@@ -412,11 +449,8 @@ async function batchEnableSubscribes() {
$toast.error(t('subscribe.batchEnableFailed', { count: failedCount }))
}
// 刷新数据
await fetchData()
// 退出批量模式
isBatchMode.value = false
selectedSubscribes.value = []
exitBatchMode()
} catch (error) {
console.error(error)
$toast.error(t('subscribe.batchEnableError'))
@@ -425,7 +459,7 @@ async function batchEnableSubscribes() {
}
}
// 批量暂停订阅
// 批量暂停已选中的订阅
async function batchPauseSubscribes() {
if (selectedSubscribes.value.length === 0) {
$toast.warning(t('subscribe.noSelectedItems'))
@@ -454,11 +488,8 @@ async function batchPauseSubscribes() {
$toast.error(t('subscribe.batchPauseFailed', { count: failedCount }))
}
// 刷新数据
await fetchData()
// 退出批量模式
isBatchMode.value = false
selectedSubscribes.value = []
exitBatchMode()
} catch (error) {
console.error(error)
$toast.error(t('subscribe.batchPauseError'))
@@ -495,13 +526,6 @@ onMounted(async () => {
}
}
// 监听批量管理模式切换事件
window.addEventListener('toggle-batch-mode', toggleBatchMode)
})
onUnmounted(() => {
// 移除事件监听器
window.removeEventListener('toggle-batch-mode', toggleBatchMode)
})
useKeepAliveRefresh(fetchData, {
@@ -510,68 +534,19 @@ useKeepAliveRefresh(fetchData, {
defineExpose({
openHistoryDialog,
enterBatchMode,
exitBatchMode,
toggleBatchMode,
toggleSelectAll,
batchEnableSubscribes,
batchPauseSubscribes,
batchDeleteSubscribes,
})
</script>
<template>
<LoadingBanner v-if="!isRefreshed" class="mt-12" />
<!-- 批量管理工具栏 -->
<div v-if="isBatchMode" class="mb-4 px-2">
<VCard class="pa-4">
<div class="d-flex align-center justify-space-between">
<div class="d-flex align-center">
<VCheckbox
:model-value="selectedSubscribes.length === displayList.length"
:indeterminate="selectedSubscribes.length > 0 && selectedSubscribes.length < displayList.length"
@update:model-value="toggleSelectAll"
hide-details
class="me-4"
/>
<span class="text-body-1 font-weight-medium">
{{ t('subscribe.selectedCount', { count: selectedSubscribes.length, total: displayList.length }) }}
</span>
</div>
<div class="d-flex gap-2">
<VBtn
color="success"
variant="outlined"
size="small"
:disabled="selectedSubscribes.length === 0"
@click="batchEnableSubscribes"
>
<VIcon icon="mdi-play" class="me-sm-1" />
<span class="d-none d-sm-inline">{{ t('subscribe.batchEnable') }}</span>
</VBtn>
<VBtn
color="info"
variant="outlined"
size="small"
:disabled="selectedSubscribes.length === 0"
@click="batchPauseSubscribes"
>
<VIcon icon="mdi-pause" class="me-sm-1" />
<span class="d-none d-sm-inline">{{ t('subscribe.batchPause') }}</span>
</VBtn>
<VBtn
color="error"
variant="outlined"
size="small"
:disabled="selectedSubscribes.length === 0"
@click="batchDeleteSubscribes"
>
<VIcon icon="mdi-delete" class="me-sm-1" />
<span class="d-none d-sm-inline">{{ t('subscribe.batchDelete') }}</span>
</VBtn>
<VBtn color="secondary" variant="outlined" size="small" @click="toggleBatchMode">
<VIcon icon="mdi-close" class="me-sm-1" />
<span class="d-none d-sm-inline">{{ t('common.cancel') }}</span>
</VBtn>
</div>
</div>
</VCard>
</div>
<VAlert v-if="sortMode" color="warning" variant="tonal" class="mb-4 mx-2 py-0 app-surface-static">
<div class="d-flex flex-wrap align-center justify-space-between gap-2 py-5">
<span>{{ t('common.sortModeHint') }}</span>