From 4481f12e3274c1de4479528e9f84540ce57d58ff Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Tue, 5 Dec 2023 08:16:56 -0800 Subject: [PATCH] Enhancement: implement document link custom field (#4799) --- docs/usage.md | 1 + src-ui/messages.xlf | 244 +++++++++++------- src-ui/src/app/app.module.ts | 2 + .../custom-field-edit-dialog.component.html | 30 +-- .../document-link.component.html | 50 ++++ .../document-link.component.scss | 14 + .../document-link.component.spec.ts | 118 +++++++++ .../document-link/document-link.component.ts | 120 +++++++++ .../common/input/tags/tags.component.html | 2 +- .../document-detail.component.html | 1 + src-ui/src/app/data/paperless-custom-field.ts | 5 + ...onsumptiontemplate_assign_custom_fields.py | 23 -- ...ntemplate_assign_custom_fields_and_more.py | 47 ++++ src/documents/models.py | 5 + src/documents/serialisers.py | 1 + src/documents/tests/test_api_custom_fields.py | 15 +- 16 files changed, 542 insertions(+), 136 deletions(-) create mode 100644 src-ui/src/app/components/common/input/document-link/document-link.component.html create mode 100644 src-ui/src/app/components/common/input/document-link/document-link.component.scss create mode 100644 src-ui/src/app/components/common/input/document-link/document-link.component.spec.ts create mode 100644 src-ui/src/app/components/common/input/document-link/document-link.component.ts delete mode 100644 src/documents/migrations/1042_consumptiontemplate_assign_custom_fields.py create mode 100644 src/documents/migrations/1042_consumptiontemplate_assign_custom_fields_and_more.py diff --git a/docs/usage.md b/docs/usage.md index 4cb55613c..fed7412b3 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -343,6 +343,7 @@ The following custom field types are supported: - `Integer`: integer number e.g. 12 - `Number`: float number e.g. 12.3456 - `Monetary`: float number with exactly two decimals, e.g. 12.30 +- `Document Link`: reference(s) to other document(s), displayed as links ## Share Links diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 1b94018f3..723abc0f9 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -294,7 +294,7 @@ src/app/components/document-detail/document-detail.component.html - 81 + 92 @@ -447,6 +447,10 @@ src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.html 30 + + src/app/components/common/input/document-link/document-link.component.html + 40 + src/app/components/common/permissions-dialog/permissions-dialog.component.html 18 @@ -459,6 +463,10 @@ src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html 14 + + src/app/components/document-detail/document-detail.component.html + 248 + src/app/components/document-list/document-list.component.html 95 @@ -766,7 +774,7 @@ src/app/components/document-detail/document-detail.component.html - 221 + 216 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -852,11 +860,11 @@ src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 46 + 47 src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 65 + 66 src/app/components/common/input/permissions/permissions-form/permissions-form.component.html @@ -879,11 +887,11 @@ src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 54 + 55 src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 73 + 74 src/app/components/common/input/permissions/permissions-form/permissions-form.component.html @@ -909,7 +917,7 @@ src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 79 + 80 src/app/components/common/input/permissions/permissions-form/permissions-form.component.html @@ -1150,7 +1158,7 @@ src/app/components/document-detail/document-detail.component.html - 34 + 45 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1217,7 +1225,7 @@ src/app/components/document-detail/document-detail.component.html - 11 + 22 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1291,7 +1299,7 @@ src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 90 + 91 src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html @@ -1335,7 +1343,7 @@ src/app/components/document-detail/document-detail.component.html - 93 + 104 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -1829,7 +1837,7 @@ src/app/components/document-detail/document-detail.component.ts - 677 + 691 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -1868,7 +1876,7 @@ src/app/components/document-detail/document-detail.component.ts - 679 + 693 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2556,32 +2564,39 @@ 37 + + Assign custom fields + + src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html + 38 + + Assign owner src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 40 + 41 Assign view permissions src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 42 + 43 Assign edit permissions src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 61 + 62 Error src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 88 + 89 src/app/components/common/edit-dialog/mail-rule-edit-dialog/mail-rule-edit-dialog.component.html @@ -2596,7 +2611,7 @@ Cancel src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.html - 89 + 90 src/app/components/common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component.html @@ -2659,35 +2674,35 @@ Consume Folder src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts - 25 + 27 API Upload src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts - 29 + 31 Mail Fetch src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts - 33 + 35 Create new consumption template src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts - 83 + 92 Edit consumption template src/app/components/common/edit-dialog/consumption-template-edit-dialog/consumption-template-edit-dialog.component.ts - 87 + 96 @@ -2807,14 +2822,14 @@ Create new item src/app/components/common/edit-dialog/edit-dialog.component.ts - 109 + 111 Edit item src/app/components/common/edit-dialog/edit-dialog.component.ts - 113 + 115 @@ -3468,6 +3483,10 @@ src/app/components/common/input/date/date.component.html 8 + + src/app/components/common/input/document-link/document-link.component.html + 8 + src/app/components/common/input/number/number.component.html 8 @@ -3518,6 +3537,13 @@ 155 + + No documents found + + src/app/components/common/input/document-link/document-link.component.ts + 44 + + Show password @@ -3627,7 +3653,7 @@ Note that permissions set here will override any existing permissions src/app/components/common/permissions-dialog/permissions-dialog.component.ts - 62 + 71 @@ -3971,7 +3997,7 @@ src/app/components/document-detail/document-detail.component.html - 103 + 114 src/app/components/document-list/document-list.component.html @@ -3994,7 +4020,7 @@ src/app/components/document-detail/document-detail.component.html - 107 + 118 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4028,7 +4054,7 @@ src/app/components/document-detail/document-detail.component.html - 18 + 29 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4215,7 +4241,7 @@ Page src/app/components/document-detail/document-detail.component.html - 3 + 4 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4226,21 +4252,35 @@ of src/app/components/document-detail/document-detail.component.html - 5,6 + 6,7 + + + + - + + src/app/components/document-detail/document-detail.component.html + 9 + + + + + + + src/app/components/document-detail/document-detail.component.html + 15 Download original src/app/components/document-detail/document-detail.component.html - 24 + 35 Redo OCR src/app/components/document-detail/document-detail.component.html - 40 + 51 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4251,7 +4291,7 @@ More like this src/app/components/document-detail/document-detail.component.html - 46 + 57 src/app/components/document-list/document-card-large/document-card-large.component.html @@ -4262,7 +4302,7 @@ Close src/app/components/document-detail/document-detail.component.html - 71 + 82 src/app/guards/dirty-saved-view.guard.ts @@ -4273,56 +4313,56 @@ Previous src/app/components/document-detail/document-detail.component.html - 76 + 87 Discard src/app/components/document-detail/document-detail.component.html - 89 + 100 Save & next src/app/components/document-detail/document-detail.component.html - 91 + 102 Save & close src/app/components/document-detail/document-detail.component.html - 92 + 103 Details src/app/components/document-detail/document-detail.component.html - 100 + 111 Archive serial number src/app/components/document-detail/document-detail.component.html - 104 + 115 Date created src/app/components/document-detail/document-detail.component.html - 105 + 116 Document type src/app/components/document-detail/document-detail.component.html - 109 + 120 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4345,7 +4385,7 @@ Storage path src/app/components/document-detail/document-detail.component.html - 111 + 122 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4364,21 +4404,21 @@ Default src/app/components/document-detail/document-detail.component.html - 112 + 123 Content src/app/components/document-detail/document-detail.component.html - 130 + 142 Metadata src/app/components/document-detail/document-detail.component.html - 139 + 151 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -4389,152 +4429,152 @@ Date modified src/app/components/document-detail/document-detail.component.html - 145 + 157 Date added src/app/components/document-detail/document-detail.component.html - 149 + 161 Media filename src/app/components/document-detail/document-detail.component.html - 153 + 165 Original filename src/app/components/document-detail/document-detail.component.html - 157 + 169 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 161 + 173 Original file size src/app/components/document-detail/document-detail.component.html - 165 + 177 Original mime type src/app/components/document-detail/document-detail.component.html - 169 + 181 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 173 + 185 Archive file size src/app/components/document-detail/document-detail.component.html - 177 + 189 Original document metadata src/app/components/document-detail/document-detail.component.html - 183 + 195 Archived document metadata src/app/components/document-detail/document-detail.component.html - 184 + 196 Preview src/app/components/document-detail/document-detail.component.html - 190 - - - - Enter Password - - src/app/components/document-detail/document-detail.component.html - 206 - - - src/app/components/document-detail/document-detail.component.html - 249 + 202 Notes src/app/components/document-detail/document-detail.component.html - 214,215 + 209,210 + + + + Enter Password + + src/app/components/document-detail/document-detail.component.html + 237 + + + src/app/components/document-detail/document-detail.component.html + 275 An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 265,267 + 279,281 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 413 + 427 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 434 + 448 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 552 + 566 src/app/components/document-detail/document-detail.component.ts - 561 + 575 Error saving document src/app/components/document-detail/document-detail.component.ts - 565 + 579 src/app/components/document-detail/document-detail.component.ts - 606 + 620 Confirm delete src/app/components/document-detail/document-detail.component.ts - 632 + 646 src/app/components/manage/management-list/management-list.component.ts @@ -4545,35 +4585,35 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 633 + 647 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 634 + 648 Delete document src/app/components/document-detail/document-detail.component.ts - 636 + 650 Error deleting document src/app/components/document-detail/document-detail.component.ts - 655 + 669 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 675 + 689 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -4584,21 +4624,28 @@ This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 676 + 690 Redo OCR operation will begin in the background. Close and re-open or reload this document after the operation has completed to see new content. src/app/components/document-detail/document-detail.component.ts - 687 + 701 Error executing operation src/app/components/document-detail/document-detail.component.ts - 698 + 712 + + + + Page Fit + + src/app/components/document-detail/document-detail.component.ts + 781 @@ -5433,7 +5480,7 @@ Initiating upload... src/app/components/file-drop/file-drop.component.ts - 87 + 88 @@ -6105,49 +6152,56 @@ Boolean src/app/data/paperless-custom-field.ts - 16 + 17 Date src/app/data/paperless-custom-field.ts - 20 + 21 Integer src/app/data/paperless-custom-field.ts - 24 + 25 Number src/app/data/paperless-custom-field.ts - 28 + 29 Monetary src/app/data/paperless-custom-field.ts - 32 + 33 Text src/app/data/paperless-custom-field.ts - 36 + 37 Url src/app/data/paperless-custom-field.ts - 40 + 41 + + + + Document Link + + src/app/data/paperless-custom-field.ts + 45 diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 8d7ea5663..6910061d2 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -106,6 +106,7 @@ import { CustomFieldEditDialogComponent } from './components/common/edit-dialog/ import { CustomFieldsDropdownComponent } from './components/common/custom-fields-dropdown/custom-fields-dropdown.component' import { ProfileEditDialogComponent } from './components/common/profile-edit-dialog/profile-edit-dialog.component' import { PdfViewerComponent } from './components/common/pdf-viewer/pdf-viewer.component' +import { DocumentLinkComponent } from './components/common/input/document-link/document-link.component' import localeAf from '@angular/common/locales/af' import localeAr from '@angular/common/locales/ar' @@ -259,6 +260,7 @@ function initializeApp(settings: SettingsService) { CustomFieldsDropdownComponent, ProfileEditDialogComponent, PdfViewerComponent, + DocumentLinkComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html b/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html index 94dc3297f..63f235b43 100644 --- a/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html +++ b/src-ui/src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.html @@ -1,16 +1,16 @@
- - - -
+ + + + diff --git a/src-ui/src/app/components/common/input/document-link/document-link.component.html b/src-ui/src/app/components/common/input/document-link/document-link.component.html new file mode 100644 index 000000000..03652a4fc --- /dev/null +++ b/src-ui/src/app/components/common/input/document-link/document-link.component.html @@ -0,0 +1,50 @@ +
+
+
+ + +
+
+
+ + + + + +
+
Loading...
+
+ +
{{document.title}} ({{document.created | customDate:'shortDate'}})
+
+
+
+ {{hint}} +
+
+
diff --git a/src-ui/src/app/components/common/input/document-link/document-link.component.scss b/src-ui/src/app/components/common/input/document-link/document-link.component.scss new file mode 100644 index 000000000..bcaa4e849 --- /dev/null +++ b/src-ui/src/app/components/common/input/document-link/document-link.component.scss @@ -0,0 +1,14 @@ +::ng-deep .ng-select-container .ng-value-container .ng-value { + background-color: transparent !important; + border-color: transparent; +} + +.sidebaricon { + cursor: pointer; +} + +.badge { + font-size: .75rem; + // --bs-primary: var(--pngx-bg-alt); + // color: var(--pngx-primary-text-contrast); +} diff --git a/src-ui/src/app/components/common/input/document-link/document-link.component.spec.ts b/src-ui/src/app/components/common/input/document-link/document-link.component.spec.ts new file mode 100644 index 000000000..d1af7ab2f --- /dev/null +++ b/src-ui/src/app/components/common/input/document-link/document-link.component.spec.ts @@ -0,0 +1,118 @@ +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { ComponentFixture, TestBed } from '@angular/core/testing' +import { + FormsModule, + NG_VALUE_ACCESSOR, + ReactiveFormsModule, +} from '@angular/forms' +import { NgSelectModule } from '@ng-select/ng-select' +import { of, throwError } from 'rxjs' +import { DocumentService } from 'src/app/services/rest/document.service' +import { DocumentLinkComponent } from './document-link.component' +import { FILTER_TITLE } from 'src/app/data/filter-rule-type' + +const documents = [ + { + id: 1, + title: 'Document 1 foo', + }, + { + id: 12, + title: 'Document 12 bar', + }, + { + id: 23, + title: 'Document 23 bar', + }, +] + +describe('DocumentLinkComponent', () => { + let component: DocumentLinkComponent + let fixture: ComponentFixture + let documentService: DocumentService + + beforeEach(() => { + TestBed.configureTestingModule({ + declarations: [DocumentLinkComponent], + imports: [ + HttpClientTestingModule, + NgSelectModule, + FormsModule, + ReactiveFormsModule, + ], + }) + documentService = TestBed.inject(DocumentService) + fixture = TestBed.createComponent(DocumentLinkComponent) + fixture.debugElement.injector.get(NG_VALUE_ACCESSOR) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should retrieve selected documents from APIs', () => { + const getSpy = jest.spyOn(documentService, 'getCachedMany') + getSpy.mockImplementation((ids) => { + return of(documents.filter((d) => ids.includes(d.id))) + }) + component.writeValue([1]) + expect(getSpy).toHaveBeenCalled() + }) + + it('should search API on select text input', () => { + const listSpy = jest.spyOn(documentService, 'listFiltered') + listSpy.mockImplementation( + (page, pageSize, sortField, sortReverse, filterRules, extraParams) => { + const docs = documents.filter((d) => + d.title.includes(filterRules[0].value) + ) + return of({ + count: docs.length, + results: docs, + all: docs.map((d) => d.id), + }) + } + ) + component.documentsInput$.next('bar') + expect(listSpy).toHaveBeenCalledWith( + 1, + null, + 'created', + true, + [{ rule_type: FILTER_TITLE, value: 'bar' }], + { truncate_content: true } + ) + listSpy.mockReturnValueOnce(throwError(() => new Error())) + component.documentsInput$.next('foo') + }) + + it('should load values correctly', () => { + jest.spyOn(documentService, 'getCachedMany').mockImplementation((ids) => { + return of(documents.filter((d) => ids.includes(d.id))) + }) + component.writeValue([12, 23]) + expect(component.value).toEqual([12, 23]) + expect(component.selectedDocuments).toEqual([documents[1], documents[2]]) + component.writeValue(null) + expect(component.value).toEqual([]) + expect(component.selectedDocuments).toEqual([]) + component.writeValue([]) + expect(component.value).toEqual([]) + expect(component.selectedDocuments).toEqual([]) + }) + + it('should support unselect', () => { + const getSpy = jest.spyOn(documentService, 'getCachedMany') + getSpy.mockImplementation((ids) => { + return of(documents.filter((d) => ids.includes(d.id))) + }) + component.writeValue([12, 23]) + component.unselect({ id: 23 }) + fixture.detectChanges() + expect(component.selectedDocuments).toEqual([documents[1]]) + }) + + it('should use correct compare, trackBy functions', () => { + expect(component.compareDocuments(documents[0], { id: 1 })).toBeTruthy() + expect(component.compareDocuments(documents[0], { id: 2 })).toBeFalsy() + expect(component.trackByFn(documents[1])).toEqual(12) + }) +}) diff --git a/src-ui/src/app/components/common/input/document-link/document-link.component.ts b/src-ui/src/app/components/common/input/document-link/document-link.component.ts new file mode 100644 index 000000000..dd7118074 --- /dev/null +++ b/src-ui/src/app/components/common/input/document-link/document-link.component.ts @@ -0,0 +1,120 @@ +import { Component, forwardRef, OnInit, Input, OnDestroy } from '@angular/core' +import { NG_VALUE_ACCESSOR } from '@angular/forms' +import { + Subject, + Observable, + takeUntil, + concat, + of, + distinctUntilChanged, + tap, + switchMap, + map, + catchError, +} from 'rxjs' +import { FILTER_TITLE } from 'src/app/data/filter-rule-type' +import { PaperlessDocument } from 'src/app/data/paperless-document' +import { DocumentService } from 'src/app/services/rest/document.service' +import { AbstractInputComponent } from '../abstract-input' + +@Component({ + providers: [ + { + provide: NG_VALUE_ACCESSOR, + useExisting: forwardRef(() => DocumentLinkComponent), + multi: true, + }, + ], + selector: 'pngx-input-document-link', + templateUrl: './document-link.component.html', + styleUrls: ['./document-link.component.scss'], +}) +export class DocumentLinkComponent + extends AbstractInputComponent + implements OnInit, OnDestroy +{ + documentsInput$ = new Subject() + foundDocuments$: Observable + loading = false + selectedDocuments: PaperlessDocument[] = [] + + private unsubscribeNotifier: Subject = new Subject() + + @Input() + notFoundText: string = $localize`No documents found` + + constructor(private documentsService: DocumentService) { + super() + } + + ngOnInit() { + this.loadDocs() + } + + writeValue(documentIDs: number[]): void { + if (!documentIDs || documentIDs.length === 0) { + this.selectedDocuments = [] + super.writeValue([]) + } else { + this.loading = true + this.documentsService + .getCachedMany(documentIDs) + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe((documents) => { + this.loading = false + this.selectedDocuments = documents + super.writeValue(documentIDs) + }) + } + } + + private loadDocs() { + this.foundDocuments$ = concat( + of([]), // default items + this.documentsInput$.pipe( + distinctUntilChanged(), + takeUntil(this.unsubscribeNotifier), + tap(() => (this.loading = true)), + switchMap((title) => + this.documentsService + .listFiltered( + 1, + null, + 'created', + true, + [{ rule_type: FILTER_TITLE, value: title }], + { truncate_content: true } + ) + .pipe( + map((results) => results.results), + catchError(() => of([])), // empty on error + tap(() => (this.loading = false)) + ) + ) + ) + ) + } + + unselect(document: PaperlessDocument): void { + this.selectedDocuments = this.selectedDocuments.filter( + (d) => d.id !== document.id + ) + this.onChange(this.selectedDocuments.map((d) => d.id)) + } + + compareDocuments( + document: PaperlessDocument, + selectedDocument: PaperlessDocument + ) { + return document.id === selectedDocument.id + } + + trackByFn(item: PaperlessDocument) { + return item.id + } + + ngOnDestroy(): void { + this.unsubscribeNotifier.next(true) + this.unsubscribeNotifier.complete() + } +} diff --git a/src-ui/src/app/components/common/input/tags/tags.component.html b/src-ui/src/app/components/common/input/tags/tags.component.html index a359bd387..3c93a167d 100644 --- a/src-ui/src/app/components/common/input/tags/tags.component.html +++ b/src-ui/src/app/components/common/input/tags/tags.component.html @@ -1,4 +1,4 @@ -
+
diff --git a/src-ui/src/app/components/document-detail/document-detail.component.html b/src-ui/src/app/components/document-detail/document-detail.component.html index 8b50e6f2e..ea14b750d 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.html +++ b/src-ui/src/app/components/document-detail/document-detail.component.html @@ -131,6 +131,7 @@ +
diff --git a/src-ui/src/app/data/paperless-custom-field.ts b/src-ui/src/app/data/paperless-custom-field.ts index 663e1507f..93bd34e33 100644 --- a/src-ui/src/app/data/paperless-custom-field.ts +++ b/src-ui/src/app/data/paperless-custom-field.ts @@ -8,6 +8,7 @@ export enum PaperlessCustomFieldDataType { Integer = 'integer', Float = 'float', Monetary = 'monetary', + DocumentLink = 'documentlink', } export const DATA_TYPE_LABELS = [ @@ -39,6 +40,10 @@ export const DATA_TYPE_LABELS = [ id: PaperlessCustomFieldDataType.Url, name: $localize`Url`, }, + { + id: PaperlessCustomFieldDataType.DocumentLink, + name: $localize`Document Link`, + }, ] export interface PaperlessCustomField extends ObjectWithId { diff --git a/src/documents/migrations/1042_consumptiontemplate_assign_custom_fields.py b/src/documents/migrations/1042_consumptiontemplate_assign_custom_fields.py deleted file mode 100644 index 08d6062ea..000000000 --- a/src/documents/migrations/1042_consumptiontemplate_assign_custom_fields.py +++ /dev/null @@ -1,23 +0,0 @@ -# Generated by Django 4.2.7 on 2023-11-30 17:44 - -from django.db import migrations -from django.db import models - - -class Migration(migrations.Migration): - dependencies = [ - ("documents", "1041_alter_consumptiontemplate_sources"), - ] - - operations = [ - migrations.AddField( - model_name="consumptiontemplate", - name="assign_custom_fields", - field=models.ManyToManyField( - blank=True, - related_name="+", - to="documents.customfield", - verbose_name="assign these custom fields", - ), - ), - ] diff --git a/src/documents/migrations/1042_consumptiontemplate_assign_custom_fields_and_more.py b/src/documents/migrations/1042_consumptiontemplate_assign_custom_fields_and_more.py new file mode 100644 index 000000000..ffd0dbefa --- /dev/null +++ b/src/documents/migrations/1042_consumptiontemplate_assign_custom_fields_and_more.py @@ -0,0 +1,47 @@ +# Generated by Django 4.2.7 on 2023-12-04 04:03 + +from django.db import migrations +from django.db import models + + +class Migration(migrations.Migration): + dependencies = [ + ("documents", "1041_alter_consumptiontemplate_sources"), + ] + + operations = [ + migrations.AddField( + model_name="consumptiontemplate", + name="assign_custom_fields", + field=models.ManyToManyField( + blank=True, + related_name="+", + to="documents.customfield", + verbose_name="assign these custom fields", + ), + ), + migrations.AddField( + model_name="customfieldinstance", + name="value_document_ids", + field=models.JSONField(null=True), + ), + migrations.AlterField( + model_name="customfield", + name="data_type", + field=models.CharField( + choices=[ + ("string", "String"), + ("url", "URL"), + ("date", "Date"), + ("boolean", "Boolean"), + ("integer", "Integer"), + ("float", "Float"), + ("monetary", "Monetary"), + ("documentlink", "Document Link"), + ], + editable=False, + max_length=50, + verbose_name="data type", + ), + ), + ] diff --git a/src/documents/models.py b/src/documents/models.py index d688253de..250a9d35b 100644 --- a/src/documents/models.py +++ b/src/documents/models.py @@ -756,6 +756,7 @@ class CustomField(models.Model): INT = ("integer", _("Integer")) FLOAT = ("float", _("Float")) MONETARY = ("monetary", _("Monetary")) + DOCUMENTLINK = ("documentlink", _("Document Link")) created = models.DateTimeField( _("created"), @@ -834,6 +835,8 @@ class CustomFieldInstance(models.Model): value_monetary = models.DecimalField(null=True, decimal_places=2, max_digits=12) + value_document_ids = models.JSONField(null=True) + class Meta: ordering = ("created",) verbose_name = _("custom field instance") @@ -868,6 +871,8 @@ class CustomFieldInstance(models.Model): return self.value_float elif self.field.data_type == CustomField.FieldDataType.MONETARY: return self.value_monetary + elif self.field.data_type == CustomField.FieldDataType.DOCUMENTLINK: + return self.value_document_ids raise NotImplementedError(self.field.data_type) diff --git a/src/documents/serialisers.py b/src/documents/serialisers.py index 2373a25dd..f01d1fc3a 100644 --- a/src/documents/serialisers.py +++ b/src/documents/serialisers.py @@ -440,6 +440,7 @@ class CustomFieldInstanceSerializer(serializers.ModelSerializer): CustomField.FieldDataType.INT: "value_int", CustomField.FieldDataType.FLOAT: "value_float", CustomField.FieldDataType.MONETARY: "value_monetary", + CustomField.FieldDataType.DOCUMENTLINK: "value_document_ids", } # An instance is attached to a document document: Document = validated_data["document"] diff --git a/src/documents/tests/test_api_custom_fields.py b/src/documents/tests/test_api_custom_fields.py index 725bd9254..cde5f302c 100644 --- a/src/documents/tests/test_api_custom_fields.py +++ b/src/documents/tests/test_api_custom_fields.py @@ -34,7 +34,9 @@ class TestCustomField(DirectoriesMixin, APITestCase): ("date", "Invoiced Date"), ("integer", "Invoice #"), ("boolean", "Is Active"), - ("float", "Total Paid"), + ("float", "Average Value"), + ("monetary", "Total Paid"), + ("documentlink", "Related Documents"), ]: resp = self.client.post( self.ENDPOINT, @@ -96,6 +98,10 @@ class TestCustomField(DirectoriesMixin, APITestCase): name="Test Custom Field Monetary", data_type=CustomField.FieldDataType.MONETARY, ) + custom_field_documentlink = CustomField.objects.create( + name="Test Custom Field Doc Link", + data_type=CustomField.FieldDataType.DOCUMENTLINK, + ) date_value = date.today() @@ -131,6 +137,10 @@ class TestCustomField(DirectoriesMixin, APITestCase): "field": custom_field_monetary.id, "value": 11.10, }, + { + "field": custom_field_documentlink.id, + "value": [1, 2, 3], + }, ], }, format="json", @@ -150,11 +160,12 @@ class TestCustomField(DirectoriesMixin, APITestCase): {"field": custom_field_url.id, "value": "https://example.com"}, {"field": custom_field_float.id, "value": 12.3456}, {"field": custom_field_monetary.id, "value": 11.10}, + {"field": custom_field_documentlink.id, "value": [1, 2, 3]}, ], ) doc.refresh_from_db() - self.assertEqual(len(doc.custom_fields.all()), 7) + self.assertEqual(len(doc.custom_fields.all()), 8) def test_change_custom_field_instance_value(self): """