From e60a7df9a2586cefb7532a3854503c51d946068e Mon Sep 17 00:00:00 2001 From: Michael Shamoon <4887959+shamoon@users.noreply.github.com> Date: Fri, 20 May 2022 15:16:17 -0700 Subject: [PATCH 01/11] Refactor query params service --- .../app-frame/app-frame.component.ts | 5 +- .../saved-view-widget.component.ts | 6 +- .../document-detail.component.ts | 6 +- .../document-list.component.html | 2 +- .../document-list/document-list.component.ts | 49 ++---- .../correspondent-list.component.ts | 6 +- .../document-type-list.component.ts | 6 +- .../management-list.component.ts | 6 +- .../storage-path-list.component.ts | 5 +- .../manage/tag-list/tag-list.component.ts | 6 +- .../services/document-list-view.service.ts | 98 +++++++---- .../src/app/services/query-params.service.ts | 163 ------------------ .../src/app/services/rest/document.service.ts | 4 +- src-ui/src/app/utils/query-params.ts | 100 +++++++++++ 14 files changed, 205 insertions(+), 257 deletions(-) delete mode 100644 src-ui/src/app/services/query-params.service.ts create mode 100644 src-ui/src/app/utils/query-params.ts diff --git a/src-ui/src/app/components/app-frame/app-frame.component.ts b/src-ui/src/app/components/app-frame/app-frame.component.ts index 675bfc920..0d43f17a2 100644 --- a/src-ui/src/app/components/app-frame/app-frame.component.ts +++ b/src-ui/src/app/components/app-frame/app-frame.component.ts @@ -22,7 +22,6 @@ import { RemoteVersionService, AppRemoteVersion, } from 'src/app/services/rest/remote-version.service' -import { QueryParamsService } from 'src/app/services/query-params.service' import { SettingsService } from 'src/app/services/settings.service' @Component({ @@ -38,7 +37,7 @@ export class AppFrameComponent { private searchService: SearchService, public savedViewService: SavedViewService, private remoteVersionService: RemoteVersionService, - private queryParamsService: QueryParamsService, + private list: DocumentListViewService, public settingsService: SettingsService ) { this.remoteVersionService @@ -94,7 +93,7 @@ export class AppFrameComponent { search() { this.closeMenu() - this.queryParamsService.navigateWithFilterRules([ + this.list.quickFilter([ { rule_type: FILTER_FULLTEXT_QUERY, value: (this.searchField.value as string).trim(), diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts index 94e0c4052..17a0d8c89 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts @@ -7,8 +7,8 @@ import { ConsumerStatusService } from 'src/app/services/consumer-status.service' import { DocumentService } from 'src/app/services/rest/document.service' import { PaperlessTag } from 'src/app/data/paperless-tag' import { FILTER_HAS_TAGS_ALL } from 'src/app/data/filter-rule-type' -import { QueryParamsService } from 'src/app/services/query-params.service' import { OpenDocumentsService } from 'src/app/services/open-documents.service' +import { DocumentListViewService } from 'src/app/services/document-list-view.service' @Component({ selector: 'app-saved-view-widget', @@ -21,7 +21,7 @@ export class SavedViewWidgetComponent implements OnInit, OnDestroy { constructor( private documentService: DocumentService, private router: Router, - private queryParamsService: QueryParamsService, + private list: DocumentListViewService, private consumerStatusService: ConsumerStatusService, public openDocumentsService: OpenDocumentsService ) {} @@ -73,7 +73,7 @@ export class SavedViewWidgetComponent implements OnInit, OnDestroy { } clickTag(tag: PaperlessTag) { - this.queryParamsService.navigateWithFilterRules([ + this.list.quickFilter([ { rule_type: FILTER_HAS_TAGS_ALL, value: tag.id.toString() }, ]) } diff --git a/src-ui/src/app/components/document-detail/document-detail.component.ts b/src-ui/src/app/components/document-detail/document-detail.component.ts index 6728f746d..c4255e9f7 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.ts +++ b/src-ui/src/app/components/document-detail/document-detail.component.ts @@ -32,7 +32,6 @@ import { import { PaperlessDocumentSuggestions } from 'src/app/data/paperless-document-suggestions' import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type' import { normalizeDateStr } from 'src/app/utils/date' -import { QueryParamsService } from 'src/app/services/query-params.service' import { StoragePathService } from 'src/app/services/rest/storage-path.service' import { PaperlessStoragePath } from 'src/app/data/paperless-storage-path' import { StoragePathEditDialogComponent } from '../common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component' @@ -120,8 +119,7 @@ export class DocumentDetailComponent private documentTitlePipe: DocumentTitlePipe, private toastService: ToastService, private settings: SettingsService, - private storagePathService: StoragePathService, - private queryParamsService: QueryParamsService + private storagePathService: StoragePathService ) {} titleKeyUp(event) { @@ -494,7 +492,7 @@ export class DocumentDetailComponent } moreLike() { - this.queryParamsService.navigateWithFilterRules([ + this.documentListViewService.quickFilter([ { rule_type: FILTER_FULLTEXT_MORELIKE, value: this.documentId.toString(), 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 e8cb78995..f812be217 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 @@ -93,7 +93,7 @@ {list.collectionSize, plural, =1 {One document} other {{{list.collectionSize || 0}} documents}} (filtered)

- 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 d9355902f..e27f14b14 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,5 +1,4 @@ import { - AfterViewInit, Component, OnDestroy, OnInit, @@ -21,7 +20,6 @@ import { import { ConsumerStatusService } from 'src/app/services/consumer-status.service' import { DocumentListViewService } from 'src/app/services/document-list-view.service' import { OpenDocumentsService } from 'src/app/services/open-documents.service' -import { QueryParamsService } from 'src/app/services/query-params.service' import { DOCUMENT_SORT_FIELDS, DOCUMENT_SORT_FIELDS_FULLTEXT, @@ -36,7 +34,7 @@ import { SaveViewConfigDialogComponent } from './save-view-config-dialog/save-vi templateUrl: './document-list.component.html', styleUrls: ['./document-list.component.scss'], }) -export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { +export class DocumentListComponent implements OnInit, OnDestroy { constructor( public list: DocumentListViewService, public savedViewService: SavedViewService, @@ -45,7 +43,6 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { private toastService: ToastService, private modalService: NgbModal, private consumerStatusService: ConsumerStatusService, - private queryParamsService: QueryParamsService, public openDocumentsService: OpenDocumentsService ) {} @@ -76,8 +73,6 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { set listSort(reverse: boolean) { this.list.sortReverse = reverse - this.queryParamsService.sortField = this.list.sortField - this.queryParamsService.sortReverse = reverse } get listSort(): boolean { @@ -86,14 +81,14 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { setSortField(field: string) { this.list.sortField = field - this.queryParamsService.sortField = field - this.queryParamsService.sortReverse = this.listSort } onSort(event: SortEvent) { this.list.setSort(event.column, event.reverse) - this.queryParamsService.sortField = event.column - this.queryParamsService.sortReverse = event.reverse + } + + setPage(page: number) { + this.list.currentPage = page } get isBulkEditing(): boolean { @@ -133,7 +128,6 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { } this.list.activateSavedView(view) this.list.reload() - this.queryParamsService.updateFromView(view) this.unmodifiedFilterRules = view.filter_rules }) @@ -148,22 +142,12 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { this.loadViewConfig(parseInt(queryParams.get('view'))) } else { this.list.activateSavedView(null) - this.queryParamsService.parseQueryParams(queryParams) + this.list.loadFromQueryParams(queryParams) this.unmodifiedFilterRules = [] } }) } - ngAfterViewInit(): void { - this.filterEditor.filterRulesChange - .pipe(takeUntil(this.unsubscribeNotifier)) - .subscribe({ - next: (filterRules) => { - this.queryParamsService.updateFilterRules(filterRules) - }, - }) - } - ngOnDestroy() { // unsubscribes all this.unsubscribeNotifier.next(this) @@ -175,9 +159,8 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { .getCached(viewId) .pipe(first()) .subscribe((view) => { - this.list.loadSavedView(view) + this.list.activateSavedView(view) this.list.reload() - this.queryParamsService.updateFromView(view) }) } @@ -246,34 +229,26 @@ export class DocumentListComponent implements OnInit, OnDestroy, AfterViewInit { clickTag(tagID: number) { this.list.selectNone() - setTimeout(() => { - this.filterEditor.addTag(tagID) - }) + this.filterEditor.addTag(tagID) } clickCorrespondent(correspondentID: number) { this.list.selectNone() - setTimeout(() => { - this.filterEditor.addCorrespondent(correspondentID) - }) + this.filterEditor.addCorrespondent(correspondentID) } clickDocumentType(documentTypeID: number) { this.list.selectNone() - setTimeout(() => { - this.filterEditor.addDocumentType(documentTypeID) - }) + this.filterEditor.addDocumentType(documentTypeID) } clickStoragePath(storagePathID: number) { this.list.selectNone() - setTimeout(() => { - this.filterEditor.addStoragePath(storagePathID) - }) + this.filterEditor.addStoragePath(storagePathID) } clickMoreLike(documentID: number) { - this.queryParamsService.navigateWithFilterRules([ + this.list.quickFilter([ { rule_type: FILTER_FULLTEXT_MORELIKE, value: documentID.toString() }, ]) } diff --git a/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts b/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts index 9eb05758c..983c36290 100644 --- a/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts +++ b/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts @@ -3,7 +3,7 @@ import { NgbModal } from '@ng-bootstrap/ng-bootstrap' import { FILTER_CORRESPONDENT } from 'src/app/data/filter-rule-type' import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent' import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe' -import { QueryParamsService } from 'src/app/services/query-params.service' +import { DocumentListViewService } from 'src/app/services/document-list-view.service' import { CorrespondentService } from 'src/app/services/rest/correspondent.service' import { ToastService } from 'src/app/services/toast.service' import { CorrespondentEditDialogComponent } from '../../common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component' @@ -20,7 +20,7 @@ export class CorrespondentListComponent extends ManagementListComponent private modalService: NgbModal, private editDialogComponent: any, private toastService: ToastService, - private queryParamsService: QueryParamsService, + private documentListViewService: DocumentListViewService, protected filterRuleType: number, public typeName: string, public typeNamePlural: string, @@ -141,7 +141,7 @@ export abstract class ManagementListComponent } filterDocuments(object: ObjectWithId) { - this.queryParamsService.navigateWithFilterRules([ + this.documentListViewService.quickFilter([ { rule_type: this.filterRuleType, value: object.id.toString() }, ]) } diff --git a/src-ui/src/app/components/manage/storage-path-list/storage-path-list.component.ts b/src-ui/src/app/components/manage/storage-path-list/storage-path-list.component.ts index dc363c4d5..1d7b726a0 100644 --- a/src-ui/src/app/components/manage/storage-path-list/storage-path-list.component.ts +++ b/src-ui/src/app/components/manage/storage-path-list/storage-path-list.component.ts @@ -3,7 +3,6 @@ import { NgbModal } from '@ng-bootstrap/ng-bootstrap' import { FILTER_STORAGE_PATH } from 'src/app/data/filter-rule-type' import { PaperlessStoragePath } from 'src/app/data/paperless-storage-path' import { DocumentListViewService } from 'src/app/services/document-list-view.service' -import { QueryParamsService } from 'src/app/services/query-params.service' import { StoragePathService } from 'src/app/services/rest/storage-path.service' import { ToastService } from 'src/app/services/toast.service' import { StoragePathEditDialogComponent } from '../../common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component' @@ -19,14 +18,14 @@ export class StoragePathListComponent extends ManagementListComponent { tagService: TagService, modalService: NgbModal, toastService: ToastService, - queryParamsService: QueryParamsService + documentListViewService: DocumentListViewService ) { super( tagService, modalService, TagEditDialogComponent, toastService, - queryParamsService, + documentListViewService, FILTER_HAS_TAGS_ALL, $localize`tag`, $localize`tags`, 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 471fc7944..52a0de296 100644 --- a/src-ui/src/app/services/document-list-view.service.ts +++ b/src-ui/src/app/services/document-list-view.service.ts @@ -10,13 +10,14 @@ import { PaperlessDocument } from '../data/paperless-document' import { PaperlessSavedView } from '../data/paperless-saved-view' import { SETTINGS_KEYS } from '../data/paperless-uisettings' import { DOCUMENT_LIST_SERVICE } from '../data/storage-keys' +import { generateParams, parseQueryParams } from '../utils/query-params' import { DocumentService, DOCUMENT_SORT_FIELDS } from './rest/document.service' import { SettingsService } from './settings.service' /** * Captures the current state of the list view. */ -interface ListViewState { +export interface ListViewState { /** * Title of the document list view. Either "Documents" (localized) or the name of a saved view. */ @@ -32,7 +33,7 @@ interface ListViewState { /** * Total amount of documents with the current filter rules. Used to calculate the number of pages. */ - collectionSize: number + collectionSize?: number /** * Currently selected sort field. @@ -85,6 +86,32 @@ export class DocumentListViewService { return this.activeListViewState.title } + constructor( + private documentService: DocumentService, + private settings: SettingsService, + private router: Router + ) { + let documentListViewConfigJson = localStorage.getItem( + DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG + ) + if (documentListViewConfigJson) { + try { + let savedState: ListViewState = JSON.parse(documentListViewConfigJson) + // Remove null elements from the restored state + Object.keys(savedState).forEach((k) => { + if (savedState[k] == null) { + delete savedState[k] + } + }) + //only use restored state attributes instead of defaults if they are not null + let newState = Object.assign(this.defaultListViewState(), savedState) + this.listViewStates.set(null, newState) + } catch (e) { + localStorage.removeItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) + } + } + } + private defaultListViewState(): ListViewState { return { title: null, @@ -122,20 +149,36 @@ export class DocumentListViewService { if (closeCurrentView) { this._activeSavedViewId = null } + this.activeListViewState.filterRules = cloneFilterRules(view.filter_rules) this.activeListViewState.sortField = view.sort_field this.activeListViewState.sortReverse = view.sort_reverse if (this._activeSavedViewId) { this.activeListViewState.title = view.name } + this.reduceSelectionToFilter() + + if (!this.router.routerState.snapshot.url.includes('/view/')) { + this.router.navigate([], { + queryParams: { view: view.id }, + }) + } } - reload(onFinish?) { + loadFromQueryParams(queryParams) { + let newState: ListViewState = parseQueryParams(queryParams) + this.activeListViewState.filterRules = newState.filterRules + this.activeListViewState.sortField = newState.sortField + this.activeListViewState.sortReverse = newState.sortReverse + this.activeListViewState.currentPage = newState.currentPage + this.reload(null, false) + } + + reload(onFinish?, updateQueryParams: boolean = true) { this.isReloading = true this.error = null let activeListViewState = this.activeListViewState - this.documentService .listFiltered( activeListViewState.currentPage, @@ -149,6 +192,19 @@ export class DocumentListViewService { this.isReloading = false activeListViewState.collectionSize = result.count activeListViewState.documents = result.results + + if (updateQueryParams && !this._activeSavedViewId) { + let base = ['/documents'] + this.router.navigate(base, { + queryParams: generateParams( + activeListViewState.filterRules, + activeListViewState.sortField, + activeListViewState.sortReverse, + activeListViewState.currentPage + ), + }) + } + if (onFinish) { onFinish() } @@ -191,6 +247,7 @@ export class DocumentListViewService { ) { this.activeListViewState.sortField = 'created' } + this._activeSavedViewId = null this.activeListViewState.filterRules = filterRules this.reload() this.reduceSelectionToFilter() @@ -202,6 +259,7 @@ export class DocumentListViewService { } set sortField(field: string) { + this._activeSavedViewId = null this.activeListViewState.sortField = field this.reload() this.saveDocumentListView() @@ -212,6 +270,7 @@ export class DocumentListViewService { } set sortReverse(reverse: boolean) { + this._activeSavedViewId = null this.activeListViewState.sortReverse = reverse this.reload() this.saveDocumentListView() @@ -237,6 +296,8 @@ export class DocumentListViewService { } set currentPage(page: number) { + if (this.activeListViewState.currentPage == page) return + this._activeSavedViewId = null this.activeListViewState.currentPage = page this.reload() this.saveDocumentListView() @@ -273,6 +334,10 @@ export class DocumentListViewService { } } + quickFilter(filterRules: FilterRule[]) { + this.filterRules = filterRules + } + getLastPage(): number { return Math.ceil(this.collectionSize / this.currentPageSize) } @@ -431,29 +496,4 @@ export class DocumentListViewService { documentIndexInCurrentView(documentID: number): number { return this.documents.map((d) => d.id).indexOf(documentID) } - - constructor( - private documentService: DocumentService, - private settings: SettingsService - ) { - let documentListViewConfigJson = localStorage.getItem( - DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG - ) - if (documentListViewConfigJson) { - try { - let savedState: ListViewState = JSON.parse(documentListViewConfigJson) - // Remove null elements from the restored state - Object.keys(savedState).forEach((k) => { - if (savedState[k] == null) { - delete savedState[k] - } - }) - //only use restored state attributes instead of defaults if they are not null - let newState = Object.assign(this.defaultListViewState(), savedState) - this.listViewStates.set(null, newState) - } catch (e) { - localStorage.removeItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) - } - } - } } diff --git a/src-ui/src/app/services/query-params.service.ts b/src-ui/src/app/services/query-params.service.ts deleted file mode 100644 index 888440aae..000000000 --- a/src-ui/src/app/services/query-params.service.ts +++ /dev/null @@ -1,163 +0,0 @@ -import { Injectable } from '@angular/core' -import { ParamMap, Params, Router } from '@angular/router' -import { FilterRule } from '../data/filter-rule' -import { FilterRuleType, FILTER_RULE_TYPES } from '../data/filter-rule-type' -import { PaperlessSavedView } from '../data/paperless-saved-view' -import { DocumentListViewService } from './document-list-view.service' - -const SORT_FIELD_PARAMETER = 'sort' -const SORT_REVERSE_PARAMETER = 'reverse' - -@Injectable({ - providedIn: 'root', -}) -export class QueryParamsService { - constructor(private router: Router, private list: DocumentListViewService) {} - - private filterParams: Params = {} - private sortParams: Params = {} - - updateFilterRules( - filterRules: FilterRule[], - updateQueryParams: boolean = true - ) { - this.filterParams = filterRulesToQueryParams(filterRules) - if (updateQueryParams) this.updateQueryParams() - } - - set sortField(field: string) { - this.sortParams[SORT_FIELD_PARAMETER] = field - this.updateQueryParams() - } - - set sortReverse(reverse: boolean) { - if (!reverse) this.sortParams[SORT_REVERSE_PARAMETER] = undefined - else this.sortParams[SORT_REVERSE_PARAMETER] = reverse - this.updateQueryParams() - } - - get params(): Params { - return { - ...this.sortParams, - ...this.filterParams, - } - } - - private updateQueryParams() { - // if we were on a saved view we navigate 'away' to /documents - let base = [] - if (this.router.routerState.snapshot.url.includes('/view/')) - base = ['/documents'] - - this.router.navigate(base, { - queryParams: this.params, - }) - } - - public parseQueryParams(queryParams: ParamMap) { - let filterRules = filterRulesFromQueryParams(queryParams) - if ( - filterRules.length || - queryParams.has(SORT_FIELD_PARAMETER) || - queryParams.has(SORT_REVERSE_PARAMETER) - ) { - this.list.filterRules = filterRules - this.list.sortField = queryParams.get(SORT_FIELD_PARAMETER) - this.list.sortReverse = - queryParams.has(SORT_REVERSE_PARAMETER) || - (!queryParams.has(SORT_FIELD_PARAMETER) && - !queryParams.has(SORT_REVERSE_PARAMETER)) - this.list.reload() - } else if ( - filterRules.length == 0 && - !queryParams.has(SORT_FIELD_PARAMETER) - ) { - // this is navigating to /documents so we need to update the params from the list - this.updateFilterRules(this.list.filterRules, false) - this.sortParams[SORT_FIELD_PARAMETER] = this.list.sortField - this.sortParams[SORT_REVERSE_PARAMETER] = this.list.sortReverse - this.router.navigate([], { - queryParams: this.params, - replaceUrl: true, - }) - } - } - - updateFromView(view: PaperlessSavedView) { - if (!this.router.routerState.snapshot.url.includes('/view/')) { - // navigation for /documents?view= - this.router.navigate([], { - queryParams: { view: view.id }, - }) - } - // make sure params are up-to-date - this.updateFilterRules(view.filter_rules, false) - this.sortParams[SORT_FIELD_PARAMETER] = this.list.sortField - this.sortParams[SORT_REVERSE_PARAMETER] = this.list.sortReverse - } - - navigateWithFilterRules(filterRules: FilterRule[]) { - this.updateFilterRules(filterRules) - this.router.navigate(['/documents'], { - queryParams: this.params, - }) - } -} - -export function filterRulesToQueryParams(filterRules: FilterRule[]): Object { - if (filterRules) { - let params = {} - for (let rule of filterRules) { - let ruleType = FILTER_RULE_TYPES.find((t) => t.id == rule.rule_type) - if (ruleType.multi) { - params[ruleType.filtervar] = params[ruleType.filtervar] - ? params[ruleType.filtervar] + ',' + rule.value - : rule.value - } else if (ruleType.isnull_filtervar && rule.value == null) { - params[ruleType.isnull_filtervar] = true - } else { - params[ruleType.filtervar] = rule.value - } - } - return params - } else { - return null - } -} - -export function filterRulesFromQueryParams(queryParams: ParamMap) { - const allFilterRuleQueryParams: string[] = FILTER_RULE_TYPES.map( - (rt) => rt.filtervar - ) - .concat(FILTER_RULE_TYPES.map((rt) => rt.isnull_filtervar)) - .filter((rt) => rt !== undefined) - - // transform query params to filter rules - let filterRulesFromQueryParams: FilterRule[] = [] - allFilterRuleQueryParams - .filter((frqp) => queryParams.has(frqp)) - .forEach((filterQueryParamName) => { - const rule_type: FilterRuleType = FILTER_RULE_TYPES.find( - (rt) => - rt.filtervar == filterQueryParamName || - rt.isnull_filtervar == filterQueryParamName - ) - const isNullRuleType = rule_type.isnull_filtervar == filterQueryParamName - const valueURIComponent: string = queryParams.get(filterQueryParamName) - const filterQueryParamValues: string[] = rule_type.multi - ? valueURIComponent.split(',') - : [valueURIComponent] - - filterRulesFromQueryParams = filterRulesFromQueryParams.concat( - // map all values to filter rules - filterQueryParamValues.map((val) => { - return { - rule_type: rule_type.id, - value: isNullRuleType ? null : val, - } - }) - ) - }) - - return filterRulesFromQueryParams -} diff --git a/src-ui/src/app/services/rest/document.service.ts b/src-ui/src/app/services/rest/document.service.ts index 8d5f80c04..190212084 100644 --- a/src-ui/src/app/services/rest/document.service.ts +++ b/src-ui/src/app/services/rest/document.service.ts @@ -11,7 +11,7 @@ import { CorrespondentService } from './correspondent.service' import { DocumentTypeService } from './document-type.service' import { TagService } from './tag.service' import { PaperlessDocumentSuggestions } from 'src/app/data/paperless-document-suggestions' -import { filterRulesToQueryParams } from '../query-params.service' +import { queryParamsFromFilterRules } from '../../utils/query-params' import { StoragePathService } from './storage-path.service' export const DOCUMENT_SORT_FIELDS = [ @@ -91,7 +91,7 @@ export class DocumentService extends AbstractPaperlessService pageSize, sortField, sortReverse, - Object.assign(extraParams, filterRulesToQueryParams(filterRules)) + Object.assign(extraParams, queryParamsFromFilterRules(filterRules)) ).pipe( map((results) => { results.results.forEach((doc) => this.addObservablesToDocument(doc)) diff --git a/src-ui/src/app/utils/query-params.ts b/src-ui/src/app/utils/query-params.ts new file mode 100644 index 000000000..a26093398 --- /dev/null +++ b/src-ui/src/app/utils/query-params.ts @@ -0,0 +1,100 @@ +import { ParamMap, Params } from '@angular/router' +import { FilterRule } from '../data/filter-rule' +import { FilterRuleType, FILTER_RULE_TYPES } from '../data/filter-rule-type' +import { ListViewState } from '../services/document-list-view.service' + +const SORT_FIELD_PARAMETER = 'sort' +const SORT_REVERSE_PARAMETER = 'reverse' +const PAGE_PARAMETER = 'page' + +export function generateParams( + filterRules: FilterRule[], + sortField: string, + sortReverse: boolean, + currentPage: number +): Params { + let params = {} + params[SORT_FIELD_PARAMETER] = sortField + params[SORT_REVERSE_PARAMETER] = sortReverse + params[PAGE_PARAMETER] = isNaN(currentPage) ? 1 : currentPage + return { + ...queryParamsFromFilterRules(filterRules), + ...params, + } +} + +export function parseQueryParams(queryParams: ParamMap): ListViewState { + let filterRules = filterRulesFromQueryParams(queryParams) + let sortField = queryParams.get(SORT_FIELD_PARAMETER) + let sortReverse = + queryParams.has(SORT_REVERSE_PARAMETER) || + (!queryParams.has(SORT_FIELD_PARAMETER) && + !queryParams.has(SORT_REVERSE_PARAMETER)) + let currentPage = queryParams.has(PAGE_PARAMETER) + ? parseInt(queryParams.get(PAGE_PARAMETER)) + : 1 + return { + currentPage: currentPage, + filterRules: filterRules, + sortField: sortField, + sortReverse: sortReverse, + } +} + +export function filterRulesFromQueryParams(queryParams: ParamMap) { + const allFilterRuleQueryParams: string[] = FILTER_RULE_TYPES.map( + (rt) => rt.filtervar + ) + .concat(FILTER_RULE_TYPES.map((rt) => rt.isnull_filtervar)) + .filter((rt) => rt !== undefined) + + // transform query params to filter rules + let filterRulesFromQueryParams: FilterRule[] = [] + allFilterRuleQueryParams + .filter((frqp) => queryParams.has(frqp)) + .forEach((filterQueryParamName) => { + const rule_type: FilterRuleType = FILTER_RULE_TYPES.find( + (rt) => + rt.filtervar == filterQueryParamName || + rt.isnull_filtervar == filterQueryParamName + ) + const isNullRuleType = rule_type.isnull_filtervar == filterQueryParamName + const valueURIComponent: string = queryParams.get(filterQueryParamName) + const filterQueryParamValues: string[] = rule_type.multi + ? valueURIComponent.split(',') + : [valueURIComponent] + + filterRulesFromQueryParams = filterRulesFromQueryParams.concat( + // map all values to filter rules + filterQueryParamValues.map((val) => { + return { + rule_type: rule_type.id, + value: isNullRuleType ? null : val, + } + }) + ) + }) + + return filterRulesFromQueryParams +} + +export function queryParamsFromFilterRules(filterRules: FilterRule[]): Params { + if (filterRules) { + let params = {} + for (let rule of filterRules) { + let ruleType = FILTER_RULE_TYPES.find((t) => t.id == rule.rule_type) + if (ruleType.multi) { + params[ruleType.filtervar] = params[ruleType.filtervar] + ? params[ruleType.filtervar] + ',' + rule.value + : rule.value + } else if (ruleType.isnull_filtervar && rule.value == null) { + params[ruleType.isnull_filtervar] = true + } else { + params[ruleType.filtervar] = rule.value + } + } + return params + } else { + return null + } +} From f6d78a0044dc9c013adcf1ac22fab5012cac4da2 Mon Sep 17 00:00:00 2001 From: Michael Shamoon <4887959+shamoon@users.noreply.github.com> Date: Fri, 20 May 2022 22:14:39 -0700 Subject: [PATCH 02/11] fix documents list without query params --- .../services/document-list-view.service.ts | 21 ++++++++----------- src-ui/src/app/utils/query-params.ts | 2 +- 2 files changed, 10 insertions(+), 13 deletions(-) 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 52a0de296..d2fbf4efe 100644 --- a/src-ui/src/app/services/document-list-view.service.ts +++ b/src-ui/src/app/services/document-list-view.service.ts @@ -1,5 +1,5 @@ import { Injectable } from '@angular/core' -import { ActivatedRoute, Params, Router } from '@angular/router' +import { ParamMap, Router } from '@angular/router' import { Observable } from 'rxjs' import { cloneFilterRules, @@ -10,7 +10,7 @@ import { PaperlessDocument } from '../data/paperless-document' import { PaperlessSavedView } from '../data/paperless-saved-view' import { SETTINGS_KEYS } from '../data/paperless-uisettings' import { DOCUMENT_LIST_SERVICE } from '../data/storage-keys' -import { generateParams, parseQueryParams } from '../utils/query-params' +import { generateParams, getStateFromQueryParams } from '../utils/query-params' import { DocumentService, DOCUMENT_SORT_FIELDS } from './rest/document.service' import { SettingsService } from './settings.service' @@ -166,13 +166,17 @@ export class DocumentListViewService { } } - loadFromQueryParams(queryParams) { - let newState: ListViewState = parseQueryParams(queryParams) + loadFromQueryParams(queryParams: ParamMap) { + const paramsEmpty: boolean = queryParams.keys.length == 0 + let newState: ListViewState = this.listViewStates.get(null) + if (!paramsEmpty) newState = getStateFromQueryParams(queryParams) + if (newState == undefined) newState = this.defaultListViewState() // if nothing in local storage + this.activeListViewState.filterRules = newState.filterRules this.activeListViewState.sortField = newState.sortField this.activeListViewState.sortReverse = newState.sortReverse this.activeListViewState.currentPage = newState.currentPage - this.reload(null, false) + this.reload(null, paramsEmpty) // update the params if there arent any } reload(onFinish?, updateQueryParams: boolean = true) { @@ -280,13 +284,6 @@ export class DocumentListViewService { return this.activeListViewState.sortReverse } - get sortParams(): Params { - return { - sortField: this.sortField, - sortReverse: this.sortReverse, - } - } - get collectionSize(): number { return this.activeListViewState.collectionSize } diff --git a/src-ui/src/app/utils/query-params.ts b/src-ui/src/app/utils/query-params.ts index a26093398..5294ad465 100644 --- a/src-ui/src/app/utils/query-params.ts +++ b/src-ui/src/app/utils/query-params.ts @@ -23,7 +23,7 @@ export function generateParams( } } -export function parseQueryParams(queryParams: ParamMap): ListViewState { +export function getStateFromQueryParams(queryParams: ParamMap): ListViewState { let filterRules = filterRulesFromQueryParams(queryParams) let sortField = queryParams.get(SORT_FIELD_PARAMETER) let sortReverse = From 1c83f489d1ab30ae4e426da583ccd884bd13b989 Mon Sep 17 00:00:00 2001 From: Michael Shamoon <4887959+shamoon@users.noreply.github.com> Date: Fri, 20 May 2022 22:23:35 -0700 Subject: [PATCH 03/11] Use 1/0 instead of true/false --- .../filter-editor/filter-editor.component.ts | 5 ++++- src-ui/src/app/utils/query-params.ts | 13 ++++++++++--- 2 files changed, 14 insertions(+), 4 deletions(-) diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts index f52435d49..421cd0693 100644 --- a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts @@ -313,7 +313,10 @@ export class FilterEditorComponent implements OnInit, OnDestroy { break case FILTER_ASN_ISNULL: this.textFilterTarget = TEXT_FILTER_TARGET_ASN - this.textFilterModifier = TEXT_FILTER_MODIFIER_NULL + this.textFilterModifier = + rule.value == 'true' || rule.value == '1' + ? TEXT_FILTER_MODIFIER_NULL + : TEXT_FILTER_MODIFIER_NOTNULL break case FILTER_ASN_GT: this.textFilterTarget = TEXT_FILTER_TARGET_ASN diff --git a/src-ui/src/app/utils/query-params.ts b/src-ui/src/app/utils/query-params.ts index 5294ad465..5c5dae8d5 100644 --- a/src-ui/src/app/utils/query-params.ts +++ b/src-ui/src/app/utils/query-params.ts @@ -15,7 +15,7 @@ export function generateParams( ): Params { let params = {} params[SORT_FIELD_PARAMETER] = sortField - params[SORT_REVERSE_PARAMETER] = sortReverse + params[SORT_REVERSE_PARAMETER] = sortReverse ? 1 : undefined params[PAGE_PARAMETER] = isNaN(currentPage) ? 1 : currentPage return { ...queryParamsFromFilterRules(filterRules), @@ -41,7 +41,9 @@ export function getStateFromQueryParams(queryParams: ParamMap): ListViewState { } } -export function filterRulesFromQueryParams(queryParams: ParamMap) { +export function filterRulesFromQueryParams( + queryParams: ParamMap +): FilterRule[] { const allFilterRuleQueryParams: string[] = FILTER_RULE_TYPES.map( (rt) => rt.filtervar ) @@ -67,6 +69,8 @@ export function filterRulesFromQueryParams(queryParams: ParamMap) { filterRulesFromQueryParams = filterRulesFromQueryParams.concat( // map all values to filter rules filterQueryParamValues.map((val) => { + if (rule_type.datatype == 'boolean') + val = val.replace('1', 'true').replace('0', 'false') return { rule_type: rule_type.id, value: isNullRuleType ? null : val, @@ -88,9 +92,12 @@ export function queryParamsFromFilterRules(filterRules: FilterRule[]): Params { ? params[ruleType.filtervar] + ',' + rule.value : rule.value } else if (ruleType.isnull_filtervar && rule.value == null) { - params[ruleType.isnull_filtervar] = true + params[ruleType.isnull_filtervar] = 1 } else { params[ruleType.filtervar] = rule.value + if (ruleType.datatype == 'boolean') + params[ruleType.filtervar] = + rule.value == 'true' || rule.value == '1' ? 1 : 0 } } return params From 48f9cb09af2fa64a960889430e52234b6047de29 Mon Sep 17 00:00:00 2001 From: Michael Shamoon <4887959+shamoon@users.noreply.github.com> Date: Fri, 20 May 2022 23:32:30 -0700 Subject: [PATCH 04/11] minor refactor query params utils --- .../services/document-list-view.service.ts | 11 +++------ src-ui/src/app/utils/query-params.ts | 24 +++++++------------ 2 files changed, 12 insertions(+), 23 deletions(-) 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 d2fbf4efe..b7ec2708d 100644 --- a/src-ui/src/app/services/document-list-view.service.ts +++ b/src-ui/src/app/services/document-list-view.service.ts @@ -10,7 +10,7 @@ import { PaperlessDocument } from '../data/paperless-document' import { PaperlessSavedView } from '../data/paperless-saved-view' import { SETTINGS_KEYS } from '../data/paperless-uisettings' import { DOCUMENT_LIST_SERVICE } from '../data/storage-keys' -import { generateParams, getStateFromQueryParams } from '../utils/query-params' +import { generateParams, parseParams } from '../utils/query-params' import { DocumentService, DOCUMENT_SORT_FIELDS } from './rest/document.service' import { SettingsService } from './settings.service' @@ -169,7 +169,7 @@ export class DocumentListViewService { loadFromQueryParams(queryParams: ParamMap) { const paramsEmpty: boolean = queryParams.keys.length == 0 let newState: ListViewState = this.listViewStates.get(null) - if (!paramsEmpty) newState = getStateFromQueryParams(queryParams) + if (!paramsEmpty) newState = parseParams(queryParams) if (newState == undefined) newState = this.defaultListViewState() // if nothing in local storage this.activeListViewState.filterRules = newState.filterRules @@ -200,12 +200,7 @@ export class DocumentListViewService { if (updateQueryParams && !this._activeSavedViewId) { let base = ['/documents'] this.router.navigate(base, { - queryParams: generateParams( - activeListViewState.filterRules, - activeListViewState.sortField, - activeListViewState.sortReverse, - activeListViewState.currentPage - ), + queryParams: generateParams(activeListViewState), }) } diff --git a/src-ui/src/app/utils/query-params.ts b/src-ui/src/app/utils/query-params.ts index 5c5dae8d5..6af44e8c9 100644 --- a/src-ui/src/app/utils/query-params.ts +++ b/src-ui/src/app/utils/query-params.ts @@ -7,23 +7,17 @@ const SORT_FIELD_PARAMETER = 'sort' const SORT_REVERSE_PARAMETER = 'reverse' const PAGE_PARAMETER = 'page' -export function generateParams( - filterRules: FilterRule[], - sortField: string, - sortReverse: boolean, - currentPage: number -): Params { - let params = {} - params[SORT_FIELD_PARAMETER] = sortField - params[SORT_REVERSE_PARAMETER] = sortReverse ? 1 : undefined - params[PAGE_PARAMETER] = isNaN(currentPage) ? 1 : currentPage - return { - ...queryParamsFromFilterRules(filterRules), - ...params, - } +export function generateParams(viewState: ListViewState): Params { + let params = queryParamsFromFilterRules(viewState.filterRules) + params[SORT_FIELD_PARAMETER] = viewState.sortField + params[SORT_REVERSE_PARAMETER] = viewState.sortReverse ? 1 : undefined + params[PAGE_PARAMETER] = isNaN(viewState.currentPage) + ? 1 + : viewState.currentPage + return params } -export function getStateFromQueryParams(queryParams: ParamMap): ListViewState { +export function parseParams(queryParams: ParamMap): ListViewState { let filterRules = filterRulesFromQueryParams(queryParams) let sortField = queryParams.get(SORT_FIELD_PARAMETER) let sortReverse = From 73a6e68e037e9f6d6e9001bfc1d56bc1f0707a34 Mon Sep 17 00:00:00 2001 From: Michael Shamoon <4887959+shamoon@users.noreply.github.com> Date: Sat, 21 May 2022 00:07:19 -0700 Subject: [PATCH 05/11] Fix conflicting rule type 22 --- src-ui/src/app/data/filter-rule-type.ts | 28 ++++++++++++++----------- 1 file changed, 16 insertions(+), 12 deletions(-) diff --git a/src-ui/src/app/data/filter-rule-type.ts b/src-ui/src/app/data/filter-rule-type.ts index 981169b7e..35e597ab6 100644 --- a/src-ui/src/app/data/filter-rule-type.ts +++ b/src-ui/src/app/data/filter-rule-type.ts @@ -1,34 +1,38 @@ export const FILTER_TITLE = 0 export const FILTER_CONTENT = 1 + export const FILTER_ASN = 2 +export const FILTER_ASN_ISNULL = 18 +export const FILTER_ASN_GT = 23 +export const FILTER_ASN_LT = 24 + export const FILTER_CORRESPONDENT = 3 + export const FILTER_DOCUMENT_TYPE = 4 + export const FILTER_IS_IN_INBOX = 5 export const FILTER_HAS_TAGS_ALL = 6 export const FILTER_HAS_ANY_TAG = 7 +export const FILTER_DOES_NOT_HAVE_TAG = 17 export const FILTER_HAS_TAGS_ANY = 22 + +export const FILTER_STORAGE_PATH = 25 + export const FILTER_CREATED_BEFORE = 8 export const FILTER_CREATED_AFTER = 9 export const FILTER_CREATED_YEAR = 10 export const FILTER_CREATED_MONTH = 11 export const FILTER_CREATED_DAY = 12 + export const FILTER_ADDED_BEFORE = 13 export const FILTER_ADDED_AFTER = 14 + export const FILTER_MODIFIED_BEFORE = 15 export const FILTER_MODIFIED_AFTER = 16 -export const FILTER_DOES_NOT_HAVE_TAG = 17 - -export const FILTER_ASN_ISNULL = 18 -export const FILTER_ASN_GT = 19 -export const FILTER_ASN_LT = 20 - -export const FILTER_TITLE_CONTENT = 21 - -export const FILTER_FULLTEXT_QUERY = 22 -export const FILTER_FULLTEXT_MORELIKE = 23 - -export const FILTER_STORAGE_PATH = 30 +export const FILTER_TITLE_CONTENT = 19 +export const FILTER_FULLTEXT_QUERY = 20 +export const FILTER_FULLTEXT_MORELIKE = 21 export const FILTER_RULE_TYPES: FilterRuleType[] = [ { From 8e2cb6d416276951a55b99c575443fddf9b81fe8 Mon Sep 17 00:00:00 2001 From: Michael Shamoon <4887959+shamoon@users.noreply.github.com> Date: Sun, 22 May 2022 08:59:43 -0700 Subject: [PATCH 06/11] Strip some characters from pasted date input --- .../common/input/date/date.component.html | 2 +- .../components/common/input/date/date.component.ts | 13 ++++++++++++- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/src-ui/src/app/components/common/input/date/date.component.html b/src-ui/src/app/components/common/input/date/date.component.html index ca6ec0b26..e742ead9b 100644 --- a/src-ui/src/app/components/common/input/date/date.component.html +++ b/src-ui/src/app/components/common/input/date/date.component.html @@ -2,7 +2,7 @@

+

diff --git a/src-ui/src/app/components/manage/settings/settings.component.html b/src-ui/src/app/components/manage/settings/settings.component.html index 7e52db59e..002cc4eed 100644 --- a/src-ui/src/app/components/manage/settings/settings.component.html +++ b/src-ui/src/app/components/manage/settings/settings.component.html @@ -22,7 +22,7 @@ - You need to reload the page after applying a new language. + You need to reload the page after applying a new language.
diff --git a/src-ui/src/app/components/manage/settings/settings.component.ts b/src-ui/src/app/components/manage/settings/settings.component.ts index d9877d281..22ecfe9bb 100644 --- a/src-ui/src/app/components/manage/settings/settings.component.ts +++ b/src-ui/src/app/components/manage/settings/settings.component.ts @@ -14,7 +14,7 @@ import { LanguageOption, SettingsService, } from 'src/app/services/settings.service' -import { ToastService } from 'src/app/services/toast.service' +import { Toast, ToastService } from 'src/app/services/toast.service' import { dirtyCheck, DirtyComponent } from '@ngneat/dirty-check-forms' import { Observable, Subscription, BehaviorSubject, first } from 'rxjs' import { SETTINGS_KEYS } from 'src/app/data/paperless-uisettings' @@ -61,6 +61,13 @@ export class SettingsComponent implements OnInit, OnDestroy, DirtyComponent { ) } + get displayLanguageIsDirty(): boolean { + return ( + this.settingsForm.get('displayLanguage').value != + this.store?.getValue()['displayLanguage'] + ) + } + constructor( public savedViewService: SavedViewService, private documentListViewService: DocumentListViewService, @@ -170,6 +177,7 @@ export class SettingsComponent implements OnInit, OnDestroy, DirtyComponent { } private saveLocalSettings() { + const reloadRequired = this.displayLanguageIsDirty // just this one, for now this.settings.set( SETTINGS_KEYS.BULK_EDIT_APPLY_ON_CLOSE, this.settingsForm.value.bulkEditApplyOnClose @@ -235,7 +243,20 @@ export class SettingsComponent implements OnInit, OnDestroy, DirtyComponent { this.store.next(this.settingsForm.value) this.documentListViewService.updatePageSize() this.settings.updateAppearanceSettings() - this.toastService.showInfo($localize`Settings saved successfully.`) + let savedToast: Toast = { + title: $localize`Settings saved`, + content: $localize`Settings were saved successfully.`, + delay: 500000, + } + if (reloadRequired) { + ;(savedToast.content = $localize`Settings were saved successfully. Reload is required to apply some changes.`), + (savedToast.actionName = $localize`Reload now`) + savedToast.action = () => { + location.reload() + } + } + + this.toastService.show(savedToast) }, error: (error) => { this.toastService.showError( diff --git a/src-ui/src/styles.scss b/src-ui/src/styles.scss index 4b78e9d21..e0934b84c 100644 --- a/src-ui/src/styles.scss +++ b/src-ui/src/styles.scss @@ -84,6 +84,10 @@ svg.logo { } } +.text-primary { + color: var(--bs-primary) !important; +} + .btn-outline-primary { border-color: var(--bs-primary) !important; color: var(--bs-primary) !important; diff --git a/src-ui/src/theme.scss b/src-ui/src/theme.scss index 732ac47d9..bf9be6662 100644 --- a/src-ui/src/theme.scss +++ b/src-ui/src/theme.scss @@ -186,7 +186,8 @@ $form-check-radio-checked-bg-image-dark: url("data:image/svg+xml,