diff --git a/docs/usage.md b/docs/usage.md index 9f561ee01..e7e1bede4 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -545,3 +545,16 @@ Paperless-ngx consists of the following components: - Optional: A database server. Paperless supports PostgreSQL, MariaDB and SQLite for storing its data. + +## Share Links + +Paperless-ngx added the abiltiy to create shareable links to files in version 2.0. You can find the button for this on the document detail screen. + +- Share links do not require a user to login and thus link directly to a file. +- Links are unique and are of the form `{paperless-url}/share/{randomly-generated-slug}`. +- Links can optionally have an expiration time set. +- After a link expires or is deleted users will be redirected to the regular paperless-ngx login. + +!!! tip + + If your paperless-ngx instance is behind a reverse-proxy you may want to create an exception to bypass any authentication layers that are part of your setup in order to make links truly publicly-accessible. Of course, do so with caution. diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index e7c04174b..cc23dd26c 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -319,7 +319,7 @@ src/app/components/document-detail/document-detail.component.html - 55 + 65 @@ -1073,7 +1073,7 @@ src/app/components/document-detail/document-detail.component.html - 198 + 208 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -1142,7 +1142,7 @@ src/app/components/document-detail/document-detail.component.html - 182 + 192 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1549,6 +1549,10 @@ src/app/components/common/permissions-select/permissions-select.component.html 9 + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 33 + src/app/components/document-detail/document-detail.component.html 11 @@ -2240,6 +2244,135 @@ 20 + + Share Links + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 6 + + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 25 + + + + No existing links + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 10,12 + + + + Copy + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 23 + + + + Share + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 28 + + + + Copied! + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 36 + + + + Share archive version + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 42 + + + + Create + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.html + 55 + + + src/app/components/manage/management-list/management-list.component.html + 2 + + + src/app/components/manage/management-list/management-list.component.html + 2 + + + src/app/components/manage/management-list/management-list.component.html + 2 + + + src/app/components/manage/management-list/management-list.component.html + 2 + + + + 1 day + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 18 + + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 85 + + + + 7 days + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 19 + + + + 30 days + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 20 + + + + Never + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 21 + + + + Error retrieving links + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 69 + + + + days + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 85 + + + + Error deleting link + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 112 + + + + Error creating link + + src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts + 140 + + Status @@ -2310,7 +2443,7 @@ src/app/components/document-detail/document-detail.component.html - 75 + 85 src/app/components/document-list/document-list.component.html @@ -2340,7 +2473,7 @@ src/app/components/document-detail/document-detail.component.html - 19 + 18 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2538,14 +2671,65 @@ Download original src/app/components/document-detail/document-detail.component.html - 25 + 24 + + + + Actions + + src/app/components/document-detail/document-detail.component.html + 34 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 86 + + + src/app/components/manage/management-list/management-list.component.html + 23 + + + src/app/components/manage/management-list/management-list.component.html + 23 + + + src/app/components/manage/management-list/management-list.component.html + 23 + + + src/app/components/manage/management-list/management-list.component.html + 23 + + + src/app/components/manage/settings/settings.component.html + 221 + + + src/app/components/manage/settings/settings.component.html + 259 + + + src/app/components/manage/settings/settings.component.html + 296 + + + src/app/components/manage/settings/settings.component.html + 347 + + + src/app/components/manage/settings/settings.component.html + 382 + + + src/app/components/manage/tasks/tasks.component.html + 44 Redo OCR src/app/components/document-detail/document-detail.component.html - 34 + 40 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2556,7 +2740,7 @@ More like this src/app/components/document-detail/document-detail.component.html - 40 + 46 src/app/components/document-list/document-card-large/document-card-large.component.html @@ -2567,7 +2751,7 @@ Close src/app/components/document-detail/document-detail.component.html - 43 + 53 src/app/guards/dirty-saved-view.guard.ts @@ -2578,35 +2762,35 @@ Previous src/app/components/document-detail/document-detail.component.html - 50 + 60 Details src/app/components/document-detail/document-detail.component.html - 72 + 82 Archive serial number src/app/components/document-detail/document-detail.component.html - 76 + 86 Date created src/app/components/document-detail/document-detail.component.html - 77 + 87 Correspondent src/app/components/document-detail/document-detail.component.html - 79 + 89 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2629,7 +2813,7 @@ Document type src/app/components/document-detail/document-detail.component.html - 81 + 91 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2652,7 +2836,7 @@ Storage path src/app/components/document-detail/document-detail.component.html - 83 + 93 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2671,21 +2855,21 @@ Default src/app/components/document-detail/document-detail.component.html - 84 + 94 Content src/app/components/document-detail/document-detail.component.html - 91 + 101 Metadata src/app/components/document-detail/document-detail.component.html - 100 + 110 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -2696,173 +2880,173 @@ Date modified src/app/components/document-detail/document-detail.component.html - 106 + 116 Date added src/app/components/document-detail/document-detail.component.html - 110 + 120 Media filename src/app/components/document-detail/document-detail.component.html - 114 + 124 Original filename src/app/components/document-detail/document-detail.component.html - 118 + 128 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 122 + 132 Original file size src/app/components/document-detail/document-detail.component.html - 126 + 136 Original mime type src/app/components/document-detail/document-detail.component.html - 130 + 140 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 134 + 144 Archive file size src/app/components/document-detail/document-detail.component.html - 138 + 148 Original document metadata src/app/components/document-detail/document-detail.component.html - 144 + 154 Archived document metadata src/app/components/document-detail/document-detail.component.html - 145 + 155 Preview src/app/components/document-detail/document-detail.component.html - 151 + 161 Enter Password src/app/components/document-detail/document-detail.component.html - 167 + 177 src/app/components/document-detail/document-detail.component.html - 218 + 228 Notes src/app/components/document-detail/document-detail.component.html - 175,176 + 185,186 Discard src/app/components/document-detail/document-detail.component.html - 194 + 204 Save & next src/app/components/document-detail/document-detail.component.html - 196 + 206 Save & close src/app/components/document-detail/document-detail.component.html - 197 + 207 An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 252,254 + 253,255 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 397 + 398 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 418 + 419 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 531 + 532 src/app/components/document-detail/document-detail.component.ts - 539 + 540 Error saving document src/app/components/document-detail/document-detail.component.ts - 543 + 544 src/app/components/document-detail/document-detail.component.ts - 584 + 585 Confirm delete src/app/components/document-detail/document-detail.component.ts - 610 + 611 src/app/components/manage/management-list/management-list.component.ts @@ -2873,35 +3057,35 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 611 + 612 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 612 + 613 Delete document src/app/components/document-detail/document-detail.component.ts - 614 + 615 Error deleting document src/app/components/document-detail/document-detail.component.ts - 633 + 634 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 653 + 654 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2912,14 +3096,14 @@ This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 654 + 655 This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 655 + 656 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2950,7 +3134,7 @@ Proceed src/app/components/document-detail/document-detail.component.ts - 657 + 658 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2977,14 +3161,14 @@ 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 - 665 + 666 Error executing operation src/app/components/document-detail/document-detail.component.ts - 676 + 677 @@ -3045,53 +3229,6 @@ 52 - - Actions - - src/app/components/document-list/bulk-editor/bulk-editor.component.html - 86 - - - src/app/components/manage/management-list/management-list.component.html - 23 - - - src/app/components/manage/management-list/management-list.component.html - 23 - - - src/app/components/manage/management-list/management-list.component.html - 23 - - - src/app/components/manage/management-list/management-list.component.html - 23 - - - src/app/components/manage/settings/settings.component.html - 221 - - - src/app/components/manage/settings/settings.component.html - 259 - - - src/app/components/manage/settings/settings.component.html - 296 - - - src/app/components/manage/settings/settings.component.html - 347 - - - src/app/components/manage/settings/settings.component.html - 382 - - - src/app/components/manage/tasks/tasks.component.html - 44 - - Include: @@ -3945,25 +4082,6 @@ 44 - - Create - - src/app/components/manage/management-list/management-list.component.html - 2 - - - src/app/components/manage/management-list/management-list.component.html - 2 - - - src/app/components/manage/management-list/management-list.component.html - 2 - - - src/app/components/manage/management-list/management-list.component.html - 2 - - Filter by: diff --git a/src-ui/package-lock.json b/src-ui/package-lock.json index d6c048877..7c3daeb98 100644 --- a/src-ui/package-lock.json +++ b/src-ui/package-lock.json @@ -24,6 +24,7 @@ "file-saver": "^2.0.5", "mime-names": "^1.0.0", "ng2-pdf-viewer": "^10.0.0", + "ngx-clipboard": "^16.0.0", "ngx-color": "^9.0.0", "ngx-cookie-service": "^16.0.1", "ngx-file-drop": "^16.0.0", @@ -14404,6 +14405,19 @@ "pdfjs-dist": "~2.16.105" } }, + "node_modules/ngx-clipboard": { + "version": "16.0.0", + "resolved": "https://registry.npmjs.org/ngx-clipboard/-/ngx-clipboard-16.0.0.tgz", + "integrity": "sha512-rZ/Eo1PqiKMiyF8tdjhmUkoUu68f7OzBJ7YH1YFeh2RAaNrerTaW8XfFOzppSckjFQqA1fwGSYuTTJlDhDag5w==", + "dependencies": { + "ngx-window-token": ">=7.0.0", + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/common": ">=13.0.0", + "@angular/core": ">=13.0.0" + } + }, "node_modules/ngx-color": { "version": "9.0.0", "resolved": "https://registry.npmjs.org/ngx-color/-/ngx-color-9.0.0.tgz", @@ -14474,6 +14488,21 @@ "@ng-bootstrap/ng-bootstrap": "^15.0.0" } }, + "node_modules/ngx-window-token": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/ngx-window-token/-/ngx-window-token-7.0.0.tgz", + "integrity": "sha512-5+XfRVSY7Dciu8xyCNMkOlH2UfwR9W2P1Pirz7caaZgOZDjFbL8aEO2stjfJJm2FFf1D6dlVHNzhLWGk9HGkqA==", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": "^14.20.0 || ^16.13.0 || >=18.10.0" + }, + "peerDependencies": { + "@angular/common": ">=13.0.0", + "@angular/core": ">=13.0.0" + } + }, "node_modules/nice-napi": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/nice-napi/-/nice-napi-1.0.2.tgz", diff --git a/src-ui/package.json b/src-ui/package.json index 76b44b85f..42ff4d792 100644 --- a/src-ui/package.json +++ b/src-ui/package.json @@ -26,6 +26,7 @@ "file-saver": "^2.0.5", "mime-names": "^1.0.0", "ng2-pdf-viewer": "^10.0.0", + "ngx-clipboard": "^16.0.0", "ngx-color": "^9.0.0", "ngx-cookie-service": "^16.0.1", "ngx-file-drop": "^16.0.0", diff --git a/src-ui/setup-jest.ts b/src-ui/setup-jest.ts index 65004742b..c0dfad9f9 100644 --- a/src-ui/setup-jest.ts +++ b/src-ui/setup-jest.ts @@ -86,6 +86,7 @@ Object.defineProperty(navigator, 'clipboard', { writeText: async () => {}, }, }) +Object.defineProperty(navigator, 'canShare', { value: () => true }) Object.defineProperty(window, 'ResizeObserver', { value: mock() }) HTMLCanvasElement.prototype.getContext = < diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index aac7a5238..f46c06cb9 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -94,6 +94,7 @@ import { PermissionsFilterDropdownComponent } from './components/common/permissi import { UsernamePipe } from './pipes/username.pipe' import { LogoComponent } from './components/common/logo/logo.component' import { IsNumberPipe } from './pipes/is-number.pipe' +import { ShareLinksDropdownComponent } from './components/common/share-links-dropdown/share-links-dropdown.component' import localeAf from '@angular/common/locales/af' import localeAr from '@angular/common/locales/ar' @@ -231,6 +232,7 @@ function initializeApp(settings: SettingsService) { UsernamePipe, LogoComponent, IsNumberPipe, + ShareLinksDropdownComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.spec.ts b/src-ui/src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.spec.ts index 93fa7f0fd..1a35fb5ef 100644 --- a/src-ui/src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.spec.ts +++ b/src-ui/src/app/components/common/edit-dialog/mail-account-edit-dialog/mail-account-edit-dialog.component.spec.ts @@ -1,7 +1,6 @@ import { ComponentFixture, TestBed, - discardPeriodicTasks, fakeAsync, tick, } from '@angular/core/testing' diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.html b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.html new file mode 100644 index 000000000..15b3ce64c --- /dev/null +++ b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.html @@ -0,0 +1,61 @@ +
+ + +
diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.scss b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.scss new file mode 100644 index 000000000..47e19d871 --- /dev/null +++ b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.scss @@ -0,0 +1,14 @@ +.share-links-dropdown { + min-width: 350px; + + // correct position on mobile + @media (max-width: 575.98px) { + &.show { + margin-left: -175px !important; + } + } +} + +.copied-badge { + right: 7.5em; +} diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.spec.ts b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.spec.ts new file mode 100644 index 000000000..c230fa870 --- /dev/null +++ b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.spec.ts @@ -0,0 +1,195 @@ +import { + HttpTestingController, + HttpClientTestingModule, +} from '@angular/common/http/testing' +import { + ComponentFixture, + TestBed, + fakeAsync, + tick, +} from '@angular/core/testing' +import { FormsModule, ReactiveFormsModule } from '@angular/forms' +import { of, throwError } from 'rxjs' +import { + PaperlessFileVersion, + PaperlessShareLink, +} from 'src/app/data/paperless-share-link' +import { ShareLinkService } from 'src/app/services/rest/share-link.service' +import { ToastService } from 'src/app/services/toast.service' +import { environment } from 'src/environments/environment' +import { ShareLinksDropdownComponent } from './share-links-dropdown.component' +import { ClipboardService } from 'ngx-clipboard' + +describe('ShareLinksDropdownComponent', () => { + let component: ShareLinksDropdownComponent + let fixture: ComponentFixture + let shareLinkService: ShareLinkService + let toastService: ToastService + let httpController: HttpTestingController + let clipboardService: ClipboardService + + beforeEach(() => { + TestBed.configureTestingModule({ + declarations: [ShareLinksDropdownComponent], + imports: [HttpClientTestingModule, FormsModule, ReactiveFormsModule], + }) + + fixture = TestBed.createComponent(ShareLinksDropdownComponent) + shareLinkService = TestBed.inject(ShareLinkService) + toastService = TestBed.inject(ToastService) + httpController = TestBed.inject(HttpTestingController) + clipboardService = TestBed.inject(ClipboardService) + + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should support refresh to retrieve links', () => { + const getSpy = jest.spyOn(shareLinkService, 'getLinksForDocument') + component.documentId = 99 + + const now = new Date() + const expiration7days = new Date() + expiration7days.setDate(now.getDate() + 7) + + getSpy.mockReturnValue( + of([ + { + id: 1, + slug: '1234slug', + created: now.toISOString(), + document: 99, + file_version: PaperlessFileVersion.Archive, + expiration: expiration7days.toISOString(), + }, + { + id: 1, + slug: '1234slug', + created: now.toISOString(), + document: 99, + file_version: PaperlessFileVersion.Original, + expiration: null, + }, + ]) + ) + + component.refresh() + expect(getSpy).toHaveBeenCalled() + + fixture.detectChanges() + + expect(component.shareLinks).toHaveLength(2) + }) + + it('should show error on refresh if needed', () => { + const toastSpy = jest.spyOn(toastService, 'showError') + jest + .spyOn(shareLinkService, 'getLinksForDocument') + .mockReturnValueOnce(throwError(() => new Error('Unable to get links'))) + component.documentId = 99 + + component.refresh() + fixture.detectChanges() + expect(toastSpy).toHaveBeenCalled() + }) + + it('should support link creation then refresh & copy url', fakeAsync(() => { + const createSpy = jest.spyOn(shareLinkService, 'createLinkForDocument') + component.documentId = 99 + component.expirationDays = 7 + component.archiveVersion = false + + const expiration = new Date() + expiration.setDate(expiration.getDate() + 7) + + const copySpy = jest.spyOn(clipboardService, 'copy') + const refreshSpy = jest.spyOn(component, 'refresh') + + component.createLink() + expect(createSpy).toHaveBeenCalledWith(99, 'original', expiration) + + httpController.expectOne(`${environment.apiBaseUrl}share_links/`).flush({ + id: 1, + slug: '1234slug', + document: 99, + expiration: expiration.toISOString(), + }) + fixture.detectChanges() + tick(3000) + + expect(copySpy).toHaveBeenCalled() + expect(refreshSpy).toHaveBeenCalled() + })) + + it('should show error on link creation if needed', () => { + component.documentId = 99 + component.expirationDays = 7 + + const expiration = new Date() + expiration.setDate(expiration.getDate() + 7) + + const toastSpy = jest.spyOn(toastService, 'showError') + + component.createLink() + + httpController + .expectOne(`${environment.apiBaseUrl}share_links/`) + .flush( + { error: 'Share link error' }, + { status: 500, statusText: 'error' } + ) + fixture.detectChanges() + + expect(toastSpy).toHaveBeenCalled() + }) + + it('should support delete links & refresh', () => { + const deleteSpy = jest.spyOn(shareLinkService, 'delete') + deleteSpy.mockReturnValue(of(true)) + const refreshSpy = jest.spyOn(component, 'refresh') + + component.delete({ id: 12 } as PaperlessShareLink) + fixture.detectChanges() + expect(deleteSpy).toHaveBeenCalledWith({ id: 12 }) + expect(refreshSpy).toHaveBeenCalled() + }) + + it('should show error on delete if needed', () => { + const toastSpy = jest.spyOn(toastService, 'showError') + jest + .spyOn(shareLinkService, 'delete') + .mockReturnValueOnce(throwError(() => new Error('Unable to delete link'))) + component.delete(null) + fixture.detectChanges() + expect(toastSpy).toHaveBeenCalled() + }) + + it('should format days remaining', () => { + const now = new Date() + const expiration7days = new Date() + expiration7days.setDate(now.getDate() + 7) + const expiration1day = new Date() + expiration1day.setDate(now.getDate() + 1) + + expect( + component.getDaysRemaining({ + expiration: expiration7days.toISOString(), + } as PaperlessShareLink) + ).toEqual('7 days') + expect( + component.getDaysRemaining({ + expiration: expiration1day.toISOString(), + } as PaperlessShareLink) + ).toEqual('1 day') + }) + + // coverage + it('should support share', () => { + const link = { slug: '12345slug' } as PaperlessShareLink + if (!('share' in navigator)) + Object.defineProperty(navigator, 'share', { value: (obj: any) => {} }) + // const navigatorSpy = jest.spyOn(navigator, 'share') + component.share(link) + // expect(navigatorSpy).toHaveBeenCalledWith({ url: component.getShareUrl(link) }) + }) +}) diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts new file mode 100644 index 000000000..1eb43fa42 --- /dev/null +++ b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts @@ -0,0 +1,149 @@ +import { Component, Input, OnInit } from '@angular/core' +import { first } from 'rxjs' +import { + PaperlessShareLink, + PaperlessFileVersion, +} from 'src/app/data/paperless-share-link' +import { ShareLinkService } from 'src/app/services/rest/share-link.service' +import { ToastService } from 'src/app/services/toast.service' +import { environment } from 'src/environments/environment' +import { ClipboardService } from 'ngx-clipboard' + +@Component({ + selector: 'app-share-links-dropdown', + templateUrl: './share-links-dropdown.component.html', + styleUrls: ['./share-links-dropdown.component.scss'], +}) +export class ShareLinksDropdownComponent implements OnInit { + EXPIRATION_OPTIONS = [ + { label: $localize`1 day`, value: 1 }, + { label: $localize`7 days`, value: 7 }, + { label: $localize`30 days`, value: 30 }, + { label: $localize`Never`, value: null }, + ] + + @Input() + title = $localize`Share Links` + + _documentId: number + + @Input() + set documentId(id: number) { + if (id !== undefined) { + this._documentId = id + this.refresh() + } + } + + @Input() + disabled: boolean = false + + shareLinks: PaperlessShareLink[] + + loading: boolean = false + + copied: number + + expirationDays: number = 7 + + archiveVersion: boolean = true + + constructor( + private shareLinkService: ShareLinkService, + private toastService: ToastService, + private clipboardService: ClipboardService + ) {} + + ngOnInit(): void { + if (this._documentId !== undefined) this.refresh() + } + + refresh() { + if (this._documentId === undefined) return + this.loading = true + this.shareLinkService + .getLinksForDocument(this._documentId) + .pipe(first()) + .subscribe({ + next: (results) => { + this.loading = false + this.shareLinks = results + }, + error: (e) => { + this.toastService.showError( + $localize`Error retrieving links`, + 10000, + e + ) + }, + }) + } + + getShareUrl(link: PaperlessShareLink): string { + return `${environment.apiBaseUrl.replace('api', 'share')}${link.slug}` + } + + getDaysRemaining(link: PaperlessShareLink): string { + const days: number = Math.ceil( + (Date.parse(link.expiration) - Date.now()) / (1000 * 60 * 60 * 24) + ) + return days === 1 ? $localize`1 day` : $localize`${days} days` + } + + copy(link: PaperlessShareLink) { + this.clipboardService.copy(this.getShareUrl(link)) + this.copied = link.id + setTimeout(() => { + this.copied = null + }, 3000) + } + + canShare(link: PaperlessShareLink): boolean { + return ( + navigator?.canShare && navigator.canShare({ url: this.getShareUrl(link) }) + ) + } + + share(link: PaperlessShareLink) { + navigator.share({ url: this.getShareUrl(link) }) + } + + delete(link: PaperlessShareLink) { + this.shareLinkService.delete(link).subscribe({ + next: () => { + this.refresh() + }, + error: (e) => { + this.toastService.showError($localize`Error deleting link`, 10000, e) + }, + }) + } + + createLink() { + let expiration + if (this.expirationDays) { + expiration = new Date() + expiration.setDate(expiration.getDate() + this.expirationDays) + } + this.loading = true + this.shareLinkService + .createLinkForDocument( + this._documentId, + this.archiveVersion + ? PaperlessFileVersion.Archive + : PaperlessFileVersion.Original, + expiration + ) + .subscribe({ + next: (result) => { + this.loading = false + this.copy(result) + this.refresh() + }, + error: (e) => { + this.loading = false + this.toastService.showError($localize`Error creating link`, 10000, e) + }, + }) + } +} diff --git a/src-ui/src/app/components/common/toasts/toasts.component.spec.ts b/src-ui/src/app/components/common/toasts/toasts.component.spec.ts index 4dd85305f..32cb15085 100644 --- a/src-ui/src/app/components/common/toasts/toasts.component.spec.ts +++ b/src-ui/src/app/components/common/toasts/toasts.component.spec.ts @@ -10,6 +10,7 @@ import { ComponentFixture } from '@angular/core/testing' import { HttpClientTestingModule } from '@angular/common/http/testing' import { of } from 'rxjs' import { NgbModule } from '@ng-bootstrap/ng-bootstrap' +import { ClipboardService } from 'ngx-clipboard' const toasts = [ { @@ -41,6 +42,7 @@ describe('ToastsComponent', () => { let component: ToastsComponent let fixture: ComponentFixture let toastService: ToastService + let clipboardService: ClipboardService beforeEach(async () => { TestBed.configureTestingModule({ @@ -57,9 +59,10 @@ describe('ToastsComponent', () => { }).compileComponents() fixture = TestBed.createComponent(ToastsComponent) - component = fixture.componentInstance - toastService = TestBed.inject(ToastService) + clipboardService = TestBed.inject(ClipboardService) + + component = fixture.componentInstance fixture.detectChanges() }) @@ -114,7 +117,7 @@ describe('ToastsComponent', () => { 'Error 2 message details' ) - const copySpy = jest.spyOn(navigator.clipboard, 'writeText') + const copySpy = jest.spyOn(clipboardService, 'copy') component.copyError(toasts[2].error) expect(copySpy).toHaveBeenCalled() diff --git a/src-ui/src/app/components/common/toasts/toasts.component.ts b/src-ui/src/app/components/common/toasts/toasts.component.ts index aa304ac4d..bdabe33ee 100644 --- a/src-ui/src/app/components/common/toasts/toasts.component.ts +++ b/src-ui/src/app/components/common/toasts/toasts.component.ts @@ -1,6 +1,7 @@ import { Component, OnDestroy, OnInit } from '@angular/core' import { Subscription } from 'rxjs' import { Toast, ToastService } from 'src/app/services/toast.service' +import { ClipboardService } from 'ngx-clipboard' @Component({ selector: 'app-toasts', @@ -8,7 +9,10 @@ import { Toast, ToastService } from 'src/app/services/toast.service' styleUrls: ['./toasts.component.scss'], }) export class ToastsComponent implements OnInit, OnDestroy { - constructor(private toastService: ToastService) {} + constructor( + private toastService: ToastService, + private clipboardService: ClipboardService + ) {} private subscription: Subscription @@ -45,7 +49,7 @@ export class ToastsComponent implements OnInit, OnDestroy { } public copyError(error: any) { - navigator.clipboard.writeText(JSON.stringify(error)) + this.clipboardService.copy(JSON.stringify(error)) this.copied = true setTimeout(() => { this.copied = false 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 41e7a78d1..6b91438f9 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 @@ -5,16 +5,15 @@
of {{previewNumPages}}
-
- - + Download @@ -25,20 +24,31 @@ Download original
- - +
+ +
+ - + +
+
+ +