diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 5b92364d2..f935b7701 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -27,6 +27,9 @@ import { PageHeaderComponent } from './components/common/page-header/page-header import { AppFrameComponent } from './components/app-frame/app-frame.component'; import { ToastsComponent } from './components/common/toasts/toasts.component'; import { FilterEditorComponent } from './components/filter-editor/filter-editor.component'; +import { FilterDropdownComponent } from './components/filter-editor/filter-dropdown/filter-dropdown.component'; +import { FilterDropdownButtonComponent } from './components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component'; +import { FilterDropdownDateComponent } from './components/filter-editor/filter-dropdown-date/filter-dropdown-date.component'; import { DocumentCardLargeComponent } from './components/document-list/document-card-large/document-card-large.component'; import { DocumentCardSmallComponent } from './components/document-list/document-card-small/document-card-small.component'; import { NgxFileDropModule } from 'ngx-file-drop'; @@ -48,6 +51,7 @@ import { PdfViewerModule } from 'ng2-pdf-viewer'; import { WelcomeWidgetComponent } from './components/dashboard/widgets/welcome-widget/welcome-widget.component'; import { YesNoPipe } from './pipes/yes-no.pipe'; import { FileSizePipe } from './pipes/file-size.pipe'; +import { FilterPipe } from './pipes/filter.pipe'; import { DocumentTitlePipe } from './pipes/document-title.pipe'; import { MetadataCollapseComponent } from './components/document-detail/metadata-collapse/metadata-collapse.component'; @@ -74,6 +78,9 @@ import { MetadataCollapseComponent } from './components/document-detail/metadata AppFrameComponent, ToastsComponent, FilterEditorComponent, + FilterDropdownComponent, + FilterDropdownButtonComponent, + FilterDropdownDateComponent, DocumentCardLargeComponent, DocumentCardSmallComponent, TextComponent, @@ -90,6 +97,7 @@ import { MetadataCollapseComponent } from './components/document-detail/metadata WelcomeWidgetComponent, YesNoPipe, FileSizePipe, + FilterPipe, DocumentTitlePipe, MetadataCollapseComponent ], @@ -110,7 +118,8 @@ import { MetadataCollapseComponent } from './components/document-detail/metadata provide: HTTP_INTERCEPTORS, useClass: CsrfInterceptor, multi: true - } + }, + FilterPipe ], bootstrap: [AppComponent] }) diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html index 8f3fced66..c2645db5e 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html @@ -7,7 +7,7 @@
-
+
{{(document.correspondent$ | async)?.name}} {{(document.correspondent$ | async)?.name}}: @@ -52,4 +52,4 @@
- \ No newline at end of file + diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html index 86e28442c..2647e702c 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html @@ -11,7 +11,7 @@ - +

@@ -44,7 +44,7 @@

