From 9e93ae952a9120baa638e43756340a440799546a Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Mon, 18 Dec 2023 08:41:51 -0800 Subject: [PATCH] Enhancement: Improved popup preview, respect embedded viewer, error handling (#4947) --- src-ui/messages.xlf | 17 +++- src-ui/src/app/app.module.ts | 2 + .../preview-popup.component.html | 22 +++++ .../preview-popup.component.scss | 9 ++ .../preview-popup.component.spec.ts | 92 +++++++++++++++++++ .../preview-popup/preview-popup.component.ts | 52 +++++++++++ .../saved-view-widget.component.html | 6 +- .../saved-view-widget.component.spec.ts | 11 ++- .../saved-view-widget.component.ts | 32 +++++-- .../document-card-large.component.html | 2 +- .../document-card-large.component.spec.ts | 2 + .../document-card-large.component.ts | 5 +- .../document-card-small.component.html | 2 +- .../document-card-small.component.spec.ts | 2 + .../document-card-small.component.ts | 5 +- .../popover-preview/popover-preview.scss | 22 ----- src-ui/src/styles.scss | 5 + 17 files changed, 235 insertions(+), 53 deletions(-) create mode 100644 src-ui/src/app/components/common/preview-popup/preview-popup.component.html create mode 100644 src-ui/src/app/components/common/preview-popup/preview-popup.component.scss create mode 100644 src-ui/src/app/components/common/preview-popup/preview-popup.component.spec.ts create mode 100644 src-ui/src/app/components/common/preview-popup/preview-popup.component.ts delete mode 100644 src-ui/src/app/components/document-list/popover-preview/popover-preview.scss diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index d9c8836cd..2d8d57be5 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -1408,7 +1408,7 @@ src/app/components/app-frame/app-frame.component.ts - 116 + 117 @@ -2257,21 +2257,21 @@ Sidebar views updated src/app/components/app-frame/app-frame.component.ts - 252 + 259 Error updating sidebar views src/app/components/app-frame/app-frame.component.ts - 255 + 262 An error occurred while saving update checking settings. src/app/components/app-frame/app-frame.component.ts - 276 + 283 @@ -3604,7 +3604,7 @@ src/app/components/document-list/document-card-small/document-card-small.component.ts - 80 + 77 @@ -3705,6 +3705,13 @@ 61 + + Error loading preview + + src/app/components/common/preview-popup/preview-popup.component.html + 3 + + Edit Profile diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 6910061d2..c3b98549a 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -107,6 +107,7 @@ import { CustomFieldsDropdownComponent } from './components/common/custom-fields import { ProfileEditDialogComponent } from './components/common/profile-edit-dialog/profile-edit-dialog.component' import { PdfViewerComponent } from './components/common/pdf-viewer/pdf-viewer.component' import { DocumentLinkComponent } from './components/common/input/document-link/document-link.component' +import { PreviewPopupComponent } from './components/common/preview-popup/preview-popup.component' import localeAf from '@angular/common/locales/af' import localeAr from '@angular/common/locales/ar' @@ -261,6 +262,7 @@ function initializeApp(settings: SettingsService) { ProfileEditDialogComponent, PdfViewerComponent, DocumentLinkComponent, + PreviewPopupComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/common/preview-popup/preview-popup.component.html b/src-ui/src/app/components/common/preview-popup/preview-popup.component.html new file mode 100644 index 000000000..71c3faf1b --- /dev/null +++ b/src-ui/src/app/components/common/preview-popup/preview-popup.component.html @@ -0,0 +1,22 @@ + + + Error loading preview + + + + + + + + + + + + + + diff --git a/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss b/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss new file mode 100644 index 000000000..af8dc565a --- /dev/null +++ b/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss @@ -0,0 +1,9 @@ +.preview-popup-container > * { + width: 30rem !important; + height: 22rem !important; + overflow-y: scroll; +} + +::ng-deep .popover.popover-preview { + max-width: 32rem; +} diff --git a/src-ui/src/app/components/common/preview-popup/preview-popup.component.spec.ts b/src-ui/src/app/components/common/preview-popup/preview-popup.component.spec.ts new file mode 100644 index 000000000..edcf63f51 --- /dev/null +++ b/src-ui/src/app/components/common/preview-popup/preview-popup.component.spec.ts @@ -0,0 +1,92 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' + +import { PreviewPopupComponent } from './preview-popup.component' +import { PdfViewerComponent } from '../pdf-viewer/pdf-viewer.component' +import { By } from '@angular/platform-browser' +import { SafeUrlPipe } from 'src/app/pipes/safeurl.pipe' +import { SettingsService } from 'src/app/services/settings.service' +import { SETTINGS_KEYS } from 'src/app/data/paperless-uisettings' +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { DocumentService } from 'src/app/services/rest/document.service' + +const doc = { + id: 10, + title: 'Document 10', + content: 'Cupcake ipsum dolor sit amet ice cream.', + original_file_name: 'sample.pdf', +} + +describe('PreviewPopupComponent', () => { + let component: PreviewPopupComponent + let fixture: ComponentFixture + let settingsService: SettingsService + let documentService: DocumentService + + beforeEach(() => { + TestBed.configureTestingModule({ + declarations: [PreviewPopupComponent, PdfViewerComponent, SafeUrlPipe], + imports: [HttpClientTestingModule], + }) + settingsService = TestBed.inject(SettingsService) + documentService = TestBed.inject(DocumentService) + jest + .spyOn(documentService, 'getPreviewUrl') + .mockImplementation((id) => doc.original_file_name) + fixture = TestBed.createComponent(PreviewPopupComponent) + component = fixture.componentInstance + component.document = doc + fixture.detectChanges() + }) + + it('should guess if file is pdf by file name', () => { + expect(component.isPdf).toBeTruthy() + component.document.archived_file_name = 'sample.pdf' + expect(component.isPdf).toBeTruthy() + component.document.archived_file_name = undefined + component.document.original_file_name = 'sample.txt' + expect(component.isPdf).toBeFalsy() + component.document.original_file_name = 'sample.pdf' + }) + + it('should return settings for native PDF viewer', () => { + settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, false) + expect(component.useNativePdfViewer).toBeFalsy() + settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, true) + expect(component.useNativePdfViewer).toBeTruthy() + }) + + it('should render object if native PDF viewer enabled', () => { + settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, true) + fixture.detectChanges() + expect(fixture.debugElement.query(By.css('object'))).not.toBeNull() + }) + + it('should render pngx viewer if native PDF viewer disabled', () => { + settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, false) + fixture.detectChanges() + expect(fixture.debugElement.query(By.css('object'))).toBeNull() + expect(fixture.debugElement.query(By.css('pngx-pdf-viewer'))).not.toBeNull() + }) + + it('should show lock icon on password error', () => { + settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, false) + component.onError({ name: 'PasswordException' }) + fixture.detectChanges() + expect(component.requiresPassword).toBeTruthy() + expect(fixture.debugElement.query(By.css('svg'))).not.toBeNull() + }) + + it('should fall back to object for non-pdf', () => { + component.document.original_file_name = 'sample.png' + fixture.detectChanges() + expect(fixture.debugElement.query(By.css('object'))).not.toBeNull() + }) + + it('should show message on error', () => { + component.onError({}) + fixture.detectChanges() + expect(fixture.debugElement.nativeElement.textContent).toContain( + 'Error loading preview' + ) + }) +}) diff --git a/src-ui/src/app/components/common/preview-popup/preview-popup.component.ts b/src-ui/src/app/components/common/preview-popup/preview-popup.component.ts new file mode 100644 index 000000000..014e31e8b --- /dev/null +++ b/src-ui/src/app/components/common/preview-popup/preview-popup.component.ts @@ -0,0 +1,52 @@ +import { Component, Input } from '@angular/core' +import { PaperlessDocument } from 'src/app/data/paperless-document' +import { SETTINGS_KEYS } from 'src/app/data/paperless-uisettings' +import { DocumentService } from 'src/app/services/rest/document.service' +import { SettingsService } from 'src/app/services/settings.service' + +@Component({ + selector: 'pngx-preview-popup', + templateUrl: './preview-popup.component.html', + styleUrls: ['./preview-popup.component.scss'], +}) +export class PreviewPopupComponent { + @Input() + document: PaperlessDocument + + error = false + + requiresPassword: boolean = false + + get renderAsObject(): boolean { + return (this.isPdf && this.useNativePdfViewer) || !this.isPdf + } + + get previewURL() { + return this.documentService.getPreviewUrl(this.document.id) + } + + get useNativePdfViewer(): boolean { + return this.settingsService.get(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER) + } + + get isPdf(): boolean { + // We dont have time to retrieve metadata, make a best guess by file name + return ( + this.document?.original_file_name?.endsWith('.pdf') || + this.document?.archived_file_name?.endsWith('.pdf') + ) + } + + constructor( + private settingsService: SettingsService, + private documentService: DocumentService + ) {} + + onError(event: any) { + if (event.name == 'PasswordException') { + this.requiresPassword = true + } else { + this.error = true + } + } +} diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html index 637a28a54..5c1eb6864 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html @@ -17,7 +17,7 @@ - + {{doc.created_date | customDate}} {{doc.title | documentTitle}} @@ -30,13 +30,13 @@ + autoClose="true" popoverClass="shadow popover-preview" container="body" (mouseenter)="mouseEnterPreviewButton(doc)" (mouseleave)="mouseLeavePreviewButton()" #popover="ngbPopover"> - + diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.spec.ts b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.spec.ts index e4a3041fc..84d1fb654 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.spec.ts +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.spec.ts @@ -29,6 +29,7 @@ import { SavedViewWidgetComponent } from './saved-view-widget.component' import { By } from '@angular/platform-browser' import { SafeUrlPipe } from 'src/app/pipes/safeurl.pipe' import { DragDropModule } from '@angular/cdk/drag-drop' +import { PreviewPopupComponent } from 'src/app/components/common/preview-popup/preview-popup.component' const savedView: PaperlessSavedView = { id: 1, @@ -74,6 +75,7 @@ describe('SavedViewWidgetComponent', () => { CustomDatePipe, DocumentTitlePipe, SafeUrlPipe, + PreviewPopupComponent, ], providers: [ PermissionsGuard, @@ -137,15 +139,18 @@ describe('SavedViewWidgetComponent', () => { ) component.ngOnInit() fixture.detectChanges() - component.mouseEnterPreview(documentResults[0]) + component.mouseEnterPreviewButton(documentResults[0]) expect(component.popover.isOpen()).toBeTruthy() expect(component.popoverHidden).toBeTruthy() tick(600) expect(component.popoverHidden).toBeFalsy() - component.mouseLeaveCard() + component.maybeClosePopover() - component.mouseEnterPreview(documentResults[1]) + component.mouseEnterPreviewButton(documentResults[1]) tick(100) + component.mouseLeavePreviewButton() + component.mouseEnterPreview() + expect(component.popover.isOpen()).toBeTruthy() component.mouseLeavePreview() tick(600) expect(component.popover.isOpen()).toBeFalsy() diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts index 982aeebaa..601d9384e 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts @@ -26,10 +26,7 @@ import { queryParamsFromFilterRules } from 'src/app/utils/query-params' @Component({ selector: 'pngx-saved-view-widget', templateUrl: './saved-view-widget.component.html', - styleUrls: [ - './saved-view-widget.component.scss', - '../../../document-list/popover-preview/popover-preview.scss', - ], + styleUrls: ['./saved-view-widget.component.scss'], }) export class SavedViewWidgetComponent extends ComponentWithPermissions @@ -121,8 +118,11 @@ export class SavedViewWidgetComponent return this.documentService.getDownloadUrl(document.id) } - mouseEnterPreview(doc: PaperlessDocument) { - this.popover = this.popovers.get(this.documents.indexOf(doc)) + mouseEnterPreviewButton(doc: PaperlessDocument) { + const newPopover = this.popovers.get(this.documents.indexOf(doc)) + if (this.popover !== newPopover && this.popover?.isOpen()) + this.popover.close() + this.popover = newPopover this.mouseOnPreview = true if (!this.popover.isOpen()) { // we're going to open but hide to pre-load content during hover delay @@ -139,12 +139,24 @@ export class SavedViewWidgetComponent } } - mouseLeavePreview() { - this.mouseOnPreview = false + mouseEnterPreview() { + this.mouseOnPreview = true } - mouseLeaveCard() { - this.popover?.close() + mouseLeavePreview() { + this.mouseOnPreview = false + this.maybeClosePopover() + } + + mouseLeavePreviewButton() { + this.mouseOnPreview = false + this.maybeClosePopover() + } + + maybeClosePopover() { + setTimeout(() => { + if (!this.mouseOnPreview) this.popover?.close() + }, 300) } getCorrespondentQueryParams(correspondentId: number): Params { diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html index ba4608d3c..b59c934f2 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html @@ -56,7 +56,7 @@ View - + diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.spec.ts b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.spec.ts index 7b3d0a5bb..7bdd8422c 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.spec.ts +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.spec.ts @@ -19,6 +19,7 @@ import { DocumentTitlePipe } from 'src/app/pipes/document-title.pipe' import { SafeUrlPipe } from 'src/app/pipes/safeurl.pipe' import { DocumentCardLargeComponent } from './document-card-large.component' import { IsNumberPipe } from 'src/app/pipes/is-number.pipe' +import { PreviewPopupComponent } from '../../common/preview-popup/preview-popup.component' const doc = { id: 10, @@ -50,6 +51,7 @@ describe('DocumentCardLargeComponent', () => { IfPermissionsDirective, SafeUrlPipe, IsNumberPipe, + PreviewPopupComponent, ], providers: [DatePipe], imports: [ diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts index 1725e5795..deb855449 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts @@ -15,10 +15,7 @@ import { ComponentWithPermissions } from '../../with-permissions/with-permission @Component({ selector: 'pngx-document-card-large', templateUrl: './document-card-large.component.html', - styleUrls: [ - './document-card-large.component.scss', - '../popover-preview/popover-preview.scss', - ], + styleUrls: ['./document-card-large.component.scss'], }) export class DocumentCardLargeComponent extends ComponentWithPermissions { constructor( diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html index 8719f2ddf..c945384d6 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html @@ -94,7 +94,7 @@ - + diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.spec.ts b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.spec.ts index b4b5efe11..28db4502e 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.spec.ts +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.spec.ts @@ -22,6 +22,7 @@ import { By } from '@angular/platform-browser' import { TagComponent } from '../../common/tag/tag.component' import { PaperlessTag } from 'src/app/data/paperless-tag' import { IsNumberPipe } from 'src/app/pipes/is-number.pipe' +import { PreviewPopupComponent } from '../../common/preview-popup/preview-popup.component' const doc = { id: 10, @@ -64,6 +65,7 @@ describe('DocumentCardSmallComponent', () => { SafeUrlPipe, TagComponent, IsNumberPipe, + PreviewPopupComponent, ], providers: [DatePipe], imports: [ diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts index 65ee5e6ad..41ef958fb 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts @@ -16,10 +16,7 @@ import { ComponentWithPermissions } from '../../with-permissions/with-permission @Component({ selector: 'pngx-document-card-small', templateUrl: './document-card-small.component.html', - styleUrls: [ - './document-card-small.component.scss', - '../popover-preview/popover-preview.scss', - ], + styleUrls: ['./document-card-small.component.scss'], }) export class DocumentCardSmallComponent extends ComponentWithPermissions { constructor( diff --git a/src-ui/src/app/components/document-list/popover-preview/popover-preview.scss b/src-ui/src/app/components/document-list/popover-preview/popover-preview.scss deleted file mode 100644 index b51e2e66b..000000000 --- a/src-ui/src/app/components/document-list/popover-preview/popover-preview.scss +++ /dev/null @@ -1,22 +0,0 @@ -::ng-deep .popover.popover-preview { - max-width: 40rem; - - .preview { - min-width: 30rem; - min-height: 18rem; - max-height: 35rem; - overflow-y: scroll; - } - - .spinner-border { - position: absolute; - top: 4rem; - left: calc(50% - 0.5rem); - z-index: 0; - } -} - -::ng-deep .popover-hidden .popover { - opacity: 0; - pointer-events: none; -} diff --git a/src-ui/src/styles.scss b/src-ui/src/styles.scss index 19898374b..e128b27fa 100644 --- a/src-ui/src/styles.scss +++ b/src-ui/src/styles.scss @@ -555,6 +555,11 @@ table.table { } } +.popover-hidden .popover { + opacity: 0; + pointer-events: none; +} + // Tour .tour-active .popover { min-width: 360px;
Error loading preview