From 548a7f05d849e5a61fc06c748e52f98857ddc6d1 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Fri, 29 Nov 2024 21:24:33 -0800 Subject: [PATCH] Enhancement: preview button for document list and trash, refactor (#8384) --- src-ui/messages.xlf | 66 +++++++++++-------- .../admin/trash/trash.component.html | 9 ++- .../preview-popup.component.html | 61 +++++++++-------- .../preview-popup.component.spec.ts | 41 ++++++++++-- .../preview-popup/preview-popup.component.ts | 50 +++++++++++++- .../document-card-large.component.html | 11 +--- .../document-card-large.component.spec.ts | 22 +------ .../document-card-large.component.ts | 27 +------- .../document-card-small.component.html | 11 +--- .../document-card-small.component.spec.ts | 22 +------ .../document-card-small.component.ts | 34 +--------- .../document-list.component.html | 7 +- src-ui/src/styles.scss | 29 ++++++-- src/documents/views.py | 2 +- 14 files changed, 210 insertions(+), 182 deletions(-) diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 012647605..bd8b89095 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -1476,11 +1476,11 @@ src/app/components/admin/trash/trash.component.html - 67 + 72 src/app/components/admin/trash/trash.component.html - 76 + 81 src/app/components/admin/trash/trash.component.ts @@ -2199,25 +2199,25 @@ days src/app/components/admin/trash/trash.component.html - 58 + 63 Restore src/app/components/admin/trash/trash.component.html - 66 + 71 src/app/components/admin/trash/trash.component.html - 73 + 78 {VAR_PLURAL, plural, =1 {One document in trash} other { total documents in trash}} src/app/components/admin/trash/trash.component.html - 89 + 94 @@ -3019,11 +3019,11 @@ src/app/components/document-list/document-card-large/document-card-large.component.html - 68 + 63 src/app/components/document-list/document-card-small/document-card-small.component.html - 140 + 135 @@ -3312,7 +3312,7 @@ src/app/components/document-list/document-card-large/document-card-large.component.html - 62 + 60 @@ -5146,7 +5146,7 @@ src/app/components/document-list/document-card-small/document-card-small.component.ts - 86 + 79 @@ -5308,7 +5308,14 @@ Error loading preview src/app/components/common/preview-popup/preview-popup.component.html - 4 + 10 + + + + Open preview + + src/app/components/common/preview-popup/preview-popup.component.ts + 37 @@ -5922,11 +5929,11 @@ src/app/components/document-list/document-card-large/document-card-large.component.html - 79 + 74 src/app/components/document-list/document-list.component.html - 323 + 328 @@ -5937,11 +5944,11 @@ src/app/components/document-list/document-card-large/document-card-large.component.html - 85 + 80 src/app/components/document-list/document-list.component.html - 330 + 335 @@ -7130,21 +7137,21 @@ src/app/components/document-list/document-list.component.html - 299 + 304 View notes src/app/components/document-list/document-card-large/document-card-large.component.html - 74 + 69 Created: src/app/components/document-list/document-card-large/document-card-large.component.html - 98,99 + 93,94 src/app/components/document-list/document-card-small/document-card-small.component.html @@ -7159,7 +7166,7 @@ Added: src/app/components/document-list/document-card-large/document-card-large.component.html - 99,100 + 94,95 src/app/components/document-list/document-card-small/document-card-small.component.html @@ -7174,7 +7181,7 @@ Modified: src/app/components/document-list/document-card-large/document-card-large.component.html - 100,101 + 95,96 src/app/components/document-list/document-card-small/document-card-small.component.html @@ -7189,7 +7196,7 @@ {VAR_PLURAL, plural, =1 {1 page} other { pages}} src/app/components/document-list/document-card-large/document-card-large.component.html - 117 + 112 src/app/components/document-list/document-card-small/document-card-small.component.html @@ -7200,7 +7207,7 @@ Shared src/app/components/document-list/document-card-large/document-card-large.component.html - 127 + 122 src/app/components/document-list/document-card-small/document-card-small.component.html @@ -7219,7 +7226,7 @@ Score: src/app/components/document-list/document-card-large/document-card-large.component.html - 132 + 127 @@ -7473,14 +7480,21 @@ Edit document src/app/components/document-list/document-list.component.html - 295 + 296 + + + + Preview document + + src/app/components/document-list/document-list.component.html + 297 Yes src/app/components/document-list/document-list.component.html - 351 + 356 src/app/pipes/yes-no.pipe.ts @@ -7491,7 +7505,7 @@ No src/app/components/document-list/document-list.component.html - 351 + 356 src/app/pipes/yes-no.pipe.ts diff --git a/src-ui/src/app/components/admin/trash/trash.component.html b/src-ui/src/app/components/admin/trash/trash.component.html index 1c66bdd44..3dbf3db2b 100644 --- a/src-ui/src/app/components/admin/trash/trash.component.html +++ b/src-ui/src/app/components/admin/trash/trash.component.html @@ -47,14 +47,19 @@ } @for (document of documentsInTrash; track document.id) { - +
- {{ document.title }} + + {{ document.title }} + + + + {{ getDaysRemaining(document) }} days
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 index f9a8b9771..18b7cb94d 100644 --- 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 @@ -1,30 +1,37 @@ -
- @if (error) { -
-

Error loading preview

-
- } @else { - @if (renderAsObject) { - @if (previewText) { -
{{previewText}}
- } @else { - - } + + + + +
+ @if (error) { +
+

Error loading preview

+
} @else { - @if (requiresPassword) { -
- -
- } - @if (!requiresPassword) { - - + @if (renderAsObject) { + @if (previewText) { +
{{previewText}}
+ } @else { + + } + } @else { + @if (requiresPassword) { +
+ +
+ } + @if (!requiresPassword) { + + + } } } - } -
+
+ 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 index 2b9f71cef..12021fc90 100644 --- 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 @@ -1,4 +1,9 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing' +import { + ComponentFixture, + fakeAsync, + TestBed, + tick, +} from '@angular/core/testing' import { PreviewPopupComponent } from './preview-popup.component' import { By } from '@angular/platform-browser' @@ -15,6 +20,8 @@ import { withInterceptorsFromDi, } from '@angular/common/http' import { of, throwError } from 'rxjs' +import { NgbPopoverModule } from '@ng-bootstrap/ng-bootstrap' +import { DocumentTitlePipe } from 'src/app/pipes/document-title.pipe' const doc = { id: 10, @@ -34,8 +41,12 @@ describe('PreviewPopupComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ - declarations: [PreviewPopupComponent, SafeUrlPipe], - imports: [NgxBootstrapIconsModule.pick(allIcons), PdfViewerModule], + declarations: [PreviewPopupComponent, SafeUrlPipe, DocumentTitlePipe], + imports: [ + NgxBootstrapIconsModule.pick(allIcons), + PdfViewerModule, + NgbPopoverModule, + ], providers: [ provideHttpClient(withInterceptorsFromDi()), provideHttpClientTesting(), @@ -70,12 +81,14 @@ describe('PreviewPopupComponent', () => { it('should render object if native PDF viewer enabled', () => { settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, true) + component.popover.open() 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) + component.popover.open() fixture.detectChanges() expect(fixture.debugElement.query(By.css('object'))).toBeNull() expect(fixture.debugElement.query(By.css('pdf-viewer'))).not.toBeNull() @@ -83,6 +96,7 @@ describe('PreviewPopupComponent', () => { it('should show lock icon on password error', () => { settingsService.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, false) + component.popover.open() component.onError({ name: 'PasswordException' }) fixture.detectChanges() expect(component.requiresPassword).toBeTruthy() @@ -93,16 +107,18 @@ describe('PreviewPopupComponent', () => { component.document.original_file_name = 'sample.png' component.document.mime_type = 'image/png' component.document.archived_file_name = undefined + component.popover.open() fixture.detectChanges() expect(fixture.debugElement.query(By.css('object'))).not.toBeNull() }) it('should show message on error', () => { + component.popover.open() component.onError({}) fixture.detectChanges() - expect(fixture.debugElement.nativeElement.textContent).toContain( - 'Error loading preview' - ) + expect( + fixture.debugElement.query(By.css('.popover')).nativeElement.textContent + ).toContain('Error loading preview') }) it('should get text content from http if appropriate', () => { @@ -122,4 +138,17 @@ describe('PreviewPopupComponent', () => { component.init() expect(component.previewText).toEqual('Preview text') }) + + it('should show preview on mouseover after delay to preload content', fakeAsync(() => { + component.mouseEnterPreview() + expect(component.popover.isOpen()).toBeTruthy() + tick(600) + component.close() + + component.mouseEnterPreview() + tick(100) + component.mouseLeavePreview() + tick(600) + expect(component.popover.isOpen()).toBeFalsy() + })) }) 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 index 6d2ede266..75f3cbb86 100644 --- 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 @@ -1,5 +1,6 @@ import { HttpClient } from '@angular/common/http' -import { Component, Input, OnDestroy } from '@angular/core' +import { Component, Input, OnDestroy, ViewChild } from '@angular/core' +import { NgbPopover } from '@ng-bootstrap/ng-bootstrap' import { first, Subject, takeUntil } from 'rxjs' import { Document } from 'src/app/data/document' import { SETTINGS_KEYS } from 'src/app/data/ui-settings' @@ -23,6 +24,18 @@ export class PreviewPopupComponent implements OnDestroy { return this._document } + @Input() + link: string + + @Input() + linkClasses: string = 'btn btn-sm btn-outline-secondary' + + @Input() + linkTarget: string = '_blank' + + @Input() + linkTitle: string = $localize`Open preview` + unsubscribeNotifier: Subject = new Subject() error = false @@ -31,6 +44,12 @@ export class PreviewPopupComponent implements OnDestroy { previewText: string + @ViewChild('popover') popover: NgbPopover + + mouseOnPreview: boolean + + popoverClass: string = 'shadow popover-preview' + get renderAsObject(): boolean { return (this.isPdf && this.useNativePdfViewer) || !this.isPdf } @@ -83,4 +102,33 @@ export class PreviewPopupComponent implements OnDestroy { this.error = true } } + + get previewUrl() { + return this.documentService.getPreviewUrl(this.document.id) + } + + mouseEnterPreview() { + this.mouseOnPreview = true + if (!this.popover.isOpen()) { + // we're going to open but hide to pre-load content during hover delay + this.popover.open() + this.popoverClass = 'shadow popover-preview pe-none opacity-0' + setTimeout(() => { + if (this.mouseOnPreview) { + // show popover + this.popoverClass = this.popoverClass.replace('pe-none opacity-0', '') + } else { + this.popover.close() + } + }, 600) + } + } + + mouseLeavePreview() { + this.mouseOnPreview = false + } + + public close() { + this.popover.close(false) + } } 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 04f3a236a..34557be31 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 @@ -1,4 +1,4 @@ -
+
@@ -56,14 +56,9 @@  Open - +  View - - - - +  Download 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 efd5076be..95b12d7ec 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 @@ -1,11 +1,6 @@ import { DatePipe } from '@angular/common' import { provideHttpClientTesting } from '@angular/common/http/testing' -import { - ComponentFixture, - TestBed, - fakeAsync, - tick, -} from '@angular/core/testing' +import { ComponentFixture, TestBed } from '@angular/core/testing' import { By } from '@angular/platform-browser' import { RouterTestingModule } from '@angular/router/testing' import { @@ -84,21 +79,6 @@ describe('DocumentCardLargeComponent', () => { expect(fixture.nativeElement.textContent).toContain('8 pages') }) - it('should show preview on mouseover after delay to preload content', fakeAsync(() => { - component.mouseEnterPreview() - expect(component.popover.isOpen()).toBeTruthy() - expect(component.popoverHidden).toBeTruthy() - tick(600) - expect(component.popoverHidden).toBeFalsy() - component.mouseLeaveCard() - - component.mouseEnterPreview() - tick(100) - component.mouseLeavePreview() - tick(600) - expect(component.popover.isOpen()).toBeFalsy() - })) - it('should trim content', () => { expect(component.contentTrimmed).toHaveLength(503) // includes ... }) 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 a3d57d950..99597ca5a 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 @@ -12,9 +12,9 @@ import { } from 'src/app/data/document' import { DocumentService } from 'src/app/services/rest/document.service' import { SettingsService } from 'src/app/services/settings.service' -import { NgbPopover } from '@ng-bootstrap/ng-bootstrap' import { SETTINGS_KEYS } from 'src/app/data/ui-settings' import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component' +import { PreviewPopupComponent } from '../../common/preview-popup/preview-popup.component' @Component({ selector: 'pngx-document-card-large', @@ -65,7 +65,7 @@ export class DocumentCardLargeComponent extends ComponentWithPermissions { @Output() clickMoreLike = new EventEmitter() - @ViewChild('popover') popover: NgbPopover + @ViewChild('popupPreview') popupPreview: PreviewPopupComponent mouseOnPreview = false popoverHidden = true @@ -112,29 +112,8 @@ export class DocumentCardLargeComponent extends ComponentWithPermissions { return this.documentService.getPreviewUrl(this.document.id) } - mouseEnterPreview() { - this.mouseOnPreview = true - if (!this.popover.isOpen()) { - // we're going to open but hide to pre-load content during hover delay - this.popover.open() - this.popoverHidden = true - setTimeout(() => { - if (this.mouseOnPreview) { - // show popover - this.popoverHidden = false - } else { - this.popover.close() - } - }, 600) - } - } - - mouseLeavePreview() { - this.mouseOnPreview = false - } - mouseLeaveCard() { - this.popover.close() + this.popupPreview.close() } get contentTrimmed() { 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 57bd6048b..60713ef02 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 @@ -1,5 +1,5 @@
-
+
@@ -129,14 +129,9 @@ - + - - - - + 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 b86453a25..0c0c82103 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 @@ -1,11 +1,6 @@ import { DatePipe } from '@angular/common' import { provideHttpClientTesting } from '@angular/common/http/testing' -import { - ComponentFixture, - TestBed, - fakeAsync, - tick, -} from '@angular/core/testing' +import { ComponentFixture, TestBed } from '@angular/core/testing' import { RouterTestingModule } from '@angular/router/testing' import { NgbPopoverModule, @@ -116,19 +111,4 @@ describe('DocumentCardSmallComponent', () => { fixture.debugElement.queryAll(By.directive(TagComponent)) ).toHaveLength(6) }) - - it('should show preview on mouseover after delay to preload content', fakeAsync(() => { - component.mouseEnterPreview() - expect(component.popover.isOpen()).toBeTruthy() - expect(component.popoverHidden).toBeTruthy() - tick(600) - expect(component.popoverHidden).toBeFalsy() - component.mouseLeaveCard() - - component.mouseEnterPreview() - tick(100) - component.mouseLeavePreview() - tick(600) - expect(component.popover.isOpen()).toBeFalsy() - })) }) 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 5cd583fb0..7397159af 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 @@ -13,9 +13,9 @@ import { } from 'src/app/data/document' import { DocumentService } from 'src/app/services/rest/document.service' import { SettingsService } from 'src/app/services/settings.service' -import { NgbPopover } from '@ng-bootstrap/ng-bootstrap' import { SETTINGS_KEYS } from 'src/app/data/ui-settings' import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component' +import { PreviewPopupComponent } from '../../common/preview-popup/preview-popup.component' @Component({ selector: 'pngx-document-card-small', @@ -61,10 +61,7 @@ export class DocumentCardSmallComponent extends ComponentWithPermissions { moreTags: number = null - @ViewChild('popover') popover: NgbPopover - - mouseOnPreview = false - popoverHidden = true + @ViewChild('popupPreview') popupPreview: PreviewPopupComponent getIsThumbInverted() { return this.settingsService.get(SETTINGS_KEYS.DARK_MODE_THUMB_INVERTED) @@ -78,10 +75,6 @@ export class DocumentCardSmallComponent extends ComponentWithPermissions { return this.documentService.getDownloadUrl(this.document.id) } - get previewUrl() { - return this.documentService.getPreviewUrl(this.document.id) - } - get privateName() { return $localize`Private` } @@ -100,29 +93,8 @@ export class DocumentCardSmallComponent extends ComponentWithPermissions { ) } - mouseEnterPreview() { - this.mouseOnPreview = true - if (!this.popover.isOpen()) { - // we're going to open but hide to pre-load content during hover delay - this.popover.open() - this.popoverHidden = true - setTimeout(() => { - if (this.mouseOnPreview) { - // show popover - this.popoverHidden = false - } else { - this.popover.close() - } - }, 600) - } - } - - mouseLeavePreview() { - this.mouseOnPreview = false - } - mouseLeaveCard() { - this.popover.close() + this.popupPreview.close() } get notesEnabled(): boolean { diff --git a/src-ui/src/app/components/document-list/document-list.component.html b/src-ui/src/app/components/document-list/document-list.component.html index 4eb9d179e..ebe3536e5 100644 --- a/src-ui/src/app/components/document-list/document-list.component.html +++ b/src-ui/src/app/components/document-list/document-list.component.html @@ -292,7 +292,12 @@ @if (activeDisplayFields.includes(DisplayField.TITLE) || activeDisplayFields.includes(DisplayField.TAGS)) { @if (activeDisplayFields.includes(DisplayField.TITLE)) { - {{d.title | documentTitle}} + } @if (activeDisplayFields.includes(DisplayField.TAGS)) { @for (t of d.tags$ | async; track t) { diff --git a/src-ui/src/styles.scss b/src-ui/src/styles.scss index 331f6e6d8..fe1466d58 100644 --- a/src-ui/src/styles.scss +++ b/src-ui/src/styles.scss @@ -564,11 +564,6 @@ table.table { } } -.popover-hidden .popover { - opacity: 0; - pointer-events: none; -} - // Tour .tour-active .popover { min-width: 360px; @@ -728,3 +723,27 @@ i-bs svg { vertical-align: middle; } } + +// fixes for buttons in preview popup +.btn-group pngx-preview-popup:not(:last-child) { + // Prevent double borders when buttons are next to each other + > .btn { + margin-left: calc(#{$btn-border-width} * -1); + } + > .btn { + @include border-end-radius(0); + } +} +.btn-group pngx-preview-popup:not(:first-child) { + > .btn { + @include border-start-radius(0); + } +} +.btn-group pngx-preview-popup { + position: relative; + flex: 1 1 auto; + + > .btn { + display: block; + } +} diff --git a/src/documents/views.py b/src/documents/views.py index 35fa8eafc..367559c6d 100644 --- a/src/documents/views.py +++ b/src/documents/views.py @@ -426,7 +426,7 @@ class DocumentViewSet( ) def file_response(self, pk, request, disposition): - doc = Document.objects.select_related("owner").get(id=pk) + doc = Document.global_objects.select_related("owner").get(id=pk) if request.user is not None and not has_perms_owner_aware( request.user, "view_document",