{{document.created | date}} - + - - \ No newline at end of file + + diff --git a/src-ui/src/app/components/document-list/document-list.component.html b/src-ui/src/app/components/document-list/document-list.component.html index c4fa0d4d7..b1b2c8f94 100644 --- a/src-ui/src/app/components/document-list/document-list.component.html +++ b/src-ui/src/app/components/document-list/document-list.component.html @@ -1,5 +1,4 @@ -
+
@@ -42,36 +42,28 @@
+
- -
- -
+ -
-
-
Filter
- -
+
+
@@ -81,7 +73,7 @@
- +
@@ -101,16 +93,16 @@ - {{(d.correspondent$ | async)?.name}} + {{(d.correspondent$ | async)?.name}} {{d.title | documentTitle}} - + - {{(d.document_type$ | async)?.name}} + {{(d.document_type$ | async)?.name}} @@ -125,5 +117,5 @@
- +
diff --git a/src-ui/src/app/components/document-list/document-list.component.ts b/src-ui/src/app/components/document-list/document-list.component.ts index 09e73dd96..8d090f001 100644 --- a/src-ui/src/app/components/document-list/document-list.component.ts +++ b/src-ui/src/app/components/document-list/document-list.component.ts @@ -1,4 +1,4 @@ -import { Component, OnInit } from '@angular/core'; +import { Component, OnInit, ViewChild } from '@angular/core'; import { Title } from '@angular/platform-browser'; import { ActivatedRoute } from '@angular/router'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; @@ -6,11 +6,16 @@ import { cloneFilterRules, FilterRule } from 'src/app/data/filter-rule'; import { FILTER_CORRESPONDENT, FILTER_DOCUMENT_TYPE, FILTER_HAS_TAG, FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type'; import { SavedViewConfig } from 'src/app/data/saved-view-config'; import { DocumentListViewService } from 'src/app/services/document-list-view.service'; +import { FilterEditorViewService } from 'src/app/services/filter-editor-view.service'; import { DOCUMENT_SORT_FIELDS } from 'src/app/services/rest/document.service'; import { SavedViewConfigService } from 'src/app/services/saved-view-config.service'; import { Toast, ToastService } from 'src/app/services/toast.service'; import { environment } from 'src/environments/environment'; import { SaveViewConfigDialogComponent } from './save-view-config-dialog/save-view-config-dialog.component'; +import { FilterEditorComponent } from 'src/app/components/filter-editor/filter-editor.component'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; +import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; +import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; @Component({ selector: 'app-document-list', @@ -22,6 +27,7 @@ export class DocumentListComponent implements OnInit { constructor( public list: DocumentListViewService, public savedViewConfigService: SavedViewConfigService, + public filterEditorService: FilterEditorViewService, public route: ActivatedRoute, private toastService: ToastService, public modalService: NgbModal, @@ -29,13 +35,18 @@ export class DocumentListComponent implements OnInit { displayMode = 'smallCards' // largeCards, smallCards, details - filterRules: FilterRule[] = [] - showFilter = false - get isFiltered() { return this.list.filterRules?.length > 0 } + set filterRules(filterRules: FilterRule[]) { + this.filterEditorService.filterRules = filterRules + } + + get filterRules(): FilterRule[] { + return this.filterEditorService.filterRules + } + getTitle() { return this.list.savedViewTitle || "Documents" } @@ -55,31 +66,29 @@ export class DocumentListComponent implements OnInit { this.route.paramMap.subscribe(params => { if (params.has('id')) { this.list.savedView = this.savedViewConfigService.getConfig(params.get('id')) - this.filterRules = this.list.filterRules - this.showFilter = false + this.filterEditorService.filterRules = this.list.filterRules this.titleService.setTitle(`${this.list.savedView.title} - ${environment.appTitle}`) } else { this.list.savedView = null - this.filterRules = this.list.filterRules - this.showFilter = this.filterRules.length > 0 + this.filterEditorService.filterRules = this.list.filterRules this.titleService.setTitle(`Documents - ${environment.appTitle}`) } this.list.clear() this.list.reload() }) + this.filterEditorService.filterRules = this.list.filterRules } applyFilterRules() { - this.list.filterRules = this.filterRules + this.list.filterRules = this.filterEditorService.filterRules } clearFilterRules() { - this.list.filterRules = this.filterRules - this.showFilter = false + this.list.filterRules = this.filterEditorService.filterRules } loadViewConfig(config: SavedViewConfig) { - this.filterRules = cloneFilterRules(config.filterRules) + this.filterEditorService.filterRules = cloneFilterRules(config.filterRules) this.list.load(config) } @@ -103,42 +112,18 @@ export class DocumentListComponent implements OnInit { }) } - filterByTag(tag_id: number) { - let filterRules = this.list.filterRules - if (filterRules.find(rule => rule.type.id == FILTER_HAS_TAG && rule.value == tag_id)) { - return - } - - filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_HAS_TAG), value: tag_id}) - this.filterRules = filterRules + clickTag(tagID: number) { + this.filterEditorService.toggleFilterByTag(tagID) this.applyFilterRules() } - filterByCorrespondent(correspondent_id: number) { - let filterRules = this.list.filterRules - let existing_rule = filterRules.find(rule => rule.type.id == FILTER_CORRESPONDENT) - if (existing_rule && existing_rule.value == correspondent_id) { - return - } else if (existing_rule) { - existing_rule.value = correspondent_id - } else { - filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_CORRESPONDENT), value: correspondent_id}) - } - this.filterRules = filterRules + clickCorrespondent(correspondentID: number) { + this.filterEditorService.toggleFilterByCorrespondent(correspondentID) this.applyFilterRules() } - filterByDocumentType(document_type_id: number) { - let filterRules = this.list.filterRules - let existing_rule = filterRules.find(rule => rule.type.id == FILTER_DOCUMENT_TYPE) - if (existing_rule && existing_rule.value == document_type_id) { - return - } else if (existing_rule) { - existing_rule.value = document_type_id - } else { - filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_DOCUMENT_TYPE), value: document_type_id}) - } - this.filterRules = filterRules + clickDocumentType(documentTypeID: number) { + this.filterEditorService.toggleFilterByDocumentType(documentTypeID) this.applyFilterRules() } diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.html b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.html new file mode 100644 index 000000000..c4befd701 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.html @@ -0,0 +1,63 @@ +
+ + +
diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.scss b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.scss new file mode 100644 index 000000000..3bdedd8a0 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.scss @@ -0,0 +1,7 @@ +.date-filter { + min-width: 250px; + + .btn-link { + line-height: 1; + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.spec.ts b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.spec.ts new file mode 100644 index 000000000..6bf59e2e7 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { FilterDropdownDateComponent } from './filter-dropdown-date.component'; + +describe('FilterDropdownDateComponent', () => { + let component: FilterDropdownDateComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ FilterDropdownDateComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(FilterDropdownDateComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.ts b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.ts new file mode 100644 index 000000000..f69028bf0 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown-date/filter-dropdown-date.component.ts @@ -0,0 +1,108 @@ +import { Component, EventEmitter, Input, Output, ElementRef, ViewChild, OnChanges, SimpleChange } from '@angular/core'; +import { FilterRule } from 'src/app/data/filter-rule'; +import { ObjectWithId } from 'src/app/data/object-with-id'; +import { NgbDate, NgbDateStruct, NgbDatepicker } from '@ng-bootstrap/ng-bootstrap'; + +@Component({ + selector: 'app-filter-dropdown-date', + templateUrl: './filter-dropdown-date.component.html', + styleUrls: ['./filter-dropdown-date.component.scss'] +}) +export class FilterDropdownDateComponent { + + @Input() + dateBefore: NgbDateStruct + + @Input() + dateAfter: NgbDateStruct + + @Input() + title: string + + @Output() + dateBeforeSet = new EventEmitter() + + @Output() + dateAfterSet = new EventEmitter() + + @ViewChild('dpAfter') dpAfter: NgbDatepicker + @ViewChild('dpBefore') dpBefore: NgbDatepicker + + _dateBefore: NgbDateStruct + _dateAfter: NgbDateStruct + + get _maxDate(): NgbDate { + let date = new Date() + return NgbDate.from({year: date.getFullYear(), month: date.getMonth() + 1, day: date.getDate()}) + } + + isStringRange(range: any) { + return typeof range == 'string' + } + + ngOnChanges(changes: SimpleChange) { + // this is a hacky workaround perhaps because of https://github.com/angular/angular/issues/11097 + let dateString: string = '' + let dateAfterChange: SimpleChange + let dateBeforeChange: SimpleChange + if (changes) { + dateAfterChange = changes['dateAfter'] + dateBeforeChange = changes['dateBefore'] + } + + if (this.dpBefore && this.dpAfter) { + let dpAfterElRef: ElementRef = this.dpAfter['_elRef'] + let dpBeforeElRef: ElementRef = this.dpBefore['_elRef'] + + if (dateAfterChange && dateAfterChange.currentValue) { + let dateAfterDate = dateAfterChange.currentValue as NgbDateStruct + dateString = `${dateAfterDate.year}-${dateAfterDate.month.toString().padStart(2,'0')}-${dateAfterDate.day.toString().padStart(2,'0')}` + dpAfterElRef.nativeElement.value = dateString + } else if (dateBeforeChange && dateBeforeChange.currentValue) { + let dateBeforeDate = dateBeforeChange.currentValue as NgbDateStruct + dateString = `${dateBeforeChange.currentValue.year}-${dateBeforeChange.currentValue.month.toString().padStart(2,'0')}-${dateBeforeChange.currentValue.day.toString().padStart(2,'0')}` + dpBeforeElRef.nativeElement.value = dateString + } else { + dpAfterElRef.nativeElement.value = dateString + dpBeforeElRef.nativeElement.value = dateString + } + } + } + + setDateQuickFilter(range: any) { + this._dateAfter = this._dateBefore = undefined + let date = new Date() + let newDate: NgbDateStruct = { year: date.getFullYear(), month: date.getMonth() + 1, day: date.getDate() } + switch (typeof range) { + case 'number': + date.setDate(date.getDate() - range) + newDate.year = date.getFullYear() + newDate.month = date.getMonth() + 1 + newDate.day = date.getDate() + break + + case 'string': + newDate.day = 1 + if (range == 'year') newDate.month = 1 + break + + default: + break + } + this._dateAfter = newDate + this.onDateSelected(this._dateAfter) + } + + onDateSelected(date:NgbDateStruct) { + let emitter = this._dateAfter && NgbDate.from(this._dateAfter).equals(date) ? this.dateAfterSet : this.dateBeforeSet + emitter.emit(date) + } + + clearAfter() { + this.dateAfterSet.next() + } + + clearBefore() { + this.dateBeforeSet.next() + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.html b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.html new file mode 100644 index 000000000..8dff12a33 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.html @@ -0,0 +1,12 @@ + diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.scss b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.scss new file mode 100644 index 000000000..41fc6acc4 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.scss @@ -0,0 +1,4 @@ +.selected-icon { + min-width: 1em; + min-height: 1em; +} diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.spec.ts b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.spec.ts new file mode 100644 index 000000000..5cf1fefa2 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { FilterDropodownButtonComponent } from './filter-dropdown-button.component'; + +describe('FilterDropodownButtonComponent', () => { + let component: FilterDropodownButtonComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ FilterDropodownButtonComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(FilterDropodownButtonComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.ts b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.ts new file mode 100644 index 000000000..d3ddd3cbf --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown-button/filter-dropdown-button.component.ts @@ -0,0 +1,32 @@ +import { Component, EventEmitter, Input, Output, OnInit } from '@angular/core'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; +import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; +import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; + +@Component({ + selector: 'app-filter-dropdown-button', + templateUrl: './filter-dropdown-button.component.html', + styleUrls: ['./filter-dropdown-button.component.scss'] +}) +export class FilterDropdownButtonComponent implements OnInit { + + @Input() + item: PaperlessTag | PaperlessDocumentType | PaperlessCorrespondent + + @Input() + selected: boolean + + @Output() + toggle = new EventEmitter() + + isTag: boolean + + ngOnInit() { + this.isTag = 'is_inbox_tag' in this.item // ~ this.item instanceof PaperlessTag + } + + toggleItem(): void { + this.selected = !this.selected + this.toggle.emit(this.item) + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.html b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.html new file mode 100644 index 000000000..0a3fe5496 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.html @@ -0,0 +1,32 @@ +
+ + +
diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.scss b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.scss new file mode 100644 index 000000000..d34729eee --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.scss @@ -0,0 +1,8 @@ +.dropdown-menu { + min-width: 250px; + + .items { + max-height: 400px; + overflow-y: scroll; + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.spec.ts b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.spec.ts new file mode 100644 index 000000000..29edd7c45 --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { FilterDropodownComponent } from './filter-dropdown.component'; + +describe('FilterDropodownComponent', () => { + let component: FilterDropodownComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ FilterDropodownComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(FilterDropodownComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.ts b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.ts new file mode 100644 index 000000000..a24e7347d --- /dev/null +++ b/src-ui/src/app/components/filter-editor/filter-dropdown/filter-dropdown.component.ts @@ -0,0 +1,60 @@ +import { Component, EventEmitter, Input, Output, ElementRef, ViewChild } from '@angular/core'; +import { Observable } from 'rxjs'; +import { Results } from 'src/app/data/results'; +import { ObjectWithId } from 'src/app/data/object-with-id'; +import { FilterPipe } from 'src/app/pipes/filter.pipe'; +import { NgbDropdown } from '@ng-bootstrap/ng-bootstrap' + +@Component({ + selector: 'app-filter-dropdown', + templateUrl: './filter-dropdown.component.html', + styleUrls: ['./filter-dropdown.component.scss'] +}) +export class FilterDropdownComponent { + + constructor(private filterPipe: FilterPipe) { } + + @Input() + items: ObjectWithId[] + + @Input() + itemsSelected: ObjectWithId[] + + @Input() + title: string + + @Input() + display: string + + @Output() + toggle = new EventEmitter() + + @ViewChild('listFilterTextInput') listFilterTextInput: ElementRef + @ViewChild('filterDropdown') filterDropdown: NgbDropdown + + filterText: string + + toggleItem(item: ObjectWithId): void { + this.toggle.emit(item) + } + + isItemSelected(item: ObjectWithId): boolean { + return this.itemsSelected?.find(i => i.id == item.id) !== undefined + } + + dropdownOpenChange(open: boolean): void { + if (open) { + setTimeout(() => { + this.listFilterTextInput.nativeElement.focus(); + }, 0); + } else { + this.filterText = '' + } + } + + listFilterEnter(): void { + let filtered = this.filterPipe.transform(this.items, this.filterText) + if (filtered.length == 1) this.toggleItem(filtered.shift()) + this.filterDropdown.close() + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.html b/src-ui/src/app/components/filter-editor/filter-editor.component.html index 48780e950..eb322414d 100644 --- a/src-ui/src/app/components/filter-editor/filter-editor.component.html +++ b/src-ui/src/app/components/filter-editor/filter-editor.component.html @@ -1,52 +1,22 @@ -
-
- {{rule.type.name}} +
+
+
Filter by:
- - - - - - - - - - - - -
-
- -
-
- -
-
- -
-
- -
-
- -
-
- +
+ + + + + + + + +
diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.scss b/src-ui/src/app/components/filter-editor/filter-editor.component.scss index e69de29bb..05df7b213 100644 --- a/src-ui/src/app/components/filter-editor/filter-editor.component.scss +++ b/src-ui/src/app/components/filter-editor/filter-editor.component.scss @@ -0,0 +1,10 @@ +.quick-filter { + min-width: 250px; + max-height: 400px; + overflow-y: scroll; + + .selected-icon { + min-width: 1em; + min-height: 1em; + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.ts b/src-ui/src/app/components/filter-editor/filter-editor.component.ts index b04127287..a6940795e 100644 --- a/src-ui/src/app/components/filter-editor/filter-editor.component.ts +++ b/src-ui/src/app/components/filter-editor/filter-editor.component.ts @@ -1,67 +1,88 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; -import { FilterRule } from 'src/app/data/filter-rule'; -import { FilterRuleType, FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type'; +import { Component, EventEmitter, Input, Output, ElementRef, AfterViewInit, QueryList, ViewChild, ViewChildren } from '@angular/core'; +import { AbstractPaperlessService } from 'src/app/services/rest/abstract-paperless-service'; +import { ObjectWithId } from 'src/app/data/object-with-id'; +import { FilterEditorViewService } from 'src/app/services/filter-editor-view.service' +import { PaperlessTag } from 'src/app/data/paperless-tag'; import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; -import { PaperlessTag } from 'src/app/data/paperless-tag'; -import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; -import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; -import { TagService } from 'src/app/services/rest/tag.service'; - +import { FilterDropdownComponent } from './filter-dropdown/filter-dropdown.component' +import { FilterDropdownDateComponent } from './filter-dropdown-date/filter-dropdown-date.component' +import { fromEvent } from 'rxjs'; +import { debounceTime, distinctUntilChanged, tap } from 'rxjs/operators'; +import { NgbDateStruct } from '@ng-bootstrap/ng-bootstrap'; @Component({ selector: 'app-filter-editor', templateUrl: './filter-editor.component.html', styleUrls: ['./filter-editor.component.scss'] }) -export class FilterEditorComponent implements OnInit { +export class FilterEditorComponent implements AfterViewInit { - constructor(private documentTypeService: DocumentTypeService, private tagService: TagService, private correspondentService: CorrespondentService) { } + constructor() { } + + @Input() + filterEditorService: FilterEditorViewService @Output() clear = new EventEmitter() - @Input() - filterRules: FilterRule[] = [] - @Output() apply = new EventEmitter() - selectedRuleType: FilterRuleType = FILTER_RULE_TYPES[0] + @ViewChild('filterTextInput') filterTextInput: ElementRef; - correspondents: PaperlessCorrespondent[] = [] - tags: PaperlessTag[] = [] - documentTypes: PaperlessDocumentType[] = [] - - newRuleClicked() { - this.filterRules.push({type: this.selectedRuleType, value: this.selectedRuleType.default}) - this.selectedRuleType = this.getRuleTypes().length > 0 ? this.getRuleTypes()[0] : null + ngAfterViewInit() { + fromEvent(this.filterTextInput.nativeElement,'keyup').pipe( + debounceTime(150), + distinctUntilChanged(), + tap() + ).subscribe((event: Event) => { + this.filterEditorService.filterText = (event.target as HTMLInputElement).value + this.applyFilters() + }) } - removeRuleClicked(rule) { - let index = this.filterRules.findIndex(r => r == rule) - if (index > -1) { - this.filterRules.splice(index, 1) - } - } - - applyClicked() { + applyFilters() { this.apply.next() } - clearClicked() { - this.filterRules.splice(0,this.filterRules.length) + clearSelected() { + this.filterEditorService.clear() this.clear.next() } - ngOnInit(): void { - this.correspondentService.listAll().subscribe(result => {this.correspondents = result.results}) - this.tagService.listAll().subscribe(result => this.tags = result.results) - this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results) + onToggleTag(tag: PaperlessTag) { + this.filterEditorService.toggleFilterByTag(tag) + this.applyFilters() } - getRuleTypes() { - return FILTER_RULE_TYPES.filter(rt => rt.multi || !this.filterRules.find(r => r.type == rt)) + onToggleCorrespondent(correspondent: PaperlessCorrespondent) { + this.filterEditorService.toggleFilterByCorrespondent(correspondent) + this.applyFilters() } + onToggleDocumentType(documentType: PaperlessDocumentType) { + this.filterEditorService.toggleFilterByDocumentType(documentType) + this.applyFilters() + } + + onDateCreatedBeforeSet(date: NgbDateStruct) { + this.filterEditorService.setDateCreatedBefore(date) + this.applyFilters() + } + + onDateCreatedAfterSet(date: NgbDateStruct) { + this.filterEditorService.setDateCreatedAfter(date) + this.applyFilters() + } + + onDateAddedBeforeSet(date: NgbDateStruct) { + this.filterEditorService.setDateAddedBefore(date) + this.applyFilters() + } + + onDateAddedAfterSet(date: NgbDateStruct) { + this.filterEditorService.setDateAddedAfter(date) + this.applyFilters() + } } diff --git a/src-ui/src/app/data/filter-rule-type.ts b/src-ui/src/app/data/filter-rule-type.ts index a35759f69..ea8e60eee 100644 --- a/src-ui/src/app/data/filter-rule-type.ts +++ b/src-ui/src/app/data/filter-rule-type.ts @@ -22,15 +22,15 @@ export const FILTER_RULE_TYPES: FilterRuleType[] = [ {id: FILTER_TITLE, name: "Title contains", filtervar: "title__icontains", datatype: "string", multi: false, default: ""}, {id: FILTER_CONTENT, name: "Content contains", filtervar: "content__icontains", datatype: "string", multi: false, default: ""}, - + {id: FILTER_ASN, name: "ASN is", filtervar: "archive_serial_number", datatype: "number", multi: false}, - + {id: FILTER_CORRESPONDENT, name: "Correspondent is", filtervar: "correspondent__id", datatype: "correspondent", multi: false}, {id: FILTER_DOCUMENT_TYPE, name: "Document type is", filtervar: "document_type__id", datatype: "document_type", multi: false}, - {id: FILTER_IS_IN_INBOX, name: "Is in Inbox", filtervar: "is_in_inbox", datatype: "boolean", multi: false, default: true}, - {id: FILTER_HAS_TAG, name: "Has tag", filtervar: "tags__id__all", datatype: "tag", multi: true}, - {id: FILTER_DOES_NOT_HAVE_TAG, name: "Does not have tag", filtervar: "tags__id__none", datatype: "tag", multi: true}, + {id: FILTER_IS_IN_INBOX, name: "Is in Inbox", filtervar: "is_in_inbox", datatype: "boolean", multi: false, default: true}, + {id: FILTER_HAS_TAG, name: "Has tag", filtervar: "tags__id__all", datatype: "tag", multi: true}, + {id: FILTER_DOES_NOT_HAVE_TAG, name: "Does not have tag", filtervar: "tags__id__none", datatype: "tag", multi: true}, {id: FILTER_HAS_ANY_TAG, name: "Has any tag", filtervar: "is_tagged", datatype: "boolean", multi: false, default: true}, {id: FILTER_CREATED_BEFORE, name: "Created before", filtervar: "created__date__lt", datatype: "date", multi: false}, @@ -42,7 +42,7 @@ export const FILTER_RULE_TYPES: FilterRuleType[] = [ {id: FILTER_ADDED_BEFORE, name: "Added before", filtervar: "added__date__lt", datatype: "date", multi: false}, {id: FILTER_ADDED_AFTER, name: "Added after", filtervar: "added__date__gt", datatype: "date", multi: false}, - + {id: FILTER_MODIFIED_BEFORE, name: "Modified before", filtervar: "modified__date__lt", datatype: "date", multi: false}, {id: FILTER_MODIFIED_AFTER, name: "Modified after", filtervar: "modified__date__gt", datatype: "date", multi: false}, ] @@ -54,4 +54,4 @@ export interface FilterRuleType { datatype: string //number, string, boolean, date multi: boolean default?: any -} \ No newline at end of file +} diff --git a/src-ui/src/app/pipes/filter.pipe.ts b/src-ui/src/app/pipes/filter.pipe.ts new file mode 100644 index 000000000..f799f40cc --- /dev/null +++ b/src-ui/src/app/pipes/filter.pipe.ts @@ -0,0 +1,17 @@ +import { Pipe, PipeTransform } from '@angular/core'; + +@Pipe({ + name: 'filter' +}) +export class FilterPipe implements PipeTransform { + transform(items: any[], searchText: string): any[] { + if (!items) return []; + if (!searchText) return items; + + return items.filter(item => { + return Object.keys(item).some(key => { + return String(item[key]).toLowerCase().includes(searchText.toLowerCase()); + }); + }); + } +} diff --git a/src-ui/src/app/services/document-list-view.service.ts b/src-ui/src/app/services/document-list-view.service.ts index 811ac3c4b..8692ed1c0 100644 --- a/src-ui/src/app/services/document-list-view.service.ts +++ b/src-ui/src/app/services/document-list-view.service.ts @@ -9,7 +9,7 @@ import { DocumentService } from './rest/document.service'; /** * This service manages the document list which is displayed using the document list view. - * + * * This service also serves saved views by transparently switching between the document list * and saved views on request. See below. */ @@ -25,7 +25,7 @@ export class DocumentListViewService { currentPage = 1 currentPageSize: number = +localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT collectionSize: number - + /** * This is the current config for the document list. The service will always remember the last settings used for the document list. */ @@ -192,7 +192,7 @@ export class DocumentListViewService { } } - constructor(private documentService: DocumentService) { + constructor(private documentService: DocumentService) { let documentListViewConfigJson = sessionStorage.getItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) if (documentListViewConfigJson) { try { diff --git a/src-ui/src/app/services/filter-editor-view.service.spec.ts b/src-ui/src/app/services/filter-editor-view.service.spec.ts new file mode 100644 index 000000000..8051bcf0d --- /dev/null +++ b/src-ui/src/app/services/filter-editor-view.service.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { FilterEditorViewService } from './filter-editor-view.service'; + +describe('FilterEditorViewService', () => { + let service: FilterEditorViewService; + + beforeEach(() => { + TestBed.configureTestingModule({}); + service = TestBed.inject(FilterEditorViewService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/services/filter-editor-view.service.ts b/src-ui/src/app/services/filter-editor-view.service.ts new file mode 100644 index 000000000..89f40189c --- /dev/null +++ b/src-ui/src/app/services/filter-editor-view.service.ts @@ -0,0 +1,189 @@ +import { Injectable } from '@angular/core'; +import { Observable } from 'rxjs'; +import { TagService } from 'src/app/services/rest/tag.service'; +import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; +import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; +import { ObjectWithId } from 'src/app/data/object-with-id'; +import { FilterRule } from 'src/app/data/filter-rule'; +import { FilterRuleType, FILTER_RULE_TYPES, FILTER_CORRESPONDENT, FILTER_DOCUMENT_TYPE, FILTER_HAS_TAG, FILTER_TITLE, FILTER_ADDED_BEFORE, FILTER_ADDED_AFTER, FILTER_CREATED_BEFORE, FILTER_CREATED_AFTER, FILTER_CREATED_YEAR, FILTER_CREATED_MONTH, FILTER_CREATED_DAY } from 'src/app/data/filter-rule-type'; +import { Results } from 'src/app/data/results' +import { PaperlessTag } from 'src/app/data/paperless-tag'; +import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; +import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; +import { NgbDate, NgbDateStruct } from '@ng-bootstrap/ng-bootstrap'; + +@Injectable({ + providedIn: 'root' +}) +export class FilterEditorViewService { + private tags$: Observable> + private correspondents$: Observable> + private documentTypes$: Observable> + + tags: PaperlessTag[] = [] + correspondents: PaperlessCorrespondent[] + documentTypes: PaperlessDocumentType[] = [] + + filterRules: FilterRule[] = [] + + constructor(private tagService: TagService, private documentTypeService: DocumentTypeService, private correspondentService: CorrespondentService) { + this.tags$ = this.tagService.listAll() + this.tags$.subscribe(result => this.tags = result.results) + this.correspondents$ = this.correspondentService.listAll() + this.correspondents$.subscribe(result => this.correspondents = result.results) + this.documentTypes$ = this.documentTypeService.listAll() + this.documentTypes$.subscribe(result => this.documentTypes = result.results) + } + + clear() { + this.filterRules = [] + } + + hasFilters() { + return this.filterRules.length > 0 + } + + set filterText(text: string) { + let filterRules = this.filterRules + let existingRule = filterRules.find(rule => rule.type.id == FILTER_TITLE) + if (existingRule && existingRule.value == text) { + return + } else if (existingRule) { + existingRule.value = text + } else { + filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_TITLE), value: text}) + } + this.filterRules = filterRules + } + + get filterText(): string { + let existingRule = this.filterRules.find(rule => rule.type.id == FILTER_TITLE) + return existingRule ? existingRule.value : '' + } + + get selectedTags(): PaperlessTag[] { + let tagRules: FilterRule[] = this.filterRules.filter(fr => fr.type.id == FILTER_HAS_TAG) + return this.tags?.filter(t => tagRules.find(tr => tr.value == t.id)) + } + + get selectedCorrespondents(): PaperlessCorrespondent[] { + let correspondentRules: FilterRule[] = this.filterRules.filter(fr => fr.type.id == FILTER_CORRESPONDENT) + return this.correspondents?.filter(c => correspondentRules.find(cr => cr.value == c.id)) + } + + get selectedDocumentTypes(): PaperlessDocumentType[] { + let documentTypeRules: FilterRule[] = this.filterRules.filter(fr => fr.type.id == FILTER_DOCUMENT_TYPE) + return this.documentTypes?.filter(dt => documentTypeRules.find(dtr => dtr.value == dt.id)) + } + + toggleFilterByTag(tag: PaperlessTag | number) { + if (typeof tag == 'number') tag = this.tags?.find(t => t.id == tag) + this.toggleFilterByItem(tag, FILTER_HAS_TAG) + } + + toggleFilterByCorrespondent(correspondent: PaperlessCorrespondent | number) { + if (typeof correspondent == 'number') correspondent = this.correspondents?.find(t => t.id == correspondent) + this.toggleFilterByItem(correspondent, FILTER_CORRESPONDENT) + } + + toggleFilterByDocumentType(documentType: PaperlessDocumentType | number) { + if (typeof documentType == 'number') documentType = this.documentTypes?.find(t => t.id == documentType) + this.toggleFilterByItem(documentType, FILTER_DOCUMENT_TYPE) + } + + private toggleFilterByItem(item: ObjectWithId, filterRuleTypeID: number) { + let filterRules = this.filterRules + let filterRuleType: FilterRuleType = FILTER_RULE_TYPES.find(t => t.id == filterRuleTypeID) + let existingRules = filterRules.filter(rule => rule.type.id == filterRuleType.id) + let existingItemRule = existingRules?.find(rule => rule.value == item.id) + + if (existingRules && existingItemRule) { // if exact rule exists just remove + filterRules.splice(filterRules.indexOf(existingItemRule), 1) + } else if (existingRules.length > 0 && filterRuleType.multi) { // e.g. tags can have multiple + filterRules.push({type: filterRuleType, value: item.id}) + } else if (existingRules.length > 0) { // correspondents & documentTypes can only be one + filterRules.find(rule => rule.type.id == filterRuleType.id).value = item.id + } else { + filterRules.push({type: filterRuleType, value: item.id}) + } + + this.filterRules = filterRules + } + + get dateCreatedBefore(): NgbDateStruct { + let createdBeforeRule: FilterRule = this.filterRules.find(fr => fr.type.id == FILTER_CREATED_BEFORE) + return createdBeforeRule ? { + year: createdBeforeRule.value.substring(0,4), + month: createdBeforeRule.value.substring(5,7), + day: createdBeforeRule.value.substring(8,10) + } : undefined + } + + get dateCreatedAfter(): NgbDateStruct { + let createdAfterRule: FilterRule = this.filterRules.find(fr => fr.type.id == FILTER_CREATED_AFTER) + return createdAfterRule ? { + year: createdAfterRule.value.substring(0,4), + month: createdAfterRule.value.substring(5,7), + day: createdAfterRule.value.substring(8,10) + } : undefined + } + + get dateAddedBefore(): NgbDateStruct { + let addedBeforeRule: FilterRule = this.filterRules.find(fr => fr.type.id == FILTER_ADDED_BEFORE) + return addedBeforeRule ? { + year: addedBeforeRule.value.substring(0,4), + month: addedBeforeRule.value.substring(5,7), + day: addedBeforeRule.value.substring(8,10) + } : undefined + } + + get dateAddedAfter(): NgbDateStruct { + let addedAfterRule: FilterRule = this.filterRules.find(fr => fr.type.id == FILTER_ADDED_AFTER) + return addedAfterRule ? { + year: addedAfterRule.value.substring(0,4), + month: addedAfterRule.value.substring(5,7), + day: addedAfterRule.value.substring(8,10) + } : undefined + } + + setDateCreatedBefore(date?: NgbDateStruct) { + if (date) this.setDateFilter(date, FILTER_CREATED_BEFORE) + else this.clearDateFilter(FILTER_CREATED_BEFORE) + } + + setDateCreatedAfter(date?: NgbDateStruct) { + if (date) this.setDateFilter(date, FILTER_CREATED_AFTER) + else this.clearDateFilter(FILTER_CREATED_AFTER) + } + + setDateAddedBefore(date?: NgbDateStruct) { + if (date) this.setDateFilter(date, FILTER_ADDED_BEFORE) + else this.clearDateFilter(FILTER_ADDED_BEFORE) + } + + setDateAddedAfter(date?: NgbDateStruct) { + if (date) this.setDateFilter(date, FILTER_ADDED_AFTER) + else this.clearDateFilter(FILTER_ADDED_AFTER) + } + + setDateFilter(date: NgbDateStruct, dateRuleTypeID: number) { + let filterRules = this.filterRules + let existingRule = filterRules.find(rule => rule.type.id == dateRuleTypeID) + let newValue = `${date.year}-${date.month.toString().padStart(2,'0')}-${date.day.toString().padStart(2,'0')}` // YYYY-MM-DD + + if (existingRule) { + existingRule.value = newValue + } else { + filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == dateRuleTypeID), value: newValue}) + } + + this.filterRules = filterRules + } + + clearDateFilter(dateRuleTypeID: number) { + let filterRules = this.filterRules + let existingRule = filterRules.find(rule => rule.type.id == dateRuleTypeID) + filterRules.splice(filterRules.indexOf(existingRule), 1) + this.filterRules = filterRules + } +}