diff --git a/Pipfile b/Pipfile index da26987cf..948e22273 100644 --- a/Pipfile +++ b/Pipfile @@ -17,6 +17,7 @@ django-extensions = "*" django-filter = "~=24.2" django-guardian = "*" django-multiselectfield = "*" +django-soft-delete = "*" djangorestframework = "==3.14.0" djangorestframework-guardian = "*" drf-writable-nested = "*" diff --git a/Pipfile.lock b/Pipfile.lock index f3753a0cd..e011e1c60 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -540,6 +540,15 @@ "index": "pypi", "version": "==0.1.12" }, + "django-soft-delete": { + "hashes": [ + "sha256:443c00a54c06d236ff8806c3260243d775cc536581d7377c2785080b1041ce1d", + "sha256:7cb4524231763a70ad79cfccd49d001b7e5fa666ec897cc044d897dd73e0146e" + ], + "index": "pypi", + "markers": "python_version >= '3.6'", + "version": "==1.0.13" + }, "djangorestframework": { "hashes": [ "sha256:579a333e6256b09489cbe0a067e66abe55c6595d8926be6b99423786334350c8", diff --git a/docs/configuration.md b/docs/configuration.md index f4c271ce1..0c3345145 100644 --- a/docs/configuration.md +++ b/docs/configuration.md @@ -219,10 +219,10 @@ database, classification model, etc). Defaults to "../data/", relative to the "src" directory. -#### [`PAPERLESS_TRASH_DIR=`](#PAPERLESS_TRASH_DIR) {#PAPERLESS_TRASH_DIR} +#### [`PAPERLESS_EMPTY_TRASH_DIR=`](#PAPERLESS_EMPTY_TRASH_DIR) {#PAPERLESS_EMPTY_TRASH_DIR} -: Instead of removing deleted documents, they are moved to this -directory. +: When documents are deleted (e.g. after emptying the trash) the original files will be moved here +instead of being removed from the filesystem. Only the original version is kept. This must be writeable by the user running paperless. When running inside docker, ensure that this path is within a permanent volume @@ -230,7 +230,9 @@ directory. Note that the directory must exist prior to using this setting. - Defaults to empty (i.e. really delete documents). + Defaults to empty (i.e. really delete files). + + This setting was previously named PAPERLESS_TRASH_DIR. #### [`PAPERLESS_MEDIA_ROOT=`](#PAPERLESS_MEDIA_ROOT) {#PAPERLESS_MEDIA_ROOT} @@ -1362,6 +1364,20 @@ processing. This only has an effect if Defaults to false. +## Trash + +#### [`EMPTY_TRASH_DELAY=`](#EMPTY_TRASH_DELAY) {#EMPTY_TRASH_DELAY} + +: Sets how long in days documents remain in the 'trash' before they are permanently deleted. + + Defaults to 30 days, minimum of 1 day. + +#### [`PAPERLESS_EMPTY_TRASH_TASK_CRON=`](#PAPERLESS_EMPTY_TRASH_TASK_CRON) {#PAPERLESS_EMPTY_TRASH_TASK_CRON} + +: Configures the schedule to empty the trash of expired deleted documents. + + Defaults to `0 1 * * *`, once per day. + ## Binaries There are a few external software packages that Paperless expects to diff --git a/docs/usage.md b/docs/usage.md index 5705be3da..034447d6e 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -478,6 +478,15 @@ As of version 2.7, Paperless-ngx automatically records all changes to a document Changes to documents are visible under the "History" tab. Note that certain changes such as those made by workflows, record the 'actor' as "System". +## Document Trash + +When you first delete a document it is moved to the 'trash' until either it is explicitly deleted or it is automatically removed after a set amount of time has passed. +You can set how long documents remain in the trash before being automatically deleted with [`EMPTY_TRASH_DELAY`](configuration.md#EMPTY_TRASH_DELAY), which defaults +to 30 days. Until the file is actually deleted (e.g. the trash is emptied), all files and database content remains intact and can be restored at any point up until that time. + +Additionally you may configure a directory where deleted files are moved to when they the trash is emptied with [`PAPERLESS_EMPTY_TRASH_DIR`](configuration.md#PAPERLESS_EMPTY_TRASH_DIR). +Note that the empty trash directory only stores the original file, the archive file and all database information is permanently removed once a document is fully deleted. + ## Best practices {#basic-searching} Paperless offers a couple tools that help you organize your document diff --git a/paperless.conf.example b/paperless.conf.example index db557a7b6..63ee7be22 100644 --- a/paperless.conf.example +++ b/paperless.conf.example @@ -19,7 +19,7 @@ #PAPERLESS_CONSUMPTION_DIR=../consume #PAPERLESS_DATA_DIR=../data -#PAPERLESS_TRASH_DIR= +#PAPERLESS_EMPTY_TRASH_DIR= #PAPERLESS_MEDIA_ROOT=../media #PAPERLESS_STATICDIR=../static #PAPERLESS_FILENAME_FORMAT= diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 56cfa9ae4..492c160c9 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -240,18 +240,18 @@ Document was added to Paperless-ngx. src/app/app.component.ts - 83 + 85 src/app/app.component.ts - 92 + 94 Open document src/app/app.component.ts - 85 + 87 src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html @@ -274,21 +274,21 @@ Could not add : src/app/app.component.ts - 107 + 109 Document is being processed by Paperless-ngx. src/app/app.component.ts - 122 + 124 Dashboard src/app/app.component.ts - 129 + 131 src/app/components/app-frame/app-frame.component.html @@ -307,7 +307,7 @@ Documents src/app/app.component.ts - 140 + 142 src/app/components/app-frame/app-frame.component.html @@ -342,7 +342,7 @@ Settings src/app/app.component.ts - 152 + 154 src/app/components/admin/settings/settings.component.html @@ -369,14 +369,14 @@ Prev src/app/app.component.ts - 158 + 160 Next src/app/app.component.ts - 159 + 161 src/app/components/document-detail/document-detail.component.html @@ -387,56 +387,56 @@ End src/app/app.component.ts - 160 + 162 The dashboard can be used to show saved views, such as an 'Inbox'. Those settings are found under Settings > Saved Views once you have created some. src/app/app.component.ts - 166 + 168 Drag-and-drop documents here to start uploading or place them in the consume folder. You can also drag-and-drop documents anywhere on all other pages of the web app. Once you do, Paperless-ngx will start training its machine learning algorithms. src/app/app.component.ts - 173 + 175 The documents list shows all of your documents and allows for filtering as well as bulk-editing. There are three different view styles: list, small cards and large cards. A list of documents currently opened for editing is shown in the sidebar. src/app/app.component.ts - 178 + 180 The filtering tools allow you to quickly find documents using various searches, dates, tags, etc. src/app/app.component.ts - 185 + 187 Any combination of filters can be saved as a 'view' which can then be displayed on the dashboard and / or sidebar. src/app/app.component.ts - 191 + 193 Tags, correspondents, document types and storage paths can all be managed using these pages. They can also be created from the document edit view. src/app/app.component.ts - 196 + 198 Manage e-mail accounts and rules for automatically importing documents. src/app/app.component.ts - 204 + 206 src/app/components/manage/mail/mail.component.html @@ -447,14 +447,14 @@ Workflows give you more control over the document pipeline. src/app/app.component.ts - 212 + 214 File Tasks shows you documents that have been consumed, are waiting to be, or may have failed during the process. src/app/app.component.ts - 220 + 222 src/app/components/admin/tasks/tasks.component.html @@ -465,28 +465,28 @@ Check out the settings for various tweaks to the web app and toggle settings for saved views. src/app/app.component.ts - 228 + 230 Thank you! 🙏 src/app/app.component.ts - 236 + 238 There are <em>tons</em> more features and info we didn't cover here, but this should get you started. Check out the documentation or visit the project on GitHub to learn more or to report issues. src/app/app.component.ts - 238 + 240 Lastly, on behalf of every contributor to this community-supported project, thank you for using Paperless-ngx! src/app/app.component.ts - 240 + 242 @@ -684,6 +684,10 @@ src/app/components/admin/tasks/tasks.component.html 23 + + src/app/components/admin/trash/trash.component.html + 45 + src/app/components/admin/users-groups/users-groups.component.html 92 @@ -976,13 +980,6 @@ 195 - - Deleting documents will always ask for confirmation. - - src/app/components/admin/settings/settings.component.html - 195 - - Apply on close @@ -1363,6 +1360,10 @@ src/app/components/admin/tasks/tasks.component.html 42 + + src/app/components/admin/trash/trash.component.html + 37 + src/app/components/admin/users-groups/users-groups.component.html 23 @@ -1422,6 +1423,22 @@ src/app/components/admin/settings/settings.component.html 369 + + src/app/components/admin/trash/trash.component.html + 67 + + + src/app/components/admin/trash/trash.component.html + 76 + + + src/app/components/admin/trash/trash.component.ts + 57 + + + src/app/components/admin/trash/trash.component.ts + 80 + src/app/components/admin/users-groups/users-groups.component.html 38 @@ -1765,6 +1782,10 @@ src/app/components/admin/tasks/tasks.component.html 9 + + src/app/components/admin/trash/trash.component.html + 8 + src/app/components/manage/management-list/management-list.component.html 3 @@ -1788,6 +1809,10 @@ src/app/components/admin/tasks/tasks.component.html 36 + + src/app/components/admin/trash/trash.component.html + 35 + src/app/components/admin/users-groups/users-groups.component.html 21 @@ -2045,6 +2070,188 @@ 141 + + Trash + + src/app/components/admin/trash/trash.component.html + 2 + + + src/app/components/app-frame/app-frame.component.html + 271 + + + src/app/components/app-frame/app-frame.component.html + 274 + + + + Manage trashed documents that are pending deletion. + + src/app/components/admin/trash/trash.component.html + 4 + + + + Restore selected + + src/app/components/admin/trash/trash.component.html + 11 + + + + Delete selected + + src/app/components/admin/trash/trash.component.html + 14 + + + + Empty trash + + src/app/components/admin/trash/trash.component.html + 17 + + + + Remaining + + src/app/components/admin/trash/trash.component.html + 36 + + + + days + + src/app/components/admin/trash/trash.component.html + 58 + + + + Restore + + src/app/components/admin/trash/trash.component.html + 66 + + + src/app/components/admin/trash/trash.component.html + 73 + + + + {VAR_PLURAL, plural, =1 {One document in trash} other { total documents in trash}} + + src/app/components/admin/trash/trash.component.html + 89 + + + + Confirm delete + + src/app/components/admin/trash/trash.component.ts + 53 + + + src/app/components/admin/trash/trash.component.ts + 74 + + + src/app/components/manage/management-list/management-list.component.ts + 203 + + + src/app/components/manage/management-list/management-list.component.ts + 320 + + + + This operation will permanently delete this document. + + src/app/components/admin/trash/trash.component.ts + 54 + + + + This operation cannot be undone. + + src/app/components/admin/trash/trash.component.ts + 55 + + + src/app/components/admin/trash/trash.component.ts + 78 + + + src/app/components/admin/users-groups/users-groups.component.ts + 116 + + + src/app/components/admin/users-groups/users-groups.component.ts + 166 + + + src/app/components/manage/custom-fields/custom-fields.component.ts + 73 + + + src/app/components/manage/mail/mail.component.ts + 114 + + + src/app/components/manage/mail/mail.component.ts + 173 + + + src/app/components/manage/management-list/management-list.component.ts + 322 + + + src/app/components/manage/workflows/workflows.component.ts + 97 + + + + Document deleted + + src/app/components/admin/trash/trash.component.ts + 63 + + + + This operation will permanently delete the selected documents. + + src/app/components/admin/trash/trash.component.ts + 76 + + + + This operation will permanently delete all documents in the trash. + + src/app/components/admin/trash/trash.component.ts + 77 + + + + Document(s) deleted + + src/app/components/admin/trash/trash.component.ts + 87 + + + + Document restored + + src/app/components/admin/trash/trash.component.ts + 97 + + + + Document(s) restored + + src/app/components/admin/trash/trash.component.ts + 106 + + Users & Groups @@ -2247,41 +2454,6 @@ 115 - - This operation cannot be undone. - - src/app/components/admin/users-groups/users-groups.component.ts - 116 - - - src/app/components/admin/users-groups/users-groups.component.ts - 166 - - - src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 714 - - - src/app/components/manage/custom-fields/custom-fields.component.ts - 73 - - - src/app/components/manage/mail/mail.component.ts - 114 - - - src/app/components/manage/mail/mail.component.ts - 173 - - - src/app/components/manage/management-list/management-list.component.ts - 322 - - - src/app/components/manage/workflows/workflows.component.ts - 97 - - Proceed @@ -2310,15 +2482,15 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 755 + 758 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 788 + 791 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 807 + 810 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2433,11 +2605,11 @@ src/app/components/app-frame/app-frame.component.html - 272 + 279 src/app/components/app-frame/app-frame.component.html - 275 + 282 @@ -2612,42 +2784,42 @@ GitHub src/app/components/app-frame/app-frame.component.html - 282 + 289 is available. src/app/components/app-frame/app-frame.component.html - 291,292 + 298,299 Click to view. src/app/components/app-frame/app-frame.component.html - 292 + 299 Paperless-ngx can automatically check for updates src/app/components/app-frame/app-frame.component.html - 296 + 303 How does this work? src/app/components/app-frame/app-frame.component.html - 303,305 + 310,312 Update available src/app/components/app-frame/app-frame.component.html - 316 + 323 @@ -2887,6 +3059,10 @@ src/app/components/common/permissions-dialog/permissions-dialog.component.html 26 + + src/app/components/document-detail/document-detail.component.ts + 776 + src/app/components/document-list/bulk-editor/bulk-editor.component.ts 401 @@ -2907,6 +3083,10 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts 579 + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 712 + Page @@ -5864,41 +6044,34 @@ 749 - - Confirm delete - - src/app/components/document-detail/document-detail.component.ts - 776 - - - src/app/components/manage/management-list/management-list.component.ts - 203 - - - src/app/components/manage/management-list/management-list.component.ts - 320 - - - - Do you really want to delete document ""? + + Do you really want to move the document "" to the trash? src/app/components/document-detail/document-detail.component.ts 777 - - The files for this document will be deleted permanently. This operation cannot be undone. + + Documents can be restored prior to permanent deletion. src/app/components/document-detail/document-detail.component.ts 778 + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 714 + - - Delete document + + Move to trash src/app/components/document-detail/document-detail.component.ts 780 + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 716 + Error deleting document @@ -5915,7 +6088,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 751 + 754 @@ -5989,7 +6162,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 784 + 787 @@ -6364,74 +6537,60 @@ 571,575 - - Delete confirm - - src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 712 - - - - This operation will permanently delete selected document(s). + + Move selected document(s) to the trash? src/app/components/document-list/bulk-editor/bulk-editor.component.ts 713 - - Delete document(s) - - src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 716 - - This operation will permanently recreate the archive files for selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 752 + 755 The archive files will be re-generated with the current settings. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 753 + 756 This operation will permanently rotate the original version of document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 785 + 788 This will alter the original copy. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 786 + 789 Merge confirm src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 805 + 808 This operation will merge selected documents into a new document. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 806 + 809 Merged document will be queued for consumption. src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 822 + 825 diff --git a/src-ui/src/app/app-routing.module.ts b/src-ui/src/app/app-routing.module.ts index 12b412f67..bbeba9e8a 100644 --- a/src-ui/src/app/app-routing.module.ts +++ b/src-ui/src/app/app-routing.module.ts @@ -26,6 +26,7 @@ import { MailComponent } from './components/manage/mail/mail.component' import { UsersAndGroupsComponent } from './components/admin/users-groups/users-groups.component' import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component' import { ConfigComponent } from './components/admin/config/config.component' +import { TrashComponent } from './components/admin/trash/trash.component' export const routes: Routes = [ { path: '', redirectTo: 'dashboard', pathMatch: 'full' }, @@ -144,6 +145,14 @@ export const routes: Routes = [ requireAdmin: true, }, }, + { + path: 'trash', + component: TrashComponent, + canActivate: [PermissionsGuard], + data: { + requireAdmin: true, + }, + }, // redirect old paths { path: 'settings/mail', diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index f9e04b069..2232dad19 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -125,6 +125,7 @@ import { CustomFieldDisplayComponent } from './components/common/custom-field-di 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 { TrashComponent } from './components/admin/trash/trash.component' import { airplane, archive, @@ -497,6 +498,7 @@ function initializeApp(settings: SettingsService) { GlobalSearchComponent, HotkeyDialogComponent, DeletePagesConfirmDialogComponent, + TrashComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/admin/settings/settings.component.html b/src-ui/src/app/components/admin/settings/settings.component.html index bcab7de33..285599639 100644 --- a/src-ui/src/app/components/admin/settings/settings.component.html +++ b/src-ui/src/app/components/admin/settings/settings.component.html @@ -192,7 +192,7 @@
- +
diff --git a/src-ui/src/app/components/admin/trash/trash.component.html b/src-ui/src/app/components/admin/trash/trash.component.html new file mode 100644 index 000000000..1c66bdd44 --- /dev/null +++ b/src-ui/src/app/components/admin/trash/trash.component.html @@ -0,0 +1,98 @@ + + + + + + + +
+ +
+ +
+ + + + + + + + + + + @if (isLoading) { + + + + } + @for (document of documentsInTrash; track document.id) { + + + + + + + } + +
+
+ + +
+
NameRemainingActions
+
+ Loading... +
+
+ + +
+
{{ document.title }}{{ getDaysRemaining(document) }} days +
+
+ +
+ + +
+
+
+
+ + +
+
+
+ +@if (!isLoading) { +
+
+ {totalDocuments, plural, =1 {One document in trash} other {{{totalDocuments || 0}} total documents in trash}} + @if (selectedDocuments.size > 0) { +  ({{selectedDocuments.size}} selected) + } +
+ @if (documentsInTrash.length > 20) { + + } +
+} diff --git a/src-ui/src/app/components/admin/trash/trash.component.scss b/src-ui/src/app/components/admin/trash/trash.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/admin/trash/trash.component.spec.ts b/src-ui/src/app/components/admin/trash/trash.component.spec.ts new file mode 100644 index 000000000..063d4bb8f --- /dev/null +++ b/src-ui/src/app/components/admin/trash/trash.component.spec.ts @@ -0,0 +1,163 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' + +import { TrashComponent } from './trash.component' +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { PageHeaderComponent } from '../../common/page-header/page-header.component' +import { + NgbModal, + NgbPaginationModule, + NgbPopoverModule, +} from '@ng-bootstrap/ng-bootstrap' +import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' +import { FormsModule, ReactiveFormsModule } from '@angular/forms' +import { TrashService } from 'src/app/services/trash.service' +import { of } from 'rxjs' +import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' +import { By } from '@angular/platform-browser' + +const documentsInTrash = [ + { + id: 1, + name: 'test1', + created: new Date('2023-03-01T10:26:03.093116Z'), + deleted_at: new Date('2023-03-01T10:26:03.093116Z'), + }, + { + id: 2, + name: 'test2', + created: new Date('2023-03-01T10:26:03.093116Z'), + deleted_at: new Date('2023-03-01T10:26:03.093116Z'), + }, +] + +describe('TrashComponent', () => { + let component: TrashComponent + let fixture: ComponentFixture + let trashService: TrashService + let modalService: NgbModal + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ + TrashComponent, + PageHeaderComponent, + ConfirmDialogComponent, + ], + imports: [ + HttpClientTestingModule, + FormsModule, + ReactiveFormsModule, + NgbPopoverModule, + NgbPaginationModule, + NgxBootstrapIconsModule.pick(allIcons), + ], + }).compileComponents() + + fixture = TestBed.createComponent(TrashComponent) + trashService = TestBed.inject(TrashService) + modalService = TestBed.inject(NgbModal) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should call correct service method on reload', () => { + const trashSpy = jest.spyOn(trashService, 'getTrash') + trashSpy.mockReturnValue( + of({ + count: 2, + all: documentsInTrash.map((d) => d.id), + results: documentsInTrash, + }) + ) + component.reload() + expect(trashSpy).toHaveBeenCalled() + expect(component.documentsInTrash).toEqual(documentsInTrash) + }) + + it('should support delete document', () => { + const trashSpy = jest.spyOn(trashService, 'emptyTrash') + let modal + modalService.activeInstances.subscribe((instances) => { + modal = instances[0] + }) + trashSpy.mockReturnValue(of('OK')) + component.delete(documentsInTrash[0]) + expect(modal).toBeDefined() + modal.componentInstance.confirmClicked.next() + expect(trashSpy).toHaveBeenCalled() + }) + + it('should support empty trash', () => { + const trashSpy = jest.spyOn(trashService, 'emptyTrash') + let modal + modalService.activeInstances.subscribe((instances) => { + modal = instances[instances.length - 1] + }) + trashSpy.mockReturnValue(of('OK')) + component.emptyTrash() + expect(modal).toBeDefined() + modal.componentInstance.confirmClicked.next() + expect(trashSpy).toHaveBeenCalled() + modal.close() + component.emptyTrash(new Set([1, 2])) + modal.componentInstance.confirmClicked.next() + expect(trashSpy).toHaveBeenCalledWith([1, 2]) + }) + + it('should support restore document', () => { + const restoreSpy = jest.spyOn(trashService, 'restoreDocuments') + const reloadSpy = jest.spyOn(component, 'reload') + restoreSpy.mockReturnValue(of('OK')) + component.restore(documentsInTrash[0]) + expect(restoreSpy).toHaveBeenCalledWith([documentsInTrash[0].id]) + expect(reloadSpy).toHaveBeenCalled() + }) + + it('should support restore all documents', () => { + const restoreSpy = jest.spyOn(trashService, 'restoreDocuments') + const reloadSpy = jest.spyOn(component, 'reload') + restoreSpy.mockReturnValue(of('OK')) + component.restoreAll() + expect(restoreSpy).toHaveBeenCalled() + expect(reloadSpy).toHaveBeenCalled() + component.restoreAll(new Set([1, 2])) + expect(restoreSpy).toHaveBeenCalledWith([1, 2]) + }) + + it('should support toggle all items in view', () => { + component.documentsInTrash = documentsInTrash + expect(component.selectedDocuments.size).toEqual(0) + const toggleAllSpy = jest.spyOn(component, 'toggleAll') + const checkButton = fixture.debugElement.queryAll( + By.css('input.form-check-input') + )[0] + checkButton.nativeElement.dispatchEvent(new Event('click')) + checkButton.nativeElement.checked = true + checkButton.nativeElement.dispatchEvent(new Event('click')) + expect(toggleAllSpy).toHaveBeenCalled() + expect(component.selectedDocuments.size).toEqual(documentsInTrash.length) + }) + + it('should support toggle item', () => { + component.selectedDocuments = new Set([1]) + component.toggleSelected(documentsInTrash[0]) + expect(component.selectedDocuments.size).toEqual(0) + component.toggleSelected(documentsInTrash[0]) + expect(component.selectedDocuments.size).toEqual(1) + }) + + it('should support clear selection', () => { + component.selectedDocuments = new Set([1]) + component.clearSelection() + expect(component.selectedDocuments.size).toEqual(0) + }) + + it('should correctly display days remaining', () => { + expect(component.getDaysRemaining(documentsInTrash[0])).toBeLessThan(0) + const tenDaysAgo = new Date() + tenDaysAgo.setDate(tenDaysAgo.getDate() - 10) + expect( + component.getDaysRemaining({ deleted_at: tenDaysAgo }) + ).toBeGreaterThan(0) // 10 days ago but depends on month + }) +}) diff --git a/src-ui/src/app/components/admin/trash/trash.component.ts b/src-ui/src/app/components/admin/trash/trash.component.ts new file mode 100644 index 000000000..b867f1706 --- /dev/null +++ b/src-ui/src/app/components/admin/trash/trash.component.ts @@ -0,0 +1,137 @@ +import { Component, OnDestroy } from '@angular/core' +import { NgbModal } from '@ng-bootstrap/ng-bootstrap' +import { Document } from 'src/app/data/document' +import { ToastService } from 'src/app/services/toast.service' +import { TrashService } from 'src/app/services/trash.service' +import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' +import { Subject, takeUntil } from 'rxjs' +import { SettingsService } from 'src/app/services/settings.service' +import { SETTINGS_KEYS } from 'src/app/data/ui-settings' + +@Component({ + selector: 'pngx-trash', + templateUrl: './trash.component.html', + styleUrl: './trash.component.scss', +}) +export class TrashComponent implements OnDestroy { + public documentsInTrash: Document[] = [] + public selectedDocuments: Set = new Set() + public allToggled: boolean = false + public page: number = 1 + public totalDocuments: number + public isLoading: boolean = false + unsubscribeNotifier: Subject = new Subject() + + constructor( + private trashService: TrashService, + private toastService: ToastService, + private modalService: NgbModal, + private settingsService: SettingsService + ) { + this.reload() + } + + ngOnDestroy() { + this.unsubscribeNotifier.next() + this.unsubscribeNotifier.complete() + } + + reload() { + this.isLoading = true + this.trashService.getTrash(this.page).subscribe((r) => { + this.documentsInTrash = r.results + this.totalDocuments = r.count + this.isLoading = false + this.selectedDocuments.clear() + }) + } + + delete(document: Document) { + let modal = this.modalService.open(ConfirmDialogComponent, { + backdrop: 'static', + }) + modal.componentInstance.title = $localize`Confirm delete` + modal.componentInstance.messageBold = $localize`This operation will permanently delete this document.` + modal.componentInstance.message = $localize`This operation cannot be undone.` + modal.componentInstance.btnClass = 'btn-danger' + modal.componentInstance.btnCaption = $localize`Delete` + modal.componentInstance.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + modal.componentInstance.buttonsEnabled = false + this.trashService.emptyTrash([document.id]).subscribe(() => { + this.toastService.showInfo($localize`Document deleted`) + modal.close() + this.reload() + }) + }) + } + + emptyTrash(documents?: Set) { + let modal = this.modalService.open(ConfirmDialogComponent, { + backdrop: 'static', + }) + modal.componentInstance.title = $localize`Confirm delete` + modal.componentInstance.messageBold = documents + ? $localize`This operation will permanently delete the selected documents.` + : $localize`This operation will permanently delete all documents in the trash.` + modal.componentInstance.message = $localize`This operation cannot be undone.` + modal.componentInstance.btnClass = 'btn-danger' + modal.componentInstance.btnCaption = $localize`Delete` + modal.componentInstance.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + this.trashService + .emptyTrash(documents ? Array.from(documents) : null) + .subscribe(() => { + this.toastService.showInfo($localize`Document(s) deleted`) + this.allToggled = false + modal.close() + this.reload() + }) + }) + } + + restore(document: Document) { + this.trashService.restoreDocuments([document.id]).subscribe(() => { + this.toastService.showInfo($localize`Document restored`) + this.reload() + }) + } + + restoreAll(documents: Set = null) { + this.trashService + .restoreDocuments(documents ? Array.from(documents) : null) + .subscribe(() => { + this.toastService.showInfo($localize`Document(s) restored`) + this.allToggled = false + this.reload() + }) + } + + toggleAll(event: PointerEvent) { + if ((event.target as HTMLInputElement).checked) { + this.selectedDocuments = new Set(this.documentsInTrash.map((t) => t.id)) + } else { + this.clearSelection() + } + } + + toggleSelected(object: Document) { + this.selectedDocuments.has(object.id) + ? this.selectedDocuments.delete(object.id) + : this.selectedDocuments.add(object.id) + } + + clearSelection() { + this.allToggled = false + this.selectedDocuments.clear() + } + + getDaysRemaining(document: Document): number { + const delay = this.settingsService.get(SETTINGS_KEYS.EMPTY_TRASH_DELAY) + const diff = new Date().getTime() - new Date(document.deleted_at).getTime() + const days = Math.ceil(diff / (1000 * 3600 * 24)) + return delay - days + } +} diff --git a/src-ui/src/app/components/app-frame/app-frame.component.html b/src-ui/src/app/components/app-frame/app-frame.component.html index ab5759ec0..20c90b402 100644 --- a/src-ui/src/app/components/app-frame/app-frame.component.html +++ b/src-ui/src/app/components/app-frame/app-frame.component.html @@ -267,6 +267,13 @@ } +