diff --git a/src-ui/cypress/e2e/documents/documents-list.cy.ts b/src-ui/cypress/e2e/documents/documents-list.cy.ts index 5c17ef5d9..847d038b8 100644 --- a/src-ui/cypress/e2e/documents/documents-list.cy.ts +++ b/src-ui/cypress/e2e/documents/documents-list.cy.ts @@ -150,7 +150,7 @@ describe('documents-list', () => { cy.contains('button', 'Corresp 11').click() cy.contains('label', 'Exclude').click() }) - cy.contains('One document') + cy.contains('3 documents') }) it('should apply tags', () => { diff --git a/src-ui/cypress/e2e/documents/query-params.cy.ts b/src-ui/cypress/e2e/documents/query-params.cy.ts index eb160e9de..78a85d185 100644 --- a/src-ui/cypress/e2e/documents/query-params.cy.ts +++ b/src-ui/cypress/e2e/documents/query-params.cy.ts @@ -190,6 +190,36 @@ describe('documents query params', () => { response.count = response.results.length } + if (req.query.hasOwnProperty('owner__id')) { + response.results = ( + documentsJson.results as Array + ).filter((d) => d.owner == req.query['owner__id']) + response.count = response.results.length + } else if (req.query.hasOwnProperty('owner__id__in')) { + const owners = req.query['owner__id__in'] + .toString() + .split(',') + .map((o) => parseInt(o)) + response.results = ( + documentsJson.results as Array + ).filter((d) => owners.includes(d.owner)) + response.count = response.results.length + } else if (req.query.hasOwnProperty('owner__id__none')) { + const owners = req.query['owner__id__none'] + .toString() + .split(',') + .map((o) => parseInt(o)) + response.results = ( + documentsJson.results as Array + ).filter((d) => !owners.includes(d.owner)) + response.count = response.results.length + } else if (req.query.hasOwnProperty('owner__isnull')) { + response.results = ( + documentsJson.results as Array + ).filter((d) => d.owner === null) + response.count = response.results.length + } + req.reply(response) }) }) @@ -202,7 +232,7 @@ describe('documents query params', () => { it('should show a list of documents reverse sorted by created', () => { cy.visit('/documents?sort=created&reverse=true') - cy.get('app-document-card-small').first().contains('sit amet') + cy.get('app-document-card-small').first().contains('Doc 6') }) it('should show a list of documents sorted by added', () => { @@ -212,7 +242,7 @@ describe('documents query params', () => { it('should show a list of documents reverse sorted by added', () => { cy.visit('/documents?sort=added&reverse=true') - cy.get('app-document-card-small').first().contains('sit amet') + cy.get('app-document-card-small').first().contains('Doc 6') }) it('should show a list of documents filtered by any tags', () => { @@ -222,12 +252,12 @@ describe('documents query params', () => { it('should show a list of documents filtered by excluded tags', () => { cy.visit('/documents?sort=created&reverse=true&tags__id__none=2,4') - cy.contains('One document') + cy.contains('3 documents') }) it('should show a list of documents filtered by no tags', () => { cy.visit('/documents?sort=created&reverse=true&is_tagged=0') - cy.contains('One document') + cy.contains('3 documents') }) it('should show a list of documents filtered by document type', () => { @@ -242,7 +272,7 @@ describe('documents query params', () => { it('should show a list of documents filtered by no document type', () => { cy.visit('/documents?sort=created&reverse=true&document_type__isnull=1') - cy.contains('One document') + cy.contains('3 documents') }) it('should show a list of documents filtered by correspondent', () => { @@ -257,7 +287,7 @@ describe('documents query params', () => { it('should show a list of documents filtered by no correspondent', () => { cy.visit('/documents?sort=created&reverse=true&correspondent__isnull=1') - cy.contains('One document') + cy.contains('3 documents') }) it('should show a list of documents filtered by storage path', () => { @@ -267,7 +297,7 @@ describe('documents query params', () => { it('should show a list of documents filtered by no storage path', () => { cy.visit('/documents?sort=created&reverse=true&storage_path__isnull=1') - cy.contains('3 documents') + cy.contains('5 documents') }) it('should show a list of documents filtered by title or content', () => { @@ -312,7 +342,7 @@ describe('documents query params', () => { cy.visit( '/documents?sort=created&reverse=true&created__date__gt=2022-03-23' ) - cy.contains('3 documents') + cy.contains('5 documents') }) it('should show a list of documents filtered by created date less than', () => { @@ -324,7 +354,7 @@ describe('documents query params', () => { it('should show a list of documents filtered by added date greater than', () => { cy.visit('/documents?sort=created&reverse=true&added__date__gt=2022-03-24') - cy.contains('2 documents') + cy.contains('4 documents') }) it('should show a list of documents filtered by added date less than', () => { @@ -338,4 +368,24 @@ describe('documents query params', () => { ) cy.contains('2 documents') }) + + it('should show a list of documents filtered by owner', () => { + cy.visit('/documents?owner__id=15') + cy.contains('One document') + }) + + it('should show a list of documents filtered by multiple owners', () => { + cy.visit('/documents?owner__id__in=6,15') + cy.contains('2 documents') + }) + + it('should show a list of documents filtered by excluded owners', () => { + cy.visit('/documents?owner__id__none=6') + cy.contains('5 documents') + }) + + it('should show a list of documents filtered by null owner', () => { + cy.visit('/documents?owner__isnull=true') + cy.contains('4 documents') + }) }) diff --git a/src-ui/cypress/fixtures/documents/documents.json b/src-ui/cypress/fixtures/documents/documents.json index 6b284f7b2..a33e4e43f 100644 --- a/src-ui/cypress/fixtures/documents/documents.json +++ b/src-ui/cypress/fixtures/documents/documents.json @@ -143,6 +143,64 @@ } }, "notes": [] + }, + { + "id": 5, + "correspondent": null, + "document_type": null, + "storage_path": null, + "title": "Doc 5", + "content": "Test document 5", + "tags": [], + "created": "2023-05-01T07:24:18Z", + "created_date": "2023-05-02", + "modified": "2023-05-02T07:24:23.264859Z", + "added": "2023-05-02T07:24:22.922631Z", + "archive_serial_number": null, + "original_file_name": "doc5.pdf", + "archived_file_name": "doc5.pdf", + "owner": 15, + "user_can_change": true, + "permissions": { + "view": { + "users": [1], + "groups": [] + }, + "change": { + "users": [], + "groups": [] + } + }, + "notes": [] + }, + { + "id": 6, + "correspondent": null, + "document_type": null, + "storage_path": null, + "title": "Doc 6", + "content": "Test document 6", + "tags": [], + "created": "2023-05-01T10:24:18Z", + "created_date": "2023-05-02", + "modified": "2023-05-02T10:24:23.264859Z", + "added": "2023-05-02T10:24:22.922631Z", + "archive_serial_number": null, + "original_file_name": "doc6.pdf", + "archived_file_name": "doc6.pdf", + "owner": 6, + "user_can_change": true, + "permissions": { + "view": { + "users": [1], + "groups": [] + }, + "change": { + "users": [], + "groups": [] + } + }, + "notes": [] } ] } diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index ed847d41a..6c6d13f6c 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -88,6 +88,10 @@ import { PermissionsUserComponent } from './components/common/input/permissions/ import { PermissionsGroupComponent } from './components/common/input/permissions/permissions-group/permissions-group.component' import { IfOwnerDirective } from './directives/if-owner.directive' import { IfObjectPermissionsDirective } from './directives/if-object-permissions.directive' +import { PermissionsDialogComponent } from './components/common/permissions-dialog/permissions-dialog.component' +import { PermissionsFormComponent } from './components/common/input/permissions/permissions-form/permissions-form.component' +import { PermissionsFilterDropdownComponent } from './components/common/permissions-filter-dropdown/permissions-filter-dropdown.component' +import { UsernamePipe } from './pipes/username.pipe' import localeAr from '@angular/common/locales/ar' import localeBe from '@angular/common/locales/be' @@ -111,8 +115,6 @@ import localeSr from '@angular/common/locales/sr' import localeSv from '@angular/common/locales/sv' import localeTr from '@angular/common/locales/tr' import localeZh from '@angular/common/locales/zh' -import { PermissionsDialogComponent } from './components/common/permissions-dialog/permissions-dialog.component' -import { PermissionsFormComponent } from './components/common/input/permissions/permissions-form/permissions-form.component' registerLocaleData(localeAr) registerLocaleData(localeBe) @@ -213,6 +215,8 @@ function initializeApp(settings: SettingsService) { IfObjectPermissionsDirective, PermissionsDialogComponent, PermissionsFormComponent, + PermissionsFilterDropdownComponent, + UsernamePipe, ], imports: [ BrowserModule, @@ -253,6 +257,7 @@ function initializeApp(settings: SettingsService) { PermissionsGuard, DirtyDocGuard, DirtySavedViewGuard, + UsernamePipe, ], bootstrap: [AppComponent], }) diff --git a/src-ui/src/app/components/common/date-dropdown/date-dropdown.component.html b/src-ui/src/app/components/common/date-dropdown/date-dropdown.component.html index 58634b4d0..40bce985d 100644 --- a/src-ui/src/app/components/common/date-dropdown/date-dropdown.component.html +++ b/src-ui/src/app/components/common/date-dropdown/date-dropdown.component.html @@ -6,7 +6,7 @@ +
+ + {{document.owner | username}} +
Score: diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts index 58a3dd4e4..d2153fb62 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts @@ -23,7 +23,7 @@ import { ComponentWithPermissions } from '../../with-permissions/with-permission export class DocumentCardLargeComponent extends ComponentWithPermissions { constructor( private documentService: DocumentService, - private settingsService: SettingsService + public settingsService: SettingsService ) { super() } 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 7365fec36..f61d586c8 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 @@ -38,15 +38,15 @@
@@ -59,18 +59,23 @@
- {{document.created_date | customDate:'mediumDate'}}
-
- - #{{document.archive_serial_number}} -
+
+
+ + #{{document.archive_serial_number}} +
+
+ + {{document.owner | username}}
diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts index fcec51ebf..62f44851e 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts @@ -24,7 +24,7 @@ import { ComponentWithPermissions } from '../../with-permissions/with-permission export class DocumentCardSmallComponent extends ComponentWithPermissions { constructor( private documentService: DocumentService, - private settingsService: SettingsService + public settingsService: SettingsService ) { super() } 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 0b7b06dbc..c0fba9325 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 @@ -142,6 +142,13 @@ [currentSortReverse]="list.sortReverse" (sort)="onSort($event)" i18n>Title + Owner {{d.title | documentTitle}} + + {{d.owner | username}} + diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html index 4e7851a57..e83688596 100644 --- a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html @@ -58,20 +58,26 @@ [documentCounts]="storagePathDocumentCounts" [allowSelectNone]="true">
-
+
-
+
+ +
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 10048f7d7..37a58c54c 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 @@ -43,6 +43,10 @@ import { FILTER_DOCUMENT_TYPE, FILTER_CORRESPONDENT, FILTER_STORAGE_PATH, + FILTER_OWNER, + FILTER_OWNER_DOES_NOT_INCLUDE, + FILTER_OWNER_ISNULL, + FILTER_OWNER_ANY, } from 'src/app/data/filter-rule-type' import { FilterableDropdownSelectionModel, @@ -59,6 +63,11 @@ import { PaperlessDocument } from 'src/app/data/paperless-document' import { PaperlessStoragePath } from 'src/app/data/paperless-storage-path' import { StoragePathService } from 'src/app/services/rest/storage-path.service' import { RelativeDate } from '../../common/date-dropdown/date-dropdown.component' +import { + OwnerFilterType, + PermissionsSelectionModel, +} from '../../common/permissions-filter-dropdown/permissions-filter-dropdown.component' +import { SettingsService } from 'src/app/services/settings.service' const TEXT_FILTER_TARGET_TITLE = 'title' const TEXT_FILTER_TARGET_TITLE_CONTENT = 'title-content' @@ -136,6 +145,15 @@ export class FilterEditorComponent implements OnInit, OnDestroy { case FILTER_ASN: return $localize`ASN: ${rule.value}` + + case FILTER_OWNER: + return $localize`Owner: ${rule.value}` + + case FILTER_OWNER_DOES_NOT_INCLUDE: + return $localize`Owner not in: ${rule.value}` + + case FILTER_OWNER_ISNULL: + return $localize`Without an owner` } } @@ -147,7 +165,8 @@ export class FilterEditorComponent implements OnInit, OnDestroy { private tagService: TagService, private correspondentService: CorrespondentService, private documentService: DocumentService, - private storagePathService: StoragePathService + private storagePathService: StoragePathService, + private settingsService: SettingsService ) {} @ViewChild('textFilterInput') @@ -241,6 +260,8 @@ export class FilterEditorComponent implements OnInit, OnDestroy { dateCreatedRelativeDate: RelativeDate dateAddedRelativeDate: RelativeDate + permissionsSelectionModel = new PermissionsSelectionModel() + _unmodifiedFilterRules: FilterRule[] = [] _filterRules: FilterRule[] = [] @@ -274,6 +295,7 @@ export class FilterEditorComponent implements OnInit, OnDestroy { this.dateCreatedRelativeDate = null this.dateAddedRelativeDate = null this.textFilterModifier = TEXT_FILTER_MODIFIER_EQUALS + this.permissionsSelectionModel.clear() value.forEach((rule) => { switch (rule.rule_type) { @@ -441,6 +463,35 @@ export class FilterEditorComponent implements OnInit, OnDestroy { this.textFilterModifier = TEXT_FILTER_MODIFIER_LT this._textFilter = rule.value break + case FILTER_OWNER: + this.permissionsSelectionModel.ownerFilter = OwnerFilterType.SELF + this.permissionsSelectionModel.hideUnowned = false + if (rule.value) + this.permissionsSelectionModel.userID = parseInt(rule.value, 10) + break + case FILTER_OWNER_ANY: + this.permissionsSelectionModel.ownerFilter = OwnerFilterType.OTHERS + if (rule.value) + this.permissionsSelectionModel.includeUsers.push( + parseInt(rule.value, 10) + ) + break + case FILTER_OWNER_DOES_NOT_INCLUDE: + this.permissionsSelectionModel.ownerFilter = OwnerFilterType.NOT_SELF + if (rule.value) + this.permissionsSelectionModel.excludeUsers.push( + parseInt(rule.value, 10) + ) + break + case FILTER_OWNER_ISNULL: + if (rule.value === 'true' || rule.value === '1') { + this.permissionsSelectionModel.hideUnowned = false + this.permissionsSelectionModel.ownerFilter = OwnerFilterType.UNOWNED + } else { + this.permissionsSelectionModel.hideUnowned = + rule.value === 'false' || rule.value === '0' + break + } } }) this.rulesModified = filterRulesDiffer( @@ -702,6 +753,40 @@ export class FilterEditorComponent implements OnInit, OnDestroy { } } } + if (this.permissionsSelectionModel.ownerFilter == OwnerFilterType.SELF) { + filterRules.push({ + rule_type: FILTER_OWNER, + value: this.permissionsSelectionModel.userID.toString(), + }) + } else if ( + this.permissionsSelectionModel.ownerFilter == OwnerFilterType.NOT_SELF + ) { + filterRules.push({ + rule_type: FILTER_OWNER_DOES_NOT_INCLUDE, + value: this.permissionsSelectionModel.excludeUsers?.join(','), + }) + } else if ( + this.permissionsSelectionModel.ownerFilter == OwnerFilterType.OTHERS + ) { + filterRules.push({ + rule_type: FILTER_OWNER_ANY, + value: this.permissionsSelectionModel.includeUsers?.join(','), + }) + } else if ( + this.permissionsSelectionModel.ownerFilter == OwnerFilterType.UNOWNED + ) { + filterRules.push({ + rule_type: FILTER_OWNER_ISNULL, + value: 'true', + }) + } + + if (this.permissionsSelectionModel.hideUnowned) { + filterRules.push({ + rule_type: FILTER_OWNER_ISNULL, + value: 'false', + }) + } return filterRules } diff --git a/src-ui/src/app/data/filter-rule-type.ts b/src-ui/src/app/data/filter-rule-type.ts index 50b80b13b..f65f52fd2 100644 --- a/src-ui/src/app/data/filter-rule-type.ts +++ b/src-ui/src/app/data/filter-rule-type.ts @@ -41,6 +41,11 @@ export const FILTER_TITLE_CONTENT = 19 export const FILTER_FULLTEXT_QUERY = 20 export const FILTER_FULLTEXT_MORELIKE = 21 +export const FILTER_OWNER = 32 +export const FILTER_OWNER_ANY = 33 +export const FILTER_OWNER_ISNULL = 34 +export const FILTER_OWNER_DOES_NOT_INCLUDE = 35 + export const FILTER_RULE_TYPES: FilterRuleType[] = [ { id: FILTER_TITLE, @@ -242,6 +247,30 @@ export const FILTER_RULE_TYPES: FilterRuleType[] = [ datatype: 'number', multi: false, }, + { + id: FILTER_OWNER, + filtervar: 'owner__id', + datatype: 'number', + multi: false, + }, + { + id: FILTER_OWNER_ANY, + filtervar: 'owner__id__in', + datatype: 'number', + multi: true, + }, + { + id: FILTER_OWNER_ISNULL, + filtervar: 'owner__isnull', + datatype: 'boolean', + multi: false, + }, + { + id: FILTER_OWNER_DOES_NOT_INCLUDE, + filtervar: 'owner__id__none', + datatype: 'number', + multi: true, + }, ] export interface FilterRuleType { diff --git a/src-ui/src/app/pipes/username.pipe.ts b/src-ui/src/app/pipes/username.pipe.ts new file mode 100644 index 000000000..79d2657a2 --- /dev/null +++ b/src-ui/src/app/pipes/username.pipe.ts @@ -0,0 +1,42 @@ +import { Pipe, PipeTransform } from '@angular/core' +import { UserService } from '../services/rest/user.service' +import { + PermissionAction, + PermissionType, + PermissionsService, +} from '../services/permissions.service' +import { PaperlessUser } from '../data/paperless-user' + +@Pipe({ + name: 'username', +}) +export class UsernamePipe implements PipeTransform { + users: PaperlessUser[] + + constructor( + permissionsService: PermissionsService, + userService: UserService + ) { + if ( + permissionsService.currentUserCan( + PermissionAction.View, + PermissionType.User + ) + ) { + userService.listAll().subscribe((r) => (this.users = r.results)) + } + } + + transform(userID: number): string { + return this.users + ? this.getName(this.users.find((u) => u.id === userID)) ?? '' + : $localize`Shared` + } + + getName(user: PaperlessUser): string { + if (!user) return '' + const name = [user.first_name, user.last_name].join(' ') + if (name.length > 1) return name.trim() + return user.username + } +} diff --git a/src-ui/src/app/services/rest/document.service.ts b/src-ui/src/app/services/rest/document.service.ts index 4ff2ee88f..08050ac85 100644 --- a/src-ui/src/app/services/rest/document.service.ts +++ b/src-ui/src/app/services/rest/document.service.ts @@ -23,6 +23,7 @@ export const DOCUMENT_SORT_FIELDS = [ { field: 'added', name: $localize`Added` }, { field: 'modified', name: $localize`Modified` }, { field: 'num_notes', name: $localize`Notes` }, + { field: 'owner', name: $localize`Owner` }, ] export const DOCUMENT_SORT_FIELDS_FULLTEXT = [