From b4e369b556f73fbb031a887e22f59a049d5bf904 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Tue, 3 Dec 2024 10:44:52 -0800 Subject: [PATCH] Enhancement: file task filtering (#8421) --- src-ui/messages.xlf | 36 ++++-- .../admin/tasks/tasks.component.html | 37 +++++- .../admin/tasks/tasks.component.scss | 11 ++ .../admin/tasks/tasks.component.spec.ts | 61 +++++++++- .../components/admin/tasks/tasks.component.ts | 106 ++++++++++++++++-- 5 files changed, 220 insertions(+), 31 deletions(-) diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index af014edea..9329edb5d 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -1880,6 +1880,10 @@ src/app/components/admin/tasks/tasks.component.html 36 + + src/app/components/admin/tasks/tasks.component.ts + 30 + src/app/components/admin/trash/trash.component.html 35 @@ -2037,7 +2041,7 @@ src/app/components/admin/tasks/tasks.component.ts - 68 + 124 @@ -2089,60 +2093,74 @@ 147,149 + + Filter by + + src/app/components/admin/tasks/tasks.component.html + 157 + + + + Result + + src/app/components/admin/tasks/tasks.component.ts + 31 + + Dismiss selected src/app/components/admin/tasks/tasks.component.ts - 31 + 77 Dismiss all src/app/components/admin/tasks/tasks.component.ts - 32 + 78 Confirm Dismiss All src/app/components/admin/tasks/tasks.component.ts - 65 + 121 Dismiss all tasks? src/app/components/admin/tasks/tasks.component.ts - 66 + 122 queued src/app/components/admin/tasks/tasks.component.ts - 135 + 207 started src/app/components/admin/tasks/tasks.component.ts - 137 + 209 completed src/app/components/admin/tasks/tasks.component.ts - 139 + 211 failed src/app/components/admin/tasks/tasks.component.ts - 141 + 213 diff --git a/src-ui/src/app/components/admin/tasks/tasks.component.html b/src-ui/src/app/components/admin/tasks/tasks.component.html index 3d40c7897..75dda1394 100644 --- a/src-ui/src/app/components/admin/tasks/tasks.component.html +++ b/src-ui/src/app/components/admin/tasks/tasks.component.html @@ -118,13 +118,13 @@ - + Failed@if (tasksService.failedFileTasks.length > 0) { {{tasksService.failedFileTasks.length}} } - + @@ -132,7 +132,7 @@ {{tasksService.completedFileTasks.length}} } - + @@ -140,7 +140,7 @@ {{tasksService.startedFileTasks.length}} } - + @@ -148,8 +148,35 @@ {{tasksService.queuedFileTasks.length}} } - + + + + + Filter by + @if (filterTargets.length > 1) { + + {{filterTargetName}} + + @for (t of filterTargets; track t.id) { + {{t.name}} + } + + + } @else { + {{filterTargetName}} + } + @if (filterText?.length) { + + + + } + + + + diff --git a/src-ui/src/app/components/admin/tasks/tasks.component.scss b/src-ui/src/app/components/admin/tasks/tasks.component.scss index 60f8f2297..325fd2c02 100644 --- a/src-ui/src/app/components/admin/tasks/tasks.component.scss +++ b/src-ui/src/app/components/admin/tasks/tasks.component.scss @@ -26,3 +26,14 @@ pre { max-width: 150px; } } + +.input-group .dropdown .btn { + border-top-right-radius: 0; + border-bottom-right-radius: 0; + border-top-left-radius: 0; + border-bottom-left-radius: 0; +} + +.z-10 { + z-index: 10; +} diff --git a/src-ui/src/app/components/admin/tasks/tasks.component.spec.ts b/src-ui/src/app/components/admin/tasks/tasks.component.spec.ts index 1ad2d5311..4d3e600ac 100644 --- a/src-ui/src/app/components/admin/tasks/tasks.component.spec.ts +++ b/src-ui/src/app/components/admin/tasks/tasks.component.spec.ts @@ -26,7 +26,7 @@ import { TasksService } from 'src/app/services/tasks.service' import { environment } from 'src/environments/environment' import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' import { PageHeaderComponent } from '../../common/page-header/page-header.component' -import { TasksComponent } from './tasks.component' +import { TasksComponent, TaskTab } from './tasks.component' import { PermissionsGuard } from 'src/app/guards/permissions.guard' import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' import { FormsModule } from '@angular/forms' @@ -167,7 +167,7 @@ describe('TasksComponent', () => { let currentTasksLength = tasks.filter( (t) => t.status === PaperlessTaskStatus.Failed ).length - component.activeTab = 'failed' + component.activeTab = TaskTab.Failed fixture.detectChanges() expect(tabButtons[0].nativeElement.textContent).toEqual( `Failed${currentTasksLength}` @@ -179,7 +179,7 @@ describe('TasksComponent', () => { currentTasksLength = tasks.filter( (t) => t.status === PaperlessTaskStatus.Complete ).length - component.activeTab = 'completed' + component.activeTab = TaskTab.Completed fixture.detectChanges() expect(tabButtons[1].nativeElement.textContent).toEqual( `Complete${currentTasksLength}` @@ -188,7 +188,7 @@ describe('TasksComponent', () => { currentTasksLength = tasks.filter( (t) => t.status === PaperlessTaskStatus.Started ).length - component.activeTab = 'started' + component.activeTab = TaskTab.Started fixture.detectChanges() expect(tabButtons[2].nativeElement.textContent).toEqual( `Started${currentTasksLength}` @@ -197,7 +197,7 @@ describe('TasksComponent', () => { currentTasksLength = tasks.filter( (t) => t.status === PaperlessTaskStatus.Pending ).length - component.activeTab = 'queued' + component.activeTab = TaskTab.Queued fixture.detectChanges() expect(tabButtons[3].nativeElement.textContent).toEqual( `Queued${currentTasksLength}` @@ -206,7 +206,7 @@ describe('TasksComponent', () => { it('should to go page 1 between tab switch', () => { component.page = 10 - component.duringTabChange(2) + component.duringTabChange() expect(component.page).toEqual(1) }) @@ -289,4 +289,53 @@ describe('TasksComponent', () => { jest.advanceTimersByTime(6000) expect(reloadSpy).toHaveBeenCalledTimes(2) }) + + it('should filter tasks by file name', () => { + const input = fixture.debugElement.query(By.css('ul input[type=text]')) + input.nativeElement.value = '191092' + input.nativeElement.dispatchEvent(new Event('input')) + jest.advanceTimersByTime(150) // debounce time + fixture.detectChanges() + expect(component.filterText).toEqual('191092') + expect( + fixture.debugElement.queryAll(By.css('table tbody tr')).length + ).toEqual(2) // 1 task x 2 lines + }) + + it('should filter tasks by result', () => { + component.activeTab = TaskTab.Failed + fixture.detectChanges() + component.filterTargetID = 1 + const input = fixture.debugElement.query(By.css('ul input[type=text]')) + input.nativeElement.value = 'duplicate' + input.nativeElement.dispatchEvent(new Event('input')) + jest.advanceTimersByTime(150) // debounce time + fixture.detectChanges() + expect(component.filterText).toEqual('duplicate') + expect( + fixture.debugElement.queryAll(By.css('table tbody tr')).length + ).toEqual(4) // 2 tasks x 2 lines + }) + + it('should support keyboard events for filtering', () => { + const input = fixture.debugElement.query(By.css('ul input[type=text]')) + input.nativeElement.value = '191092' + input.nativeElement.dispatchEvent( + new KeyboardEvent('keyup', { key: 'Enter' }) + ) + expect(component.filterText).toEqual('191092') // no debounce needed + input.nativeElement.dispatchEvent( + new KeyboardEvent('keyup', { key: 'Escape' }) + ) + expect(component.filterText).toEqual('') + }) + + it('should reset filter and target on tab switch', () => { + component.filterText = '191092' + component.filterTargetID = 1 + component.activeTab = TaskTab.Completed + component.beforeTabChange() + expect(component.filterText).toEqual('') + expect(component.filterTargetID).toEqual(0) + }) }) diff --git a/src-ui/src/app/components/admin/tasks/tasks.component.ts b/src-ui/src/app/components/admin/tasks/tasks.component.ts index 7b01090d5..6539b3692 100644 --- a/src-ui/src/app/components/admin/tasks/tasks.component.ts +++ b/src-ui/src/app/components/admin/tasks/tasks.component.ts @@ -1,12 +1,36 @@ import { Component, OnInit, OnDestroy } from '@angular/core' import { Router } from '@angular/router' import { NgbModal } from '@ng-bootstrap/ng-bootstrap' -import { first } from 'rxjs' +import { + debounceTime, + distinctUntilChanged, + filter, + first, + Subject, + takeUntil, +} from 'rxjs' import { PaperlessTask } from 'src/app/data/paperless-task' import { TasksService } from 'src/app/services/tasks.service' import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component' +export enum TaskTab { + Queued = 'queued', + Started = 'started', + Completed = 'completed', + Failed = 'failed', +} + +enum TaskFilterTargetID { + Name, + Result, +} + +const FILTER_TARGETS = [ + { id: TaskFilterTargetID.Name, name: $localize`Name` }, + { id: TaskFilterTargetID.Result, name: $localize`Result` }, +] + @Component({ selector: 'pngx-tasks', templateUrl: './tasks.component.html', @@ -16,7 +40,7 @@ export class TasksComponent extends ComponentWithPermissions implements OnInit, OnDestroy { - public activeTab: string + public activeTab: TaskTab public selectedTasks: Set = new Set() public togggleAll: boolean = false public expandedTask: number @@ -26,6 +50,28 @@ export class TasksComponent public autoRefreshInterval: any + private _filterText: string = '' + get filterText() { + return this._filterText + } + set filterText(value: string) { + this.filterDebounce.next(value) + } + + public filterTargetID: TaskFilterTargetID = TaskFilterTargetID.Name + public get filterTargetName(): string { + return this.filterTargets.find((t) => t.id == this.filterTargetID).name + } + private filterDebounce: Subject = new Subject() + + public get filterTargets(): Array<{ id: number; name: string }> { + return [TaskTab.Failed, TaskTab.Completed].includes(this.activeTab) + ? FILTER_TARGETS + : FILTER_TARGETS.slice(0, 1) + } + + private unsubscribeNotifier: Subject = new Subject() + get dismissButtonText(): string { return this.selectedTasks.size > 0 ? $localize`Dismiss selected` @@ -43,11 +89,21 @@ export class TasksComponent ngOnInit() { this.tasksService.reload() this.toggleAutoRefresh() + + this.filterDebounce + .pipe( + takeUntil(this.unsubscribeNotifier), + debounceTime(100), + distinctUntilChanged(), + filter((query) => !query.length || query.length > 2) + ) + .subscribe((query) => (this._filterText = query)) } ngOnDestroy() { this.tasksService.cancelPending() clearInterval(this.autoRefreshInterval) + this.unsubscribeNotifier.next(this) } dismissTask(task: PaperlessTask) { @@ -96,19 +152,30 @@ export class TasksComponent get currentTasks(): PaperlessTask[] { let tasks: PaperlessTask[] = [] switch (this.activeTab) { - case 'queued': + case TaskTab.Queued: tasks = this.tasksService.queuedFileTasks break - case 'started': + case TaskTab.Started: tasks = this.tasksService.startedFileTasks break - case 'completed': + case TaskTab.Completed: tasks = this.tasksService.completedFileTasks break - case 'failed': + case TaskTab.Failed: tasks = this.tasksService.failedFileTasks break } + if (this._filterText.length) { + tasks = tasks.filter((t) => { + if (this.filterTargetID == TaskFilterTargetID.Name) { + return t.task_file_name + .toLowerCase() + .includes(this._filterText.toLowerCase()) + } else if (this.filterTargetID == TaskFilterTargetID.Result) { + return t.result.toLowerCase().includes(this._filterText.toLowerCase()) + } + }) + } return tasks } @@ -125,19 +192,24 @@ export class TasksComponent this.selectedTasks.clear() } - duringTabChange(navID: number) { + duringTabChange() { this.page = 1 } + beforeTabChange() { + this.resetFilter() + this.filterTargetID = TaskFilterTargetID.Name + } + get activeTabLocalized(): string { switch (this.activeTab) { - case 'queued': + case TaskTab.Queued: return $localize`queued` - case 'started': + case TaskTab.Started: return $localize`started` - case 'completed': + case TaskTab.Completed: return $localize`completed` - case 'failed': + case TaskTab.Failed: return $localize`failed` } } @@ -152,4 +224,16 @@ export class TasksComponent }, 5000) } } + + public resetFilter() { + this._filterText = '' + } + + filterInputKeyup(event: KeyboardEvent) { + if (event.key == 'Enter') { + this._filterText = (event.target as HTMLInputElement).value + } else if (event.key === 'Escape') { + this.resetFilter() + } + } }