From 3e62f13f96665511b4ad1981d47546d53c53cd61 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Wed, 22 May 2024 16:01:15 -0700 Subject: [PATCH] Enhancement: delete pages PDF action (#6772) --- docs/api.md | 4 + docs/usage.md | 9 +- src-ui/messages.xlf | 270 +++++++++++------- src-ui/src/app/app.module.ts | 6 + ...delete-pages-confirm-dialog.component.html | 54 ++++ ...delete-pages-confirm-dialog.component.scss | 28 ++ ...ete-pages-confirm-dialog.component.spec.ts | 55 ++++ .../delete-pages-confirm-dialog.component.ts | 64 +++++ .../rotate-confirm-dialog.component.html | 20 +- .../document-detail.component.html | 12 +- .../document-detail.component.spec.ts | 37 ++- .../document-detail.component.ts | 63 +++- src/documents/bulk_edit.py | 26 ++ src/documents/serialisers.py | 17 ++ src/documents/tests/test_api_bulk_edit.py | 92 ++++++ src/documents/tests/test_bulk_edit.py | 43 +++ 16 files changed, 658 insertions(+), 142 deletions(-) create mode 100644 src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html create mode 100644 src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.scss create mode 100644 src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.spec.ts create mode 100644 src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.ts diff --git a/docs/api.md b/docs/api.md index 160b7c07e..07714e690 100644 --- a/docs/api.md +++ b/docs/api.md @@ -424,6 +424,10 @@ The following methods are supported: - `rotate` - Requires `parameters`: - `"degrees": DEGREES`. Must be an integer i.e. 90, 180, 270 +- `delete_pages` + - Requires `parameters`: + - `"pages": [..]` The list should be a list of integers e.g. `"[2,3,4]"` + - The delete_pages operation only accepts a single document. ### Objects diff --git a/docs/usage.md b/docs/usage.md index 52713fb86..8706a7911 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -462,15 +462,16 @@ Paperless-ngx added the ability to create shareable links to files in version 2. ## PDF Actions -Paperless-ngx supports 3 basic editing operations for PDFs (these operations cannot be performed on non-PDF files): +Paperless-ngx supports four basic editing operations for PDFs (these operations currently cannot be performed on non-PDF files): -- Merging documents: available when selecting multiple documents for 'bulk editing' +- Merging documents: available when selecting multiple documents for 'bulk editing'. - Rotating documents: available when selecting multiple documents for 'bulk editing' and from an individual document's details page. -- Splitting documents: available from an individual document's details page +- Splitting documents: available from an individual document's details page. +- Deleting pages: available from an individual document's details page. !!! important - Note that rotation alters the Paperless-ngx _original_ file, which would, for example, invalidate a digital signature. + Note that rotation and deleting pages alter the Paperless-ngx _original_ file, which would, for example, invalidate a digital signature. ## Document History diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 502b29e30..9af66a45d 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -368,7 +368,7 @@ src/app/components/document-detail/document-detail.component.html - 91 + 95 @@ -540,7 +540,7 @@ src/app/components/document-detail/document-detail.component.html - 333 + 337 @@ -599,7 +599,7 @@ src/app/components/document-detail/document-detail.component.html - 325 + 329 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -729,7 +729,7 @@ src/app/components/document-detail/document-detail.component.html - 342 + 346 src/app/components/document-list/document-list.component.html @@ -1069,7 +1069,7 @@ src/app/components/document-detail/document-detail.component.html - 301 + 305 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1567,7 +1567,7 @@ src/app/components/document-detail/document-detail.component.html - 113 + 117 @@ -2249,7 +2249,7 @@ src/app/components/document-detail/document-detail.component.ts - 809 + 818 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2292,15 +2292,19 @@ src/app/components/document-detail/document-detail.component.ts - 811 + 820 src/app/components/document-detail/document-detail.component.ts - 1104 + 1113 src/app/components/document-detail/document-detail.component.ts - 1142 + 1150 + + + src/app/components/document-detail/document-detail.component.ts + 1191 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2909,6 +2913,47 @@ 579 + + Page + + src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html + 11 + + + src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html + 11 + + + src/app/components/document-detail/document-detail.component.html + 5 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 11 + + + + of + + src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html + 13 + + + src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html + 13 + + + src/app/components/document-detail/document-detail.component.html + 7,8 + + + + Pages to remove + + src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html + 16 + + Documents: @@ -2941,33 +2986,7 @@ Note that only PDFs will be rotated. src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html - 35 - - - - Page - - src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html - 11 - - - src/app/components/document-detail/document-detail.component.html - 5 - - - src/app/components/document-list/bulk-editor/bulk-editor.component.html - 11 - - - - of - - src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html - 13 - - - src/app/components/document-detail/document-detail.component.html - 7,8 + 25 @@ -5383,15 +5402,22 @@ 114 + + Delete page(s) + + src/app/components/document-detail/document-detail.component.html + 65 + + Close src/app/components/document-detail/document-detail.component.html - 85 + 89 src/app/components/document-detail/document-detail.component.ts - 1160 + 1168 src/app/guards/dirty-saved-view.guard.ts @@ -5402,21 +5428,21 @@ Previous src/app/components/document-detail/document-detail.component.html - 88 + 92 Details src/app/components/document-detail/document-detail.component.html - 101 + 105 Title src/app/components/document-detail/document-detail.component.html - 104 + 108 src/app/components/document-list/document-list.component.html @@ -5439,21 +5465,21 @@ Archive serial number src/app/components/document-detail/document-detail.component.html - 105 + 109 Date created src/app/components/document-detail/document-detail.component.html - 106 + 110 Correspondent src/app/components/document-detail/document-detail.component.html - 108 + 112 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -5480,7 +5506,7 @@ Document type src/app/components/document-detail/document-detail.component.html - 110 + 114 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -5507,7 +5533,7 @@ Storage path src/app/components/document-detail/document-detail.component.html - 112 + 116 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -5530,14 +5556,14 @@ Content src/app/components/document-detail/document-detail.component.html - 197 + 201 Metadata src/app/components/document-detail/document-detail.component.html - 206 + 210 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -5548,175 +5574,175 @@ Date modified src/app/components/document-detail/document-detail.component.html - 213 + 217 Date added src/app/components/document-detail/document-detail.component.html - 217 + 221 Media filename src/app/components/document-detail/document-detail.component.html - 221 + 225 Original filename src/app/components/document-detail/document-detail.component.html - 225 + 229 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 229 + 233 Original file size src/app/components/document-detail/document-detail.component.html - 233 + 237 Original mime type src/app/components/document-detail/document-detail.component.html - 237 + 241 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 242 + 246 Archive file size src/app/components/document-detail/document-detail.component.html - 248 + 252 Original document metadata src/app/components/document-detail/document-detail.component.html - 257 + 261 Archived document metadata src/app/components/document-detail/document-detail.component.html - 260 + 264 Preview src/app/components/document-detail/document-detail.component.html - 267 + 271 Notes src/app/components/document-detail/document-detail.component.html - 279,282 + 283,286 History src/app/components/document-detail/document-detail.component.html - 290 + 294 Save & next src/app/components/document-detail/document-detail.component.html - 327 + 331 Save & close src/app/components/document-detail/document-detail.component.html - 330 + 334 Enter Password src/app/components/document-detail/document-detail.component.html - 381 + 385 An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 330,332 + 339,341 Document changes detected src/app/components/document-detail/document-detail.component.ts - 353 + 362 The version of this document in your browser session appears older than the existing version. src/app/components/document-detail/document-detail.component.ts - 354 + 363 Saving the document here may overwrite other changes that were made. To restore the existing version, discard your changes or close the document. src/app/components/document-detail/document-detail.component.ts - 355 + 364 Ok src/app/components/document-detail/document-detail.component.ts - 357 + 366 Next document src/app/components/document-detail/document-detail.component.ts - 464 + 473 Previous document src/app/components/document-detail/document-detail.component.ts - 474 + 483 Close document src/app/components/document-detail/document-detail.component.ts - 482 + 491 src/app/services/open-documents.service.ts @@ -5727,50 +5753,50 @@ Save document src/app/components/document-detail/document-detail.component.ts - 489 + 498 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 531 + 540 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 556 + 565 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 678 + 687 src/app/components/document-detail/document-detail.component.ts - 692 + 701 Error saving document src/app/components/document-detail/document-detail.component.ts - 696 + 705 src/app/components/document-detail/document-detail.component.ts - 737 + 746 Confirm delete src/app/components/document-detail/document-detail.component.ts - 764 + 773 src/app/components/manage/management-list/management-list.component.ts @@ -5785,35 +5811,35 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 765 + 774 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 766 + 775 Delete document src/app/components/document-detail/document-detail.component.ts - 768 + 777 Error deleting document src/app/components/document-detail/document-detail.component.ts - 787 + 796 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 807 + 816 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -5824,63 +5850,63 @@ This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 808 + 817 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 - 819 + 828 Error executing operation src/app/components/document-detail/document-detail.component.ts - 830 + 839 Page Fit src/app/components/document-detail/document-detail.component.ts - 899 + 908 Split confirm src/app/components/document-detail/document-detail.component.ts - 1102 + 1111 This operation will split the selected document(s) into new documents. src/app/components/document-detail/document-detail.component.ts - 1103 + 1112 Split operation will begin in the background. src/app/components/document-detail/document-detail.component.ts - 1118 + 1127 Error executing split operation src/app/components/document-detail/document-detail.component.ts - 1127 + 1136 Rotate confirm src/app/components/document-detail/document-detail.component.ts - 1139 + 1148 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -5891,32 +5917,49 @@ This operation will permanently rotate the original version of the current document. src/app/components/document-detail/document-detail.component.ts - 1140 - - - - This will alter the original copy. - - src/app/components/document-detail/document-detail.component.ts - 1141 - - - src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 786 + 1149 Rotation will begin in the background. Close and re-open the document after the operation has completed to see the changes. src/app/components/document-detail/document-detail.component.ts - 1157 + 1165 Error executing rotate operation src/app/components/document-detail/document-detail.component.ts - 1169 + 1177 + + + + Delete pages confirm + + src/app/components/document-detail/document-detail.component.ts + 1189 + + + + This operation will permanently delete the selected pages from the original document. + + src/app/components/document-detail/document-detail.component.ts + 1190 + + + + Delete pages operation will begin in the background. Close and re-open or reload this document after the operation has completed to see the changes. + + src/app/components/document-detail/document-detail.component.ts + 1205 + + + + Error executing delete pages operation + + src/app/components/document-detail/document-detail.component.ts + 1214 @@ -6277,6 +6320,13 @@ 785 + + This will alter the original copy. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 786 + + Merge confirm diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index bb5a62249..f9e04b069 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -124,6 +124,7 @@ import { DragDropSelectComponent } from './components/common/input/drag-drop-sel import { CustomFieldDisplayComponent } from './components/common/custom-field-display/custom-field-display.component' import { GlobalSearchComponent } from './components/app-frame/global-search/global-search.component' import { HotkeyDialogComponent } from './components/common/hotkey-dialog/hotkey-dialog.component' +import { DeletePagesConfirmDialogComponent } from './components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component' import { airplane, archive, @@ -160,6 +161,7 @@ import { clipboardCheckFill, clipboardFill, dash, + dashCircle, diagram3, dice5, doorOpen, @@ -174,6 +176,7 @@ import { fileEarmarkCheck, fileEarmarkFill, fileEarmarkLock, + fileEarmarkMinus, files, fileText, filter, @@ -259,6 +262,7 @@ const icons = { clipboardCheckFill, clipboardFill, dash, + dashCircle, diagram3, dice5, doorOpen, @@ -273,6 +277,7 @@ const icons = { fileEarmarkCheck, fileEarmarkFill, fileEarmarkLock, + fileEarmarkMinus, files, fileText, filter, @@ -491,6 +496,7 @@ function initializeApp(settings: SettingsService) { CustomFieldDisplayComponent, GlobalSearchComponent, HotkeyDialogComponent, + DeletePagesConfirmDialogComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html new file mode 100644 index 000000000..01bf5d3fd --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.html @@ -0,0 +1,54 @@ + + + + + +
+ +
+
diff --git a/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.scss b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.scss new file mode 100644 index 000000000..f74de973d --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.scss @@ -0,0 +1,28 @@ +.pdf-viewer-container { + background-color: gray; + height: 350px; + + pdf-viewer { + width: 100%; + height: 100%; + } +} + +.mw-60 { + max-width: 60px; +} + +div.position-absolute:has(.form-check-input:checked) { + background-color: rgba(var(--bs-dark-rgb), 0.4); +} + +.form-check-input { + &:checked { + background-color: var(--bs-danger); + border-color: var(--bs-danger); + } + &:focus { + box-shadow: 0 0 0 0.25rem rgba(var(--bs-danger-rgb), var(--pngx-focus-alpha)); + border-color: var(--bs-danger); + } +} diff --git a/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.spec.ts b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.spec.ts new file mode 100644 index 000000000..78fab8c8d --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.spec.ts @@ -0,0 +1,55 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' +import { DeletePagesConfirmDialogComponent } from './delete-pages-confirm-dialog.component' +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' +import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' +import { FormsModule, ReactiveFormsModule } from '@angular/forms' +import { PdfViewerComponent } from 'ng2-pdf-viewer' + +describe('DeletePagesConfirmDialogComponent', () => { + let component: DeletePagesConfirmDialogComponent + let fixture: ComponentFixture + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [DeletePagesConfirmDialogComponent, PdfViewerComponent], + providers: [NgbActiveModal, SafeHtmlPipe], + imports: [ + HttpClientTestingModule, + NgxBootstrapIconsModule.pick(allIcons), + FormsModule, + ReactiveFormsModule, + ], + }).compileComponents() + fixture = TestBed.createComponent(DeletePagesConfirmDialogComponent) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should return a string with comma-separated pages', () => { + component.pages = [1, 2, 3, 4] + expect(component.pagesString).toEqual('1, 2, 3, 4') + }) + + it('should update totalPages when pdf is loaded', () => { + component.pdfPreviewLoaded({ numPages: 5 } as any) + expect(component.totalPages).toEqual(5) + }) + + it('should update checks when page is rendered', () => { + const event = { + target: document.createElement('div'), + detail: { pageNumber: 1 }, + } as any + component.pageRendered(event) + expect(component['checks'].length).toEqual(1) + }) + + it('should update pages when page check is changed', () => { + component.pageCheckChanged(1) + expect(component.pages).toEqual([1]) + component.pageCheckChanged(1) + expect(component.pages).toEqual([]) + }) +}) diff --git a/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.ts b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.ts new file mode 100644 index 000000000..c47dea0ed --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component.ts @@ -0,0 +1,64 @@ +import { Component, TemplateRef, ViewChild } from '@angular/core' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { DocumentService } from 'src/app/services/rest/document.service' +import { ConfirmDialogComponent } from '../confirm-dialog.component' +import { PDFDocumentProxy, PdfViewerComponent } from 'ng2-pdf-viewer' + +@Component({ + selector: 'pngx-delete-pages-confirm-dialog', + templateUrl: './delete-pages-confirm-dialog.component.html', + styleUrl: './delete-pages-confirm-dialog.component.scss', +}) +export class DeletePagesConfirmDialogComponent extends ConfirmDialogComponent { + public documentID: number + public pages: number[] = [] + public currentPage: number = 1 + public totalPages: number + + @ViewChild('pdfViewer') pdfViewer: PdfViewerComponent + @ViewChild('pageCheckOverlay') pageCheckOverlay!: TemplateRef + private checks: HTMLElement[] = [] + + public get pagesString(): string { + return this.pages.join(', ') + } + + public get pdfSrc(): string { + return this.documentService.getPreviewUrl(this.documentID) + } + + constructor( + activeModal: NgbActiveModal, + private documentService: DocumentService + ) { + super(activeModal) + } + + public pdfPreviewLoaded(pdf: PDFDocumentProxy) { + this.totalPages = pdf.numPages + } + + pageRendered(event: CustomEvent) { + const pageDiv = event.target as HTMLDivElement + const check = this.pageCheckOverlay.createEmbeddedView({ + page: event.detail.pageNumber, + }) + this.checks[event.detail.pageNumber - 1] = check.rootNodes[0] + pageDiv?.insertBefore(check.rootNodes[0], pageDiv.firstChild) + this.updateChecks() + } + + pageCheckChanged(pageNumber: number) { + if (!this.pages.includes(pageNumber)) this.pages.push(pageNumber) + else if (this.pages.includes(pageNumber)) + this.pages.splice(this.pages.indexOf(pageNumber), 1) + this.updateChecks() + } + + private updateChecks() { + this.checks.forEach((check, i) => { + const input = check.getElementsByTagName('input')[0] + input.checked = this.pages.includes(i + 1) + }) + } +} diff --git a/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html index 00e8996d0..e996ecb44 100644 --- a/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html +++ b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html @@ -21,21 +21,19 @@ -
-
- @if (messageBold) { -

{{messageBold}}

- } - @if (message) { -

- } -
-
@if (showPDFNote) {

Note that only PDFs will be rotated.

} - } @else { - @switch (contentRenderType) { + @switch (archiveContentRenderType) { @case (ContentRenderType.PDF) { @if (!useNativePdfViewer) {
diff --git a/src-ui/src/app/components/document-detail/document-detail.component.spec.ts b/src-ui/src/app/components/document-detail/document-detail.component.spec.ts index d27c13ef1..b8a6389f2 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.spec.ts +++ b/src-ui/src/app/components/document-detail/document-detail.component.spec.ts @@ -81,6 +81,7 @@ import { environment } from 'src/environments/environment' import { RotateConfirmDialogComponent } from '../common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' import { SplitConfirmDialogComponent } from '../common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component' import { PdfViewerModule } from 'ng2-pdf-viewer' +import { DeletePagesConfirmDialogComponent } from '../common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component' const doc: Document = { id: 3, @@ -178,6 +179,7 @@ describe('DocumentDetailComponent', () => { CustomFieldsDropdownComponent, SplitConfirmDialogComponent, RotateConfirmDialogComponent, + DeletePagesConfirmDialogComponent, ], providers: [ DocumentTitlePipe, @@ -1035,7 +1037,9 @@ describe('DocumentDetailComponent', () => { component.metadata = { has_archive_version: true } initNormally() fixture.detectChanges() - expect(component.contentRenderType).toEqual(component.ContentRenderType.PDF) + expect(component.archiveContentRenderType).toEqual( + component.ContentRenderType.PDF + ) expect( fixture.debugElement.query(By.css('pdf-viewer-container')) ).not.toBeUndefined() @@ -1045,7 +1049,7 @@ describe('DocumentDetailComponent', () => { original_mime_type: 'text/plain', } fixture.detectChanges() - expect(component.contentRenderType).toEqual( + expect(component.archiveContentRenderType).toEqual( component.ContentRenderType.Text ) expect( @@ -1057,7 +1061,7 @@ describe('DocumentDetailComponent', () => { original_mime_type: 'image/jpg', } fixture.detectChanges() - expect(component.contentRenderType).toEqual( + expect(component.archiveContentRenderType).toEqual( component.ContentRenderType.Image ) expect( @@ -1070,7 +1074,7 @@ describe('DocumentDetailComponent', () => { 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', } fixture.detectChanges() - expect(component.contentRenderType).toEqual( + expect(component.archiveContentRenderType).toEqual( component.ContentRenderType.Other ) expect( @@ -1130,6 +1134,31 @@ describe('DocumentDetailComponent', () => { req.flush(true) }) + it('should support delete pages', () => { + let modal: NgbModalRef + modalService.activeInstances.subscribe((m) => (modal = m[0])) + initNormally() + component.deletePages() + expect(modal).not.toBeUndefined() + modal.componentInstance.documentID = doc.id + modal.componentInstance.pages = [1, 2] + modal.componentInstance.confirm() + let req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + expect(req.request.body).toEqual({ + documents: [doc.id], + method: 'delete_pages', + parameters: { pages: [1, 2] }, + }) + req.error(new ProgressEvent('failed')) + modal.componentInstance.confirm() + req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + req.flush(true) + }) + it('should support keyboard shortcuts', () => { initNormally() 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 820d7fbd5..23753f55b 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 @@ -68,6 +68,7 @@ import { CustomFieldInstance } from 'src/app/data/custom-field-instance' import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' import { SplitConfirmDialogComponent } from '../common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component' import { RotateConfirmDialogComponent } from '../common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' +import { DeletePagesConfirmDialogComponent } from '../common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component' import { HotKeyService } from 'src/app/services/hot-key.service' import { PDFDocumentProxy } from 'ng2-pdf-viewer' @@ -216,19 +217,27 @@ export class DocumentDetailComponent return this.settings.get(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER) } - get contentRenderType(): ContentRenderType { - if (!this.metadata) return ContentRenderType.Unknown - const contentType = this.metadata?.has_archive_version - ? 'application/pdf' - : this.metadata?.original_mime_type + get archiveContentRenderType(): ContentRenderType { + return this.getRenderType( + this.metadata?.has_archive_version + ? 'application/pdf' + : this.metadata?.original_mime_type + ) + } - if (contentType === 'application/pdf') { + get originalContentRenderType(): ContentRenderType { + return this.getRenderType(this.metadata?.original_mime_type) + } + + private getRenderType(mimeType: string): ContentRenderType { + if (!mimeType) return ContentRenderType.Unknown + if (mimeType === 'application/pdf') { return ContentRenderType.PDF } else if ( - ['text/plain', 'application/csv', 'text/csv'].includes(contentType) + ['text/plain', 'application/csv', 'text/csv'].includes(mimeType) ) { return ContentRenderType.Text - } else if (contentType?.indexOf('image/') === 0) { + } else if (mimeType?.indexOf('image/') === 0) { return ContentRenderType.Image } return ContentRenderType.Other @@ -1138,7 +1147,6 @@ export class DocumentDetailComponent }) modal.componentInstance.title = $localize`Rotate confirm` modal.componentInstance.messageBold = $localize`This operation will permanently rotate the original version of the current document.` - modal.componentInstance.message = $localize`This will alter the original copy.` modal.componentInstance.btnCaption = $localize`Proceed` modal.componentInstance.documentID = this.document.id modal.componentInstance.showPDFNote = false @@ -1173,4 +1181,41 @@ export class DocumentDetailComponent }) }) } + + deletePages() { + let modal = this.modalService.open(DeletePagesConfirmDialogComponent, { + backdrop: 'static', + }) + modal.componentInstance.title = $localize`Delete pages confirm` + modal.componentInstance.messageBold = $localize`This operation will permanently delete the selected pages from the original document.` + modal.componentInstance.btnCaption = $localize`Proceed` + modal.componentInstance.documentID = this.document.id + modal.componentInstance.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + modal.componentInstance.buttonsEnabled = false + this.documentsService + .bulkEdit([this.document.id], 'delete_pages', { + pages: modal.componentInstance.pages, + }) + .pipe(first(), takeUntil(this.unsubscribeNotifier)) + .subscribe({ + next: () => { + this.toastService.showInfo( + $localize`Delete pages operation will begin in the background. Close and re-open or reload this document after the operation has completed to see the changes.` + ) + modal.close() + }, + error: (error) => { + if (modal) { + modal.componentInstance.buttonsEnabled = true + } + this.toastService.showError( + $localize`Error executing delete pages operation`, + error + ) + }, + }) + }) + } } diff --git a/src/documents/bulk_edit.py b/src/documents/bulk_edit.py index f59ef1af3..8dbdbc2dd 100644 --- a/src/documents/bulk_edit.py +++ b/src/documents/bulk_edit.py @@ -325,3 +325,29 @@ def split(doc_ids: list[int], pages: list[list[int]]): logger.exception(f"Error splitting document {doc.id}: {e}") return "OK" + + +def delete_pages(doc_ids: list[int], pages: list[int]): + logger.info( + f"Attempting to delete pages {pages} from {len(doc_ids)} documents", + ) + doc = Document.objects.get(id=doc_ids[0]) + pages = sorted(pages) # sort pages to avoid index issues + import pikepdf + + try: + with pikepdf.open(doc.source_path, allow_overwriting_input=True) as pdf: + offset = 1 # pages are 1-indexed + for page_num in pages: + pdf.pages.remove(pdf.pages[page_num - offset]) + offset += 1 # remove() changes the index of the pages + pdf.remove_unreferenced_resources() + pdf.save() + doc.checksum = hashlib.md5(doc.source_path.read_bytes()).hexdigest() + doc.save() + update_document_archive_file.delay(document_id=doc.id) + logger.info(f"Deleted pages {pages} from document {doc.id}") + except Exception as e: + logger.exception(f"Error deleting pages from document {doc.id}: {e}") + + return "OK" diff --git a/src/documents/serialisers.py b/src/documents/serialisers.py index 9d722ca5d..c92765e69 100644 --- a/src/documents/serialisers.py +++ b/src/documents/serialisers.py @@ -944,6 +944,7 @@ class BulkEditSerializer( "rotate", "merge", "split", + "delete_pages", ], label="Method", write_only=True, @@ -1000,6 +1001,8 @@ class BulkEditSerializer( return bulk_edit.merge elif method == "split": return bulk_edit.split + elif method == "delete_pages": + return bulk_edit.delete_pages else: raise serializers.ValidationError("Unsupported method.") @@ -1128,6 +1131,14 @@ class BulkEditSerializer( except ValueError: raise serializers.ValidationError("invalid pages specified") + def _validate_parameters_delete_pages(self, parameters): + if "pages" not in parameters: + raise serializers.ValidationError("pages not specified") + if not isinstance(parameters["pages"], list): + raise serializers.ValidationError("pages must be a list") + if not all(isinstance(i, int) for i in parameters["pages"]): + raise serializers.ValidationError("pages must be a list of integers") + def validate(self, attrs): method = attrs["method"] parameters = attrs["parameters"] @@ -1154,6 +1165,12 @@ class BulkEditSerializer( "Split method only supports one document", ) self._validate_parameters_split(parameters) + elif method == bulk_edit.delete_pages: + if len(attrs["documents"]) > 1: + raise serializers.ValidationError( + "Delete pages method only supports one document", + ) + self._validate_parameters_delete_pages(parameters) return attrs diff --git a/src/documents/tests/test_api_bulk_edit.py b/src/documents/tests/test_api_bulk_edit.py index c38ed8cfd..7078aca12 100644 --- a/src/documents/tests/test_api_bulk_edit.py +++ b/src/documents/tests/test_api_bulk_edit.py @@ -1065,3 +1065,95 @@ class TestBulkEditAPI(DirectoriesMixin, APITestCase): self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) self.assertIn(b"Split method only supports one document", response.content) + + @mock.patch("documents.serialisers.bulk_edit.delete_pages") + def test_delete_pages(self, m): + m.return_value = "OK" + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "delete_pages", + "parameters": {"pages": [1, 2, 3, 4]}, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_200_OK) + + m.assert_called_once() + args, kwargs = m.call_args + self.assertCountEqual(args[0], [self.doc2.id]) + self.assertEqual(kwargs["pages"], [1, 2, 3, 4]) + + def test_delete_pages_invalid_params(self): + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [ + self.doc1.id, + self.doc2.id, + ], # only one document supported + "method": "delete_pages", + "parameters": { + "pages": [1, 2, 3, 4], + }, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn( + b"Delete pages method only supports one document", + response.content, + ) + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "delete_pages", + "parameters": {}, # pages not specified + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn(b"pages not specified", response.content) + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "delete_pages", + "parameters": {"pages": "1-3"}, # not a list + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn(b"pages must be a list", response.content) + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "delete_pages", + "parameters": {"pages": ["1-3"]}, # not ints + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn(b"pages must be a list of integers", response.content) diff --git a/src/documents/tests/test_bulk_edit.py b/src/documents/tests/test_bulk_edit.py index 831fa9461..16579c887 100644 --- a/src/documents/tests/test_bulk_edit.py +++ b/src/documents/tests/test_bulk_edit.py @@ -585,3 +585,46 @@ class TestPDFActions(DirectoriesMixin, TestCase): mock_update_documents.assert_called_once() mock_chord.assert_called_once() self.assertEqual(result, "OK") + + @mock.patch("documents.tasks.update_document_archive_file.delay") + @mock.patch("pikepdf.Pdf.save") + def test_delete_pages(self, mock_pdf_save, mock_update_archive_file): + """ + GIVEN: + - Existing documents + WHEN: + - Delete pages action is called with 1 document and 2 pages + THEN: + - Save should be called once + - Archive file should be updated once + """ + doc_ids = [self.doc2.id] + pages = [1, 3] + result = bulk_edit.delete_pages(doc_ids, pages) + mock_pdf_save.assert_called_once() + mock_update_archive_file.assert_called_once() + self.assertEqual(result, "OK") + + @mock.patch("documents.tasks.update_document_archive_file.delay") + @mock.patch("pikepdf.Pdf.save") + def test_delete_pages_with_error(self, mock_pdf_save, mock_update_archive_file): + """ + GIVEN: + - Existing documents + WHEN: + - Delete pages action is called with 1 document and 2 pages + - PikePDF raises an error + THEN: + - Save should be called once + - Archive file should not be updated + """ + mock_pdf_save.side_effect = Exception("Error saving PDF") + doc_ids = [self.doc2.id] + pages = [1, 3] + + with self.assertLogs("paperless.bulk_edit", level="ERROR") as cm: + bulk_edit.delete_pages(doc_ids, pages) + error_str = cm.output[0] + expected_str = "Error deleting pages from document" + self.assertIn(expected_str, error_str) + mock_update_archive_file.assert_not_called()