diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index a3e9b3330..b04f86ebc 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -1931,7 +1931,7 @@ src/app/components/document-detail/document-detail.component.ts - 701 + 716 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -1970,7 +1970,7 @@ src/app/components/document-detail/document-detail.component.ts - 703 + 718 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -4214,31 +4214,31 @@ - Hello , welcome to + Hello , welcome to src/app/components/dashboard/dashboard.component.ts - 41 + 38 - Welcome to + Welcome to src/app/components/dashboard/dashboard.component.ts - 43 + 40 Dashboard updated src/app/components/dashboard/dashboard.component.ts - 74 + 71 Error updating dashboard src/app/components/dashboard/dashboard.component.ts - 77 + 74 @@ -4752,7 +4752,7 @@ - Notes + Notes src/app/components/document-detail/document-detail.component.html 286,289 @@ -4787,78 +4787,78 @@ An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 276,278 + 284,286 Document changes detected src/app/components/document-detail/document-detail.component.ts - 298 + 307 The version of this document in your browser session appears older than the existing version. src/app/components/document-detail/document-detail.component.ts - 299 + 308 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 - 300 + 309 Ok src/app/components/document-detail/document-detail.component.ts - 301 + 311 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 437 + 448 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 458 + 473 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 576 + 591 src/app/components/document-detail/document-detail.component.ts - 585 + 600 Error saving document src/app/components/document-detail/document-detail.component.ts - 589 + 604 src/app/components/document-detail/document-detail.component.ts - 630 + 645 Confirm delete src/app/components/document-detail/document-detail.component.ts - 656 + 671 src/app/components/manage/management-list/management-list.component.ts @@ -4869,35 +4869,35 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 657 + 672 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 658 + 673 Delete document src/app/components/document-detail/document-detail.component.ts - 660 + 675 Error deleting document src/app/components/document-detail/document-detail.component.ts - 679 + 694 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 699 + 714 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -4908,28 +4908,28 @@ This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 700 + 715 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 - 711 + 726 Error executing operation src/app/components/document-detail/document-detail.component.ts - 722 + 737 Page Fit src/app/components/document-detail/document-detail.component.ts - 791 + 806 @@ -7095,21 +7095,21 @@ Successfully completed one-time migratration of settings to the database! src/app/services/settings.service.ts - 468 + 471 Unable to migrate settings to the database, please try saving manually. src/app/services/settings.service.ts - 469 + 472 You can restart the tour from the settings page. src/app/services/settings.service.ts - 539 + 542 diff --git a/src-ui/src/app/components/common/pdf-viewer/pdf-viewer.component.ts b/src-ui/src/app/components/common/pdf-viewer/pdf-viewer.component.ts index 472fcab6a..274b06032 100644 --- a/src-ui/src/app/components/common/pdf-viewer/pdf-viewer.component.ts +++ b/src-ui/src/app/components/common/pdf-viewer/pdf-viewer.component.ts @@ -465,33 +465,42 @@ export class PdfViewerComponent this.clear() - this.setupViewer() - - this.loadingTask = PDFJS.getDocument(this.getDocumentParams()) - - this.loadingTask!.onProgress = (progressData: PDFProgressData) => { - this.onProgress.emit(progressData) + if (this.pdfViewer) { + this.pdfViewer._resetView() + this.pdfViewer = null } - const src = this.src + this.setupViewer() - from(this.loadingTask!.promise as Promise) - .pipe(takeUntil(this.destroy$)) - .subscribe({ - next: (pdf) => { - this._pdf = pdf - this.lastLoaded = src + try { + this.loadingTask = PDFJS.getDocument(this.getDocumentParams()) - this.afterLoadComplete.emit(pdf) - this.resetPdfDocument() + this.loadingTask!.onProgress = (progressData: PDFProgressData) => { + this.onProgress.emit(progressData) + } - this.update() - }, - error: (error) => { - this.lastLoaded = null - this.onError.emit(error) - }, - }) + const src = this.src + + from(this.loadingTask!.promise as Promise) + .pipe(takeUntil(this.destroy$)) + .subscribe({ + next: (pdf) => { + this._pdf = pdf + this.lastLoaded = src + + this.afterLoadComplete.emit(pdf) + this.resetPdfDocument() + + this.update() + }, + error: (error) => { + this.lastLoaded = null + this.onError.emit(error) + }, + }) + } catch (e) { + this.onError.emit(e) + } } private update() { 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 5000af93d..eaf7d298a 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 @@ -21,361 +21,361 @@ + Delete + -
- - +
+ + - Download - + Download + - @if (metadata?.has_archive_version) { -
- - -
- } + @if (metadata?.has_archive_version) { + + } +
-
- -
- +
+ +
+ - -
-
+ +
+
- - + + - - + + -
-
+
+
-
- -
-
- - - -
- - -
- - - -
- -
-
- -
- - @if (renderAsPlainText) { -
- } - @if (requiresPassword) { -
-
- -
-
- } -
+
+
+
+ + +
- -
- - - @if (hasNext()) { - - } - @if (!hasNext()) { - - } - - -
-
+ +
- - @if (!metadata) { -
-
-
- Loading... -
+ + +
+ + +
+ +
+ + @if (renderAsPlainText) { +
+ } + @if (requiresPassword) { +
+
+ +
+
+ } +
+ +
+ + +
+ + + @if (hasNext()) { + + } + @if (!hasNext()) { + + } + + +
+
+ + + @if (!metadata) { +
+
+
+ Loading... +
+
+ } + @if (getContentType() === 'application/pdf') { + @if (!useNativePdfViewer ) { +
+ + +
+ } @else { + + } + } + @if (renderAsPlainText) { +
+ } + @if (showPasswordField) { +
+
+ +
+
+ } +
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 1492c3e0e..51a77fb91 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 @@ -255,9 +255,6 @@ describe('DocumentDetailComponent', () => { router = TestBed.inject(Router) activatedRoute = TestBed.inject(ActivatedRoute) - jest - .spyOn(activatedRoute, 'paramMap', 'get') - .mockReturnValue(of(convertToParamMap({ id: 3 }))) openDocumentsService = TestBed.inject(OpenDocumentsService) documentService = TestBed.inject(DocumentService) modalService = TestBed.inject(NgbModal) @@ -295,6 +292,17 @@ describe('DocumentDetailComponent', () => { expect(navigateSpy).toHaveBeenCalledWith(['documents', 3, 'notes']) }) + it('should forward id without section to details', () => { + const navigateSpy = jest.spyOn(router, 'navigate') + jest + .spyOn(activatedRoute, 'paramMap', 'get') + .mockReturnValue(of(convertToParamMap({ id: 3 }))) + fixture.detectChanges() + expect(navigateSpy).toHaveBeenCalledWith(['documents', 3, 'details'], { + replaceUrl: true, + }) + }) + it('should update title after debounce', fakeAsync(() => { initNormally() component.titleInput.value = 'Foo Bar' @@ -320,6 +328,7 @@ describe('DocumentDetailComponent', () => { }) it('should load already-opened document via param', () => { + initNormally() jest.spyOn(documentService, 'get').mockReturnValueOnce(of(doc)) jest.spyOn(openDocumentsService, 'getOpenDocument').mockReturnValue(doc) jest.spyOn(customFieldsService, 'listAll').mockReturnValue( @@ -400,8 +409,11 @@ describe('DocumentDetailComponent', () => { }) it('should 404 on invalid id', () => { - jest.spyOn(documentService, 'get').mockReturnValueOnce(of(null)) const navigateSpy = jest.spyOn(router, 'navigate') + jest + .spyOn(activatedRoute, 'paramMap', 'get') + .mockReturnValue(of(convertToParamMap({ id: 999, section: 'details' }))) + jest.spyOn(documentService, 'get').mockReturnValueOnce(of(null)) fixture.detectChanges() expect(navigateSpy).toHaveBeenCalledWith(['404'], { replaceUrl: true }) }) @@ -936,11 +948,33 @@ describe('DocumentDetailComponent', () => { expect(refreshSpy).toHaveBeenCalled() }) + it('should get suggestions', () => { + const suggestionsSpy = jest.spyOn(documentService, 'getSuggestions') + suggestionsSpy.mockReturnValue(of({ tags: [1, 2] })) + initNormally() + expect(suggestionsSpy).toHaveBeenCalled() + expect(component.suggestions).toEqual({ tags: [1, 2] }) + }) + + it('should show error if needed for get suggestions', () => { + const suggestionsSpy = jest.spyOn(documentService, 'getSuggestions') + const errorSpy = jest.spyOn(toastService, 'showError') + suggestionsSpy.mockImplementationOnce(() => + throwError(() => new Error('failed to get suggestions')) + ) + initNormally() + expect(suggestionsSpy).toHaveBeenCalled() + expect(errorSpy).toHaveBeenCalled() + }) + it('should warn when open document does not match doc retrieved from backend on init', () => { const modalSpy = jest.spyOn(modalService, 'open') const openDoc = Object.assign({}, doc) // simulate a document being modified elsewhere and db updated doc.modified = new Date() + jest + .spyOn(activatedRoute, 'paramMap', 'get') + .mockReturnValue(of(convertToParamMap({ id: 3, section: 'details' }))) jest.spyOn(documentService, 'get').mockReturnValueOnce(of(doc)) jest.spyOn(openDocumentsService, 'getOpenDocument').mockReturnValue(openDoc) jest.spyOn(customFieldsService, 'listAll').mockReturnValue( @@ -951,12 +985,13 @@ describe('DocumentDetailComponent', () => { }) ) fixture.detectChanges() // calls ngOnInit - expect(modalSpy).toHaveBeenCalledWith(ConfirmDialogComponent, { - backdrop: 'static', - }) + expect(modalSpy).toHaveBeenCalledWith(ConfirmDialogComponent) }) function initNormally() { + jest + .spyOn(activatedRoute, 'paramMap', 'get') + .mockReturnValue(of(convertToParamMap({ id: 3, section: 'details' }))) jest .spyOn(documentService, 'get') .mockReturnValueOnce(of(Object.assign({}, doc))) 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 83057b8cd..16eb9599c 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 @@ -33,6 +33,7 @@ import { map, debounceTime, distinctUntilChanged, + filter, } from 'rxjs/operators' import { DocumentSuggestions } from 'src/app/data/document-suggestions' import { @@ -257,6 +258,13 @@ export class DocumentDetailComponent this.route.paramMap .pipe( + filter((paramMap) => { + // only init when changing docs & section is set + return ( + +paramMap.get('id') !== this.documentId && + paramMap.get('section')?.length > 0 + ) + }), takeUntil(this.unsubscribeNotifier), switchMap((paramMap) => { const documentId = +paramMap.get('id') @@ -295,15 +303,12 @@ export class DocumentDetailComponent new Date(doc.modified) > new Date(openDocument.modified) && !this.modalService.hasOpenModals() ) { - let modal = this.modalService.open(ConfirmDialogComponent, { - backdrop: 'static', - }) + let modal = this.modalService.open(ConfirmDialogComponent) modal.componentInstance.title = $localize`Document changes detected` modal.componentInstance.messageBold = $localize`The version of this document in your browser session appears older than the existing version.` modal.componentInstance.message = $localize`Saving the document here may overwrite other changes that were made. To restore the existing version, discard your changes or close the document.` - modal.componentInstance.cancelBtnCaption = $localize`Ok` - modal.componentInstance.cancelBtnClass = 'btn-primary' - modal.componentInstance.btnClass = 'visually-hidden' + modal.componentInstance.cancelBtnClass = 'visually-hidden' + modal.componentInstance.btnCaption = $localize`Ok` } if (this.documentForm.dirty) { @@ -425,11 +430,14 @@ export class DocumentDetailComponent updateComponent(doc: Document) { this.document = doc this.requiresPassword = false - // this.customFields = doc.custom_fields.concat([]) this.updateFormForCustomFields() this.documentsService .getMetadata(doc.id) - .pipe(first()) + .pipe( + first(), + takeUntil(this.unsubscribeNotifier), + takeUntil(this.docChangeNotifier) + ) .subscribe({ next: (result) => { this.metadata = result @@ -450,7 +458,11 @@ export class DocumentDetailComponent ) { this.documentsService .getSuggestions(doc.id) - .pipe(first(), takeUntil(this.unsubscribeNotifier)) + .pipe( + first(), + takeUntil(this.unsubscribeNotifier), + takeUntil(this.docChangeNotifier) + ) .subscribe({ next: (result) => { this.suggestions = result