diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 981394b87..7499a56ff 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -241,18 +241,18 @@ Document was added to Paperless-ngx. src/app/app.component.ts - 93 + 95 src/app/app.component.ts - 102 + 104 Open document src/app/app.component.ts - 95 + 97 src/app/components/admin/trash/trash.component.ts @@ -279,21 +279,21 @@ Could not add : src/app/app.component.ts - 117 + 119 Document is being processed by Paperless-ngx. src/app/app.component.ts - 132 + 134 Dashboard src/app/app.component.ts - 139 + 141 src/app/components/app-frame/app-frame.component.html @@ -312,7 +312,7 @@ Documents src/app/app.component.ts - 150 + 152 src/app/components/app-frame/app-frame.component.html @@ -351,7 +351,7 @@ Settings src/app/app.component.ts - 162 + 164 src/app/components/admin/settings/settings.component.html @@ -374,74 +374,74 @@ Prev src/app/app.component.ts - 168 + 170 Next src/app/app.component.ts - 169 + 171 src/app/components/document-detail/document-detail.component.html - 95 + 100 End src/app/app.component.ts - 170 + 172 The dashboard can be used to show saved views, such as an 'Inbox'. Views are found under Manage > Saved Views once you have created some. src/app/app.component.ts - 176 + 178 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 - 183 + 185 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 - 188 + 190 The filtering tools allow you to quickly find documents using various searches, dates, tags, etc. src/app/app.component.ts - 195 + 197 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 - 201 + 203 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 - 206 + 208 Manage e-mail accounts and rules for automatically importing documents. src/app/app.component.ts - 214 + 216 src/app/components/manage/mail/mail.component.html @@ -452,14 +452,14 @@ Workflows give you more control over the document pipeline. src/app/app.component.ts - 222 + 224 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 - 230 + 232 src/app/components/admin/tasks/tasks.component.html @@ -470,28 +470,28 @@ Check out the settings for various tweaks to the web app. src/app/app.component.ts - 238 + 240 Thank you! 🙏 src/app/app.component.ts - 246 + 248 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 - 248 + 250 Lastly, on behalf of every contributor to this community-supported project, thank you for using Paperless-ngx! src/app/app.component.ts - 250 + 252 @@ -534,7 +534,7 @@ src/app/components/document-detail/document-detail.component.html - 348 + 353 @@ -593,7 +593,7 @@ src/app/components/document-detail/document-detail.component.html - 341 + 346 src/app/components/document-list/bulk-editor/custom-fields-bulk-edit-dialog/custom-fields-bulk-edit-dialog.component.html @@ -739,7 +739,7 @@ src/app/components/document-detail/document-detail.component.html - 361 + 366 src/app/components/document-list/document-list.component.html @@ -1162,7 +1162,7 @@ src/app/components/document-detail/document-detail.component.html - 317 + 322 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1757,7 +1757,7 @@ src/app/components/document-detail/document-detail.component.html - 45 + 50 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2232,7 +2232,7 @@ src/app/components/document-detail/document-detail.component.ts - 924 + 926 @@ -2509,19 +2509,19 @@ src/app/components/document-detail/document-detail.component.ts - 948 + 950 src/app/components/document-detail/document-detail.component.ts - 1255 + 1303 src/app/components/document-detail/document-detail.component.ts - 1294 + 1342 src/app/components/document-detail/document-detail.component.ts - 1335 + 1383 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2954,7 +2954,7 @@ src/app/components/document-detail/document-detail.component.html - 29 + 34 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -3115,7 +3115,7 @@ src/app/components/document-detail/document-detail.component.ts - 901 + 903 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -4142,7 +4142,7 @@ src/app/components/document-detail/document-detail.component.html - 283 + 288 @@ -4974,7 +4974,7 @@ Not assigned src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts - 380 + 392 Filter drop down element to filter for documents with no correspondent/type/tag assigned @@ -4982,7 +4982,7 @@ Open filter src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts - 501 + 513 @@ -6183,14 +6183,14 @@ Download original src/app/components/document-detail/document-detail.component.html - 36 + 41 Reprocess src/app/components/document-detail/document-detail.component.html - 49 + 54 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6201,7 +6201,7 @@ More like this src/app/components/document-detail/document-detail.component.html - 53 + 58 src/app/components/document-list/document-card-large/document-card-large.component.html @@ -6212,14 +6212,14 @@ Split src/app/components/document-detail/document-detail.component.html - 57 + 62 Rotate src/app/components/document-detail/document-detail.component.html - 61 + 66 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6230,18 +6230,18 @@ Delete page(s) src/app/components/document-detail/document-detail.component.html - 65 + 70 Close src/app/components/document-detail/document-detail.component.html - 89 + 94 src/app/components/document-detail/document-detail.component.ts - 1312 + 1360 src/app/guards/dirty-saved-view.guard.ts @@ -6252,21 +6252,21 @@ Previous src/app/components/document-detail/document-detail.component.html - 92 + 97 Details src/app/components/document-detail/document-detail.component.html - 105 + 110 Title src/app/components/document-detail/document-detail.component.html - 108 + 113 src/app/components/document-list/document-list.component.html @@ -6289,21 +6289,21 @@ Archive serial number src/app/components/document-detail/document-detail.component.html - 109 + 114 Date created src/app/components/document-detail/document-detail.component.html - 110 + 115 Correspondent src/app/components/document-detail/document-detail.component.html - 112 + 117 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6330,7 +6330,7 @@ Document type src/app/components/document-detail/document-detail.component.html - 114 + 119 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6357,7 +6357,7 @@ Storage path src/app/components/document-detail/document-detail.component.html - 116 + 121 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6380,7 +6380,7 @@ Default src/app/components/document-detail/document-detail.component.html - 117 + 122 src/app/components/manage/saved-views/saved-views.component.html @@ -6391,14 +6391,14 @@ Content src/app/components/document-detail/document-detail.component.html - 213 + 218 Metadata src/app/components/document-detail/document-detail.component.html - 222 + 227 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -6409,175 +6409,175 @@ Date modified src/app/components/document-detail/document-detail.component.html - 229 + 234 Date added src/app/components/document-detail/document-detail.component.html - 233 + 238 Media filename src/app/components/document-detail/document-detail.component.html - 237 + 242 Original filename src/app/components/document-detail/document-detail.component.html - 241 + 246 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 245 + 250 Original file size src/app/components/document-detail/document-detail.component.html - 249 + 254 Original mime type src/app/components/document-detail/document-detail.component.html - 253 + 258 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 258 + 263 Archive file size src/app/components/document-detail/document-detail.component.html - 264 + 269 Original document metadata src/app/components/document-detail/document-detail.component.html - 273 + 278 Archived document metadata src/app/components/document-detail/document-detail.component.html - 276 + 281 Notes src/app/components/document-detail/document-detail.component.html - 295,298 + 300,303 History src/app/components/document-detail/document-detail.component.html - 306 + 311 Save & next src/app/components/document-detail/document-detail.component.html - 343 + 348 Save & close src/app/components/document-detail/document-detail.component.html - 346 + 351 Document loading... src/app/components/document-detail/document-detail.component.html - 356 + 361 Enter Password src/app/components/document-detail/document-detail.component.html - 410 + 415 An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 406,408 + 411,413 Document changes detected src/app/components/document-detail/document-detail.component.ts - 436 + 434 The version of this document in your browser session appears older than the existing version. src/app/components/document-detail/document-detail.component.ts - 437 + 435 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 - 438 + 436 Ok src/app/components/document-detail/document-detail.component.ts - 440 + 438 Next document src/app/components/document-detail/document-detail.component.ts - 547 + 545 Previous document src/app/components/document-detail/document-detail.component.ts - 557 + 555 Close document src/app/components/document-detail/document-detail.component.ts - 565 + 563 src/app/services/open-documents.service.ts @@ -6588,64 +6588,64 @@ Save document src/app/components/document-detail/document-detail.component.ts - 572 + 570 Save and close / next src/app/components/document-detail/document-detail.component.ts - 581 + 579 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 633 + 631 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 662 + 660 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 813 + 811 src/app/components/document-detail/document-detail.component.ts - 827 + 825 Error saving document src/app/components/document-detail/document-detail.component.ts - 831 + 829 src/app/components/document-detail/document-detail.component.ts - 874 + 872 Do you really want to move the document "" to the trash? src/app/components/document-detail/document-detail.component.ts - 902 + 904 Documents can be restored prior to permanent deletion. src/app/components/document-detail/document-detail.component.ts - 903 + 905 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -6656,7 +6656,7 @@ Move to trash src/app/components/document-detail/document-detail.component.ts - 905 + 907 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -6667,7 +6667,7 @@ Reprocess confirm src/app/components/document-detail/document-detail.component.ts - 944 + 946 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -6678,70 +6678,77 @@ This operation will permanently recreate the archive file for this document. src/app/components/document-detail/document-detail.component.ts - 945 + 947 The archive file will be re-generated with the current settings. src/app/components/document-detail/document-detail.component.ts - 946 + 948 Reprocess 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 - 956 + 958 Error executing operation src/app/components/document-detail/document-detail.component.ts - 967 + 969 + + + + Error downloading document + + src/app/components/document-detail/document-detail.component.ts + 1016 Page Fit src/app/components/document-detail/document-detail.component.ts - 1040 + 1088 Split confirm src/app/components/document-detail/document-detail.component.ts - 1253 + 1301 This operation will split the selected document(s) into new documents. src/app/components/document-detail/document-detail.component.ts - 1254 + 1302 Split operation will begin in the background. src/app/components/document-detail/document-detail.component.ts - 1270 + 1318 Error executing split operation src/app/components/document-detail/document-detail.component.ts - 1279 + 1327 Rotate confirm src/app/components/document-detail/document-detail.component.ts - 1292 + 1340 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -6752,60 +6759,60 @@ This operation will permanently rotate the original version of the current document. src/app/components/document-detail/document-detail.component.ts - 1293 + 1341 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 - 1309 + 1357 Error executing rotate operation src/app/components/document-detail/document-detail.component.ts - 1321 + 1369 Delete pages confirm src/app/components/document-detail/document-detail.component.ts - 1333 + 1381 This operation will permanently delete the selected pages from the original document. src/app/components/document-detail/document-detail.component.ts - 1334 + 1382 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 - 1349 + 1397 Error executing delete pages operation src/app/components/document-detail/document-detail.component.ts - 1358 + 1406 An error occurred loading tiff: src/app/components/document-detail/document-detail.component.ts - 1398 + 1446 src/app/components/document-detail/document-detail.component.ts - 1402 + 1450 diff --git a/src-ui/package-lock.json b/src-ui/package-lock.json index c09875247..060c1afc8 100644 --- a/src-ui/package-lock.json +++ b/src-ui/package-lock.json @@ -29,6 +29,7 @@ "ngx-bootstrap-icons": "^1.9.3", "ngx-color": "^9.0.0", "ngx-cookie-service": "^19.0.0", + "ngx-device-detector": "^9.0.0", "ngx-file-drop": "^16.0.0", "ngx-ui-tour-ng-bootstrap": "^16.0.0", "rxjs": "^7.8.1", @@ -14248,6 +14249,19 @@ "@angular/core": "^19.0.0" } }, + "node_modules/ngx-device-detector": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/ngx-device-detector/-/ngx-device-detector-9.0.0.tgz", + "integrity": "sha512-zpio/wqH1GnxIpWCdA7cp5fmWf7YLycgzfXzQHmB9vaS7eAcqpBF5mxDS65IhE7TzNTJziDrYJCT+9tVqBcsDg==", + "license": "MIT", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/common": "^19.0.0", + "@angular/core": "^19.0.0" + } + }, "node_modules/ngx-file-drop": { "version": "16.0.0", "resolved": "https://registry.npmjs.org/ngx-file-drop/-/ngx-file-drop-16.0.0.tgz", diff --git a/src-ui/package.json b/src-ui/package.json index f9d4d8158..0e19f4240 100644 --- a/src-ui/package.json +++ b/src-ui/package.json @@ -31,6 +31,7 @@ "ngx-bootstrap-icons": "^1.9.3", "ngx-color": "^9.0.0", "ngx-cookie-service": "^19.0.0", + "ngx-device-detector": "^9.0.0", "ngx-file-drop": "^16.0.0", "ngx-ui-tour-ng-bootstrap": "^16.0.0", "rxjs": "^7.8.1", diff --git a/src-ui/setup-jest.ts b/src-ui/setup-jest.ts index 244938606..52dccaf02 100644 --- a/src-ui/setup-jest.ts +++ b/src-ui/setup-jest.ts @@ -100,6 +100,15 @@ Object.defineProperty(navigator, 'clipboard', { }, }) Object.defineProperty(navigator, 'canShare', { value: () => true }) +if (!navigator.share) { + Object.defineProperty(navigator, 'share', { value: jest.fn() }) +} +if (!URL.createObjectURL) { + Object.defineProperty(window.URL, 'createObjectURL', { value: jest.fn() }) +} +if (!URL.revokeObjectURL) { + Object.defineProperty(window.URL, 'revokeObjectURL', { value: jest.fn() }) +} Object.defineProperty(window, 'ResizeObserver', { value: mock() }) Object.defineProperty(window, 'location', { configurable: true, 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 bc0bf41fd..02fec3cf7 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 @@ -25,15 +25,20 @@
- - Download - + @if (metadata?.has_archive_version) {
- +
} 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 0a2e5605f..8b2a84534 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 @@ -24,6 +24,7 @@ import { NgbModalRef, } from '@ng-bootstrap/ng-bootstrap' import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' +import { DeviceDetectorService } from 'ngx-device-detector' import { of, throwError } from 'rxjs' import { routes } from 'src/app/app-routing.module' import { Correspondent } from 'src/app/data/correspondent' @@ -127,6 +128,7 @@ describe('DocumentDetailComponent', () => { let documentListViewService: DocumentListViewService let settingsService: SettingsService let customFieldsService: CustomFieldsService + let deviceDetectorService: DeviceDetectorService let httpTestingController: HttpTestingController let componentRouterService: ComponentRouterService @@ -264,6 +266,7 @@ describe('DocumentDetailComponent', () => { settingsService = TestBed.inject(SettingsService) settingsService.currentUser = { id: 1 } customFieldsService = TestBed.inject(CustomFieldsService) + deviceDetectorService = TestBed.inject(DeviceDetectorService) fixture = TestBed.createComponent(DocumentDetailComponent) httpTestingController = TestBed.inject(HttpTestingController) componentRouterService = TestBed.inject(ComponentRouterService) @@ -1268,4 +1271,38 @@ describe('DocumentDetailComponent', () => { .error(new ErrorEvent('failed')) expect(component.tiffError).not.toBeUndefined() }) + + it('should support download using share sheet on mobile, direct download otherwise', () => { + const shareSpy = jest.spyOn(navigator, 'share') + const createSpy = jest.spyOn(document, 'createElement') + const urlRevokeSpy = jest.spyOn(URL, 'revokeObjectURL') + initNormally() + + // Mobile + jest.spyOn(deviceDetectorService, 'isDesktop').mockReturnValue(false) + component.download() + httpTestingController + .expectOne(`${environment.apiBaseUrl}documents/${doc.id}/download/`) + .error(new ProgressEvent('failed')) + expect(shareSpy).not.toHaveBeenCalled() + + component.download(true) + httpTestingController + .expectOne( + `${environment.apiBaseUrl}documents/${doc.id}/download/?original=true` + ) + .flush(new ArrayBuffer(100)) + expect(shareSpy).toHaveBeenCalled() + + // Desktop + shareSpy.mockClear() + jest.spyOn(deviceDetectorService, 'isDesktop').mockReturnValue(true) + component.download() + httpTestingController + .expectOne(`${environment.apiBaseUrl}documents/${doc.id}/download/`) + .flush(new ArrayBuffer(100)) + expect(shareSpy).not.toHaveBeenCalled() + expect(createSpy).toHaveBeenCalledWith('a') + expect(urlRevokeSpy).toHaveBeenCalled() + }) }) 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 8d1b35071..0378fbb97 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 @@ -20,6 +20,7 @@ import { import { dirtyCheck, DirtyComponent } from '@ngneat/dirty-check-forms' import { PDFDocumentProxy, PdfViewerModule } from 'ng2-pdf-viewer' import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' +import { DeviceDetectorService } from 'ngx-device-detector' import { BehaviorSubject, Observable, Subject } from 'rxjs' import { debounceTime, @@ -195,8 +196,6 @@ export class DocumentDetailComponent previewUrl: string thumbUrl: string previewText: string - downloadUrl: string - downloadOriginalUrl: string previewLoaded: boolean = false tiffURL: string tiffError: string @@ -234,6 +233,9 @@ export class DocumentDetailComponent ogDate: Date customFields: CustomField[] + + public downloading: boolean = false + public readonly CustomFieldDataType = CustomFieldDataType public readonly ContentRenderType = ContentRenderType @@ -274,7 +276,8 @@ export class DocumentDetailComponent private customFieldsService: CustomFieldsService, private http: HttpClient, private hotKeyService: HotKeyService, - private componentRouterService: ComponentRouterService + private componentRouterService: ComponentRouterService, + private deviceDetectorService: DeviceDetectorService ) { super() } @@ -417,13 +420,6 @@ export class DocumentDetailComponent .pipe( switchMap((doc) => { this.documentId = doc.id - this.downloadUrl = this.documentsService.getDownloadUrl( - this.documentId - ) - this.downloadOriginalUrl = this.documentsService.getDownloadUrl( - this.documentId, - true - ) this.suggestions = null const openDocument = this.openDocumentService.getOpenDocument( this.documentId @@ -978,6 +974,52 @@ export class DocumentDetailComponent }) } + download(original: boolean = false) { + this.downloading = true + const downloadUrl = this.documentsService.getDownloadUrl( + this.documentId, + original + ) + this.http.get(downloadUrl, { responseType: 'blob' }).subscribe({ + next: (blob) => { + this.downloading = false + const blobParts = [blob] + const file = new File( + blobParts, + original + ? this.document.original_file_name + : this.document.archived_file_name, + { + type: original ? this.document.mime_type : 'application/pdf', + } + ) + if ( + !this.deviceDetectorService.isDesktop() && + navigator.canShare && + navigator.canShare({ files: [file] }) + ) { + navigator.share({ + files: [file], + }) + } else { + const url = URL.createObjectURL(blob) + const a = document.createElement('a') + a.href = url + a.download = this.document.title + a.click() + URL.revokeObjectURL(url) + } + }, + error: (error) => { + this.downloading = false + this.toastService.showError( + $localize`Error downloading document`, + error + ) + }, + }) + } + hasNext() { return this.documentListViewService.hasNext(this.documentId) }