1
0
mirror of https://github.com/xuthus83/pigallery2.git synced 2025-01-14 14:43:17 +08:00
pigallery2/frontend/app/gallery/lightbox/lightbox.gallery.component.ts

729 lines
20 KiB
TypeScript
Raw Normal View History

2018-05-03 18:23:48 -04:00
import {
ChangeDetectorRef,
Component,
ElementRef,
HostListener,
OnDestroy,
OnInit,
QueryList,
ViewChild
} from '@angular/core';
2018-03-30 15:30:30 -04:00
import {GalleryPhotoComponent} from '../grid/photo/photo.grid.gallery.component';
import {Dimension} from '../../model/IRenderable';
import {FullScreenService} from '../fullscreen.service';
import {OverlayService} from '../overlay.service';
import {animate, AnimationBuilder, AnimationPlayer, style} from '@angular/animations';
import {GalleryLightboxMediaComponent} from './media/media.lightbox.gallery.component';
2018-05-22 20:27:07 -04:00
import {Observable, Subscription, timer} from 'rxjs';
2018-05-03 18:23:48 -04:00
import {filter} from 'rxjs/operators';
import {ActivatedRoute, Params, Router} from '@angular/router';
import {PageHelper} from '../../model/page.helper';
import {QueryService} from '../../model/query.service';
2018-11-04 19:28:32 +01:00
import {MediaDTO} from '../../../../common/entities/MediaDTO';
2018-11-30 15:36:42 +01:00
import {QueryParams} from '../../../../common/QueryParams';
import {GalleryService} from '../gallery.service';
2018-12-05 19:15:24 +01:00
import {PhotoDTO} from '../../../../common/entities/PhotoDTO';
export enum LightboxStates {
2018-05-28 14:03:12 -04:00
Open = 1,
Closing = 2,
Closed = 3
}
2018-12-18 20:12:51 +01:00
export enum PlayBackStates {
Paused = 1,
Play = 2,
FastForward = 3
}
@Component({
2018-05-03 18:23:48 -04:00
selector: 'app-gallery-lightbox',
2018-11-11 19:22:46 +01:00
styleUrls: ['./lightbox.gallery.component.css', './inputrange.css'],
2017-07-10 22:00:22 +02:00
templateUrl: './lightbox.gallery.component.html'
})
2018-05-03 18:23:48 -04:00
export class GalleryLightboxComponent implements OnDestroy, OnInit {
2017-07-17 18:30:16 +02:00
2018-12-19 21:41:01 +01:00
readonly MAX_ZOOM = 10;
2018-12-18 20:12:51 +01:00
@ViewChild('photo') mediaElement: GalleryLightboxMediaComponent;
2018-03-30 15:30:30 -04:00
@ViewChild('lightbox') lightboxElement: ElementRef;
@ViewChild('root') root: ElementRef;
public navigation = {hasPrev: true, hasNext: true};
2018-11-28 23:49:33 +01:00
public blackCanvasOpacity = 0;
2017-07-19 22:40:27 +02:00
private activePhotoId: number = null;
public activePhoto: GalleryPhotoComponent;
private gridPhotoQL: QueryList<GalleryPhotoComponent>;
public status: LightboxStates = LightboxStates.Closed;
private subscription: {
photosChange: Subscription,
route: Subscription
} = {
photosChange: null,
route: null
};
private timer: Observable<number>;
private timerSub: Subscription;
2018-12-18 20:12:51 +01:00
public playBackState: PlayBackStates = PlayBackStates.Paused;
public PlayBackStates = PlayBackStates;
public controllersDimmed = false;
public controllersAlwaysOn = false;
public controllersVisible = true;
public infoPanelVisible = false;
public infoPanelWidth = 0;
2017-07-16 20:49:12 +02:00
public animating = false;
2018-05-03 18:23:48 -04:00
startPhotoDimension: Dimension = <Dimension>{top: 0, left: 0, width: 0, height: 0};
2018-11-28 23:49:33 +01:00
iPvisibilityTimer: number = null;
visibilityTimer: number = null;
delayedMediaShow: string = null;
2018-12-18 00:06:25 +01:00
public zoom = 1;
public drag = {x: 0, y: 0};
private prevDrag = {x: 0, y: 0};
private prevZoom = 1;
2016-07-06 12:53:49 +02:00
2017-07-10 22:00:22 +02:00
constructor(public fullScreenService: FullScreenService,
private changeDetector: ChangeDetectorRef,
private overlayService: OverlayService,
private _builder: AnimationBuilder,
private router: Router,
private queryService: QueryService,
private galleryService: GalleryService,
private route: ActivatedRoute) {
}
2016-04-30 18:33:07 +02:00
ngOnInit(): void {
2018-05-03 18:23:48 -04:00
this.timer = timer(1000, 2000);
this.subscription.route = this.route.queryParams.subscribe((params: Params) => {
2018-11-30 15:36:42 +01:00
if (params[QueryParams.gallery.photo] && params[QueryParams.gallery.photo] !== '') {
if (!this.gridPhotoQL) {
return this.delayedMediaShow = params[QueryParams.gallery.photo];
}
2018-11-30 15:36:42 +01:00
this.onNavigateTo(params[QueryParams.gallery.photo]);
} else if (this.status === LightboxStates.Open) {
this.delayedMediaShow = null;
2018-12-23 15:04:52 +01:00
this.hideLightbox();
}
});
}
2016-04-30 21:36:24 +02:00
2018-12-05 19:15:24 +01:00
ngOnDestroy(): void {
this.pause();
if (this.subscription.photosChange != null) {
this.subscription.photosChange.unsubscribe();
}
if (this.subscription.route != null) {
this.subscription.route.unsubscribe();
}
2017-07-19 22:40:27 +02:00
if (this.visibilityTimer != null) {
clearTimeout(this.visibilityTimer);
}
if (this.iPvisibilityTimer != null) {
clearTimeout(this.iPvisibilityTimer);
}
}
onNavigateTo(photoStringId: string) {
if (this.activePhoto && this.queryService.getMediaStringId(this.activePhoto.gridPhoto.media) === photoStringId) {
return;
}
2018-12-18 00:06:25 +01:00
2018-12-19 22:32:59 +01:00
this.Zoom = 1;
const photo = this.gridPhotoQL.find(i => this.queryService.getMediaStringId(i.gridPhoto.media) === photoStringId);
if (!photo) {
return this.delayedMediaShow = photoStringId;
}
if (this.status === LightboxStates.Closed) {
2018-11-04 19:28:32 +01:00
this.showLigthbox(photo.gridPhoto.media);
} else {
this.showPhoto(this.gridPhotoQL.toArray().indexOf(photo));
}
this.delayedMediaShow = null;
}
setGridPhotoQL(value: QueryList<GalleryPhotoComponent>) {
if (this.subscription.photosChange != null) {
this.subscription.photosChange.unsubscribe();
}
this.gridPhotoQL = value;
this.subscription.photosChange = this.gridPhotoQL.changes.subscribe(() => {
if (this.activePhotoId != null && this.gridPhotoQL.length > this.activePhotoId) {
this.updateActivePhoto(this.activePhotoId);
}
if (this.delayedMediaShow) {
this.onNavigateTo(this.delayedMediaShow);
}
});
if (this.delayedMediaShow) {
this.onNavigateTo(this.delayedMediaShow);
}
}
2018-12-18 20:12:51 +01:00
pan($event: any) {
if (!this.activePhoto || this.activePhoto.gridPhoto.isVideo()) {
return;
}
2018-12-18 00:06:25 +01:00
if (this.zoom === 1) {
return;
}
this.drag.x = this.prevDrag.x + $event.deltaX;
this.drag.y = this.prevDrag.y + $event.deltaY;
this.checkZoomAndDrag();
if ($event.isFinal) {
this.prevDrag = {
x: this.drag.x,
y: this.drag.y,
};
}
}
2018-12-18 20:12:51 +01:00
wheel($event: any) {
if (!this.activePhoto || this.activePhoto.gridPhoto.isVideo()) {
return;
}
2018-12-19 21:41:01 +01:00
if ($event.deltaY < 0) {
this.zoomIn();
} else {
this.zoomOut();
}
2018-12-18 00:06:25 +01:00
}
@HostListener('pinch', ['$event'])
pinch($event: any) {
2018-12-18 20:12:51 +01:00
if (!this.activePhoto || this.activePhoto.gridPhoto.isVideo()) {
return;
}
this.showControls();
2018-12-19 22:32:59 +01:00
this.Zoom = this.prevZoom * $event.scale;
2018-12-18 00:06:25 +01:00
}
@HostListener('pinchend', ['$event'])
pinchend($event: any) {
2018-12-18 20:12:51 +01:00
if (!this.activePhoto || this.activePhoto.gridPhoto.isVideo()) {
return;
}
this.showControls();
2018-12-19 22:32:59 +01:00
this.Zoom = this.prevZoom * $event.scale;
2018-12-18 00:06:25 +01:00
this.prevZoom = this.zoom;
}
2018-12-18 20:12:51 +01:00
tap($event: any) {
if (!this.activePhoto || this.activePhoto.gridPhoto.isVideo()) {
return;
}
2018-12-18 00:06:25 +01:00
if ($event.tapCount < 2) {
return;
}
2018-12-18 20:12:51 +01:00
this.showControls();
2018-12-18 00:06:25 +01:00
if (this.zoom > 1) {
2018-12-19 22:32:59 +01:00
this.Zoom = 1;
2018-12-18 00:06:25 +01:00
this.prevZoom = this.zoom;
return;
} else {
2018-12-19 22:32:59 +01:00
this.Zoom = 5;
2018-12-18 00:06:25 +01:00
this.prevZoom = this.zoom;
return;
}
}
2018-12-19 21:41:01 +01:00
zoomIn() {
this.showControls();
2018-12-19 22:32:59 +01:00
this.Zoom = this.zoom + this.zoom / 10;
2018-12-19 21:41:01 +01:00
}
zoomOut() {
this.showControls();
2018-12-19 22:32:59 +01:00
this.Zoom = this.zoom - this.zoom / 10;
2018-12-19 21:41:01 +01:00
}
2018-12-18 00:06:25 +01:00
2018-12-19 22:32:59 +01:00
public get Zoom(): number {
return this.zoom;
}
public set Zoom(zoom: number) {
2018-12-18 00:06:25 +01:00
if (!this.activePhoto || this.activePhoto.gridPhoto.isVideo()) {
return;
}
if (zoom < 1) {
zoom = 1;
}
2018-12-18 20:12:51 +01:00
if (zoom > this.MAX_ZOOM) {
zoom = this.MAX_ZOOM;
2018-12-18 00:06:25 +01:00
}
2018-12-18 20:12:51 +01:00
if (this.zoom === zoom) {
return;
}
this.pause();
2018-12-18 00:06:25 +01:00
this.drag.x = this.drag.x / this.zoom * zoom;
this.drag.y = this.drag.y / this.zoom * zoom;
this.prevDrag.x = this.drag.x;
this.prevDrag.y = this.drag.y;
this.zoom = zoom;
this.checkZoomAndDrag();
}
private checkZoomAndDrag() {
const fixDrag = (drag: { x: number, y: number }) => {
if (this.zoom === 1) {
drag.y = 0;
drag.x = 0;
return;
}
if (!this.activePhoto) {
return;
}
const photoAspect = MediaDTO.calcRotatedAspectRatio(this.activePhoto.gridPhoto.media);
const widthFilled = photoAspect > this.getWindowAspectRatio();
const divWidth = this.getPhotoFrameWidth();
const divHeight = this.getPhotoFrameHeight();
const size = {
width: (widthFilled ? divWidth : divHeight * photoAspect) * this.zoom,
height: (widthFilled ? divWidth / photoAspect : divHeight) * this.zoom
};
const widthDrag = Math.abs(divWidth - size.width) / 2;
const heightDrag = Math.abs(divHeight - size.height) / 2;
if (divWidth > size.width) {
drag.x = 0;
}
if (divHeight > size.height) {
drag.y = 0;
}
if (drag.x < -widthDrag) {
drag.x = -widthDrag;
}
if (drag.x > widthDrag) {
drag.x = widthDrag;
}
if (drag.y < -heightDrag) {
drag.y = -heightDrag;
}
if (drag.y > heightDrag) {
drag.y = heightDrag;
}
};
if (this.zoom < 1) {
this.zoom = 1;
}
2018-12-18 20:12:51 +01:00
if (this.zoom > this.MAX_ZOOM) {
2018-12-19 21:41:01 +01:00
this.zoom = this.MAX_ZOOM;
2018-12-18 00:06:25 +01:00
}
fixDrag(this.drag);
fixDrag(this.prevDrag);
}
//noinspection JSUnusedGlobalSymbols
@HostListener('window:resize', ['$event'])
onResize() {
if (this.activePhoto) {
2017-07-10 22:00:22 +02:00
this.animateLightbox();
this.updateActivePhoto(this.activePhotoId);
}
}
public nextImage() {
if (this.activePhotoId + 1 < this.gridPhotoQL.length) {
this.navigateToPhoto(this.activePhotoId + 1);
2016-05-12 23:00:38 +02:00
}
}
2016-05-12 23:00:38 +02:00
public prevImage() {
this.pause();
if (this.activePhotoId > 0) {
this.navigateToPhoto(this.activePhotoId - 1);
2016-05-12 23:00:38 +02:00
}
}
2016-05-12 23:00:38 +02:00
private navigateToPhoto(photoIndex: number) {
this.router.navigate([],
{queryParams: this.queryService.getParams(this.gridPhotoQL.toArray()[photoIndex].gridPhoto.media)}).catch(console.error);
}
2017-07-10 22:00:22 +02:00
private showPhoto(photoIndex: number, resize: boolean = true) {
this.activePhoto = null;
this.changeDetector.detectChanges();
2017-07-10 22:00:22 +02:00
this.updateActivePhoto(photoIndex, resize);
}
2018-11-04 19:28:32 +01:00
public showLigthbox(photo: MediaDTO) {
2018-12-19 22:32:59 +01:00
this.Zoom = 1;
this.controllersVisible = true;
this.showControls();
this.status = LightboxStates.Open;
2018-05-03 18:23:48 -04:00
const selectedPhoto = this.findPhotoComponent(photo);
if (selectedPhoto === null) {
2018-03-30 15:30:30 -04:00
throw new Error('Can\'t find Photo');
}
2017-07-10 22:00:22 +02:00
const lightboxDimension = selectedPhoto.getDimension();
lightboxDimension.top -= PageHelper.ScrollY;
2017-07-15 23:54:17 +02:00
this.animating = true;
2018-11-04 19:28:32 +01:00
this.animatePhoto(selectedPhoto.getDimension(), this.calcLightBoxPhotoDimension(selectedPhoto.gridPhoto.media)).onDone(() => {
2017-07-15 23:54:17 +02:00
this.animating = false;
});
2017-07-10 22:00:22 +02:00
this.animateLightbox(
lightboxDimension,
<Dimension>{
top: 0,
left: 0,
2017-07-16 20:49:12 +02:00
width: this.getPhotoFrameWidth(),
height: this.getPhotoFrameHeight()
2017-07-10 22:00:22 +02:00
});
this.blackCanvasOpacity = 0;
this.startPhotoDimension = selectedPhoto.getDimension();
2018-05-03 18:23:48 -04:00
// disable scroll
2017-07-10 22:00:22 +02:00
this.overlayService.showOverlay();
this.blackCanvasOpacity = 1.0;
this.showPhoto(this.gridPhotoQL.toArray().indexOf(selectedPhoto), false);
}
2018-05-03 18:23:48 -04:00
@HostListener('window:keydown', ['$event'])
onKeyPress(e: KeyboardEvent) {
if (this.status !== LightboxStates.Open) {
2018-05-03 18:23:48 -04:00
return;
2018-03-30 15:30:30 -04:00
}
2018-05-03 18:23:48 -04:00
const event: KeyboardEvent = window.event ? <any>window.event : e;
switch (event.key) {
case 'ArrowLeft':
2018-05-03 18:23:48 -04:00
if (this.activePhotoId > 0) {
this.prevImage();
}
break;
case 'ArrowRight':
if (this.activePhotoId < this.gridPhotoQL.length - 1) {
this.nextImage();
}
break;
case 'i':
case 'I':
if (this.isInfoPanelAnimating()) {
return;
}
if (this.infoPanelVisible) {
this.hideInfoPanel(true);
} else {
this.showInfoPanel();
}
break;
case 'f':
case 'F':
if (this.fullScreenService.isFullScreenEnabled()) {
this.fullScreenService.exitFullScreen();
} else {
this.fullScreenService.showFullScreen(this.root.nativeElement);
}
break;
2018-12-19 22:40:37 +01:00
case '-':
this.zoomOut();
break;
case '+':
this.zoomIn();
break;
case 'c':
case 'C':
this.controllersAlwaysOn = !this.controllersAlwaysOn;
break;
case 'Escape': // escape
2018-05-03 18:23:48 -04:00
this.hide();
break;
2018-11-11 19:22:46 +01:00
case ' ': // space
if (this.activePhoto && this.activePhoto.gridPhoto.isVideo()) {
this.mediaElement.playPause();
2018-11-11 19:22:46 +01:00
}
break;
2018-03-30 15:30:30 -04:00
}
}
public hide() {
this.router.navigate([],
{queryParams: this.queryService.getParams()}).catch(console.error);
}
2018-12-23 15:04:52 +01:00
private hideLightbox() {
2018-12-19 22:32:59 +01:00
this.Zoom = 1;
this.controllersVisible = false;
this.status = LightboxStates.Closing;
this.fullScreenService.exitFullScreen();
2017-07-12 18:31:19 +02:00
this.pause();
2017-07-15 23:54:17 +02:00
this.animating = true;
2017-07-10 22:00:22 +02:00
const lightboxDimension = this.activePhoto.getDimension();
lightboxDimension.top -= PageHelper.ScrollY;
this.blackCanvasOpacity = 0;
2017-07-10 22:00:22 +02:00
2018-11-04 19:28:32 +01:00
this.animatePhoto(this.calcLightBoxPhotoDimension(this.activePhoto.gridPhoto.media), this.activePhoto.getDimension());
2017-07-10 22:00:22 +02:00
this.animateLightbox(<Dimension>{
top: 0,
left: 0,
2017-07-16 20:49:12 +02:00
width: this.getPhotoFrameWidth(),
height: this.getPhotoFrameHeight()
2017-07-15 23:54:17 +02:00
}, lightboxDimension).onDone(() => {
this.status = LightboxStates.Closed;
this.activePhoto = null;
2017-07-19 22:40:27 +02:00
this.activePhotoId = null;
this.overlayService.hideOverlay();
2017-07-15 23:54:17 +02:00
});
2017-07-10 22:41:30 +02:00
this.hideInfoPanel(false);
}
2017-07-15 23:54:17 +02:00
animatePhoto(from: Dimension, to: Dimension = from): AnimationPlayer {
const elem = this._builder.build([
2018-12-23 15:04:52 +01:00
style(Dimension.toString(from)),
2017-07-10 22:00:22 +02:00
animate(300,
2018-12-23 15:04:52 +01:00
style(Dimension.toString(to)))
2017-07-10 22:00:22 +02:00
])
.create(this.mediaElement.elementRef.nativeElement);
2017-07-15 23:54:17 +02:00
elem.play();
return elem;
2017-07-10 22:00:22 +02:00
}
animateLightbox(from: Dimension = <Dimension>{
top: 0,
left: 0,
2017-07-16 20:49:12 +02:00
width: this.getPhotoFrameWidth(),
height: this.getPhotoFrameHeight()
2017-07-15 23:54:17 +02:00
}, to: Dimension = from): AnimationPlayer {
const elem = this._builder.build([
2018-12-23 15:04:52 +01:00
style(Dimension.toString(from)),
2017-07-10 22:00:22 +02:00
animate(300,
2018-12-23 15:04:52 +01:00
style(Dimension.toString(to)))
2017-07-10 22:00:22 +02:00
])
2017-07-15 23:54:17 +02:00
.create(this.lightboxElement.nativeElement);
elem.play();
return elem;
2017-07-10 22:00:22 +02:00
}
public toggleInfoPanel() {
2018-05-03 18:23:48 -04:00
if (this.infoPanelWidth !== 400) {
this.showInfoPanel();
} else {
this.hideInfoPanel();
}
}
2018-05-03 18:23:48 -04:00
hideInfoPanel(_animate: boolean = true) {
2018-11-28 23:49:33 +01:00
this.iPvisibilityTimer = window.setTimeout(() => {
2018-05-03 18:23:48 -04:00
this.iPvisibilityTimer = null;
this.infoPanelVisible = false;
2018-12-18 00:06:25 +01:00
this.checkZoomAndDrag();
2018-05-03 18:23:48 -04:00
}, 1000);
2018-11-04 19:28:32 +01:00
const starPhotoPos = this.calcLightBoxPhotoDimension(this.activePhoto.gridPhoto.media);
2018-05-03 18:23:48 -04:00
this.infoPanelWidth = 0;
2018-11-04 19:28:32 +01:00
const endPhotoPos = this.calcLightBoxPhotoDimension(this.activePhoto.gridPhoto.media);
2018-05-03 18:23:48 -04:00
if (_animate) {
this.animatePhoto(starPhotoPos, endPhotoPos);
}
if (_animate) {
this.animateLightbox(<Dimension>{
top: 0,
left: 0,
2018-05-13 22:10:56 -04:00
width: Math.max(this.getPhotoFrameWidth() - 400, 0),
2018-05-03 18:23:48 -04:00
height: this.getPhotoFrameHeight()
},
<Dimension>{
top: 0,
left: 0,
width: this.getPhotoFrameWidth(),
height: this.getPhotoFrameHeight()
});
}
}
public play() {
this.pause();
this.timerSub = this.timer.pipe(filter(t => t % 2 === 0)).subscribe(() => {
if (this.mediaElement.imageLoadFinished === false) {
2018-05-03 18:23:48 -04:00
return;
}
2018-11-11 19:22:46 +01:00
// do not skip video if its playing
if (this.activePhoto && this.activePhoto.gridPhoto.isVideo() &&
!this.mediaElement.Paused) {
2018-11-11 19:22:46 +01:00
return;
}
2018-05-03 18:23:48 -04:00
if (this.navigation.hasNext) {
this.nextImage();
} else {
this.navigateToPhoto(0);
2018-05-03 18:23:48 -04:00
}
});
2018-12-18 20:12:51 +01:00
this.playBackState = PlayBackStates.Play;
2018-05-03 18:23:48 -04:00
}
isInfoPanelAnimating(): boolean {
return this.iPvisibilityTimer != null;
}
showInfoPanel() {
this.infoPanelVisible = true;
2017-07-10 22:00:22 +02:00
2018-11-04 19:28:32 +01:00
const starPhotoPos = this.calcLightBoxPhotoDimension(this.activePhoto.gridPhoto.media);
2017-07-10 22:00:22 +02:00
this.infoPanelWidth = 400;
2018-11-04 19:28:32 +01:00
const endPhotoPos = this.calcLightBoxPhotoDimension(this.activePhoto.gridPhoto.media);
2017-07-10 22:00:22 +02:00
this.animatePhoto(starPhotoPos, endPhotoPos);
this.animateLightbox(<Dimension>{
top: 0,
left: 0,
2017-07-16 20:49:12 +02:00
width: this.getPhotoFrameWidth() + 400,
height: this.getPhotoFrameHeight()
2017-07-10 22:00:22 +02:00
},
<Dimension>{
top: 0,
left: 0,
2017-07-16 20:49:12 +02:00
width: this.getPhotoFrameWidth(),
height: this.getPhotoFrameHeight()
2017-07-10 22:00:22 +02:00
});
if (this.iPvisibilityTimer != null) {
clearTimeout(this.iPvisibilityTimer);
}
2018-12-18 00:06:25 +01:00
this.checkZoomAndDrag();
}
2018-05-03 18:23:48 -04:00
public fastForward() {
this.pause();
this.timerSub = this.timer.subscribe(() => {
if (this.mediaElement.imageLoadFinished === false) {
2018-05-03 18:23:48 -04:00
return;
2018-11-11 19:22:46 +01:00
}
if (this.activePhoto && this.activePhoto.gridPhoto.isVideo() &&
!this.mediaElement.Paused) {
2018-11-11 19:22:46 +01:00
return;
2018-05-03 18:23:48 -04:00
}
if (this.navigation.hasNext) {
this.nextImage();
} else {
this.navigateToPhoto(0);
2018-05-03 18:23:48 -04:00
}
});
2018-12-18 20:12:51 +01:00
this.playBackState = PlayBackStates.FastForward;
}
2016-04-30 21:36:24 +02:00
2018-05-13 22:10:56 -04:00
public getPhotoFrameWidth(): number {
return Math.max(window.innerWidth - this.infoPanelWidth, 0);
}
2016-07-06 12:53:49 +02:00
2018-05-13 22:10:56 -04:00
public getPhotoFrameHeight(): number {
return window.innerHeight;
}
2016-12-30 11:58:04 +01:00
2018-05-13 22:10:56 -04:00
public getWindowAspectRatio(): number {
return Math.round(this.getPhotoFrameWidth() / this.getPhotoFrameHeight() * 100) / 100;
}
2018-05-03 18:23:48 -04:00
private updateActivePhoto(photoIndex: number, resize: boolean = true) {
const pcList = this.gridPhotoQL.toArray();
2016-12-30 11:58:04 +01:00
2018-05-03 18:23:48 -04:00
if (photoIndex < 0 || photoIndex > this.gridPhotoQL.length) {
2018-11-04 19:28:32 +01:00
throw new Error('Can\'t find the media');
2018-05-03 18:23:48 -04:00
}
this.activePhotoId = photoIndex;
this.activePhoto = pcList[photoIndex];
if (resize) {
2018-11-04 19:28:32 +01:00
this.animatePhoto(this.calcLightBoxPhotoDimension(this.activePhoto.gridPhoto.media));
2018-05-03 18:23:48 -04:00
}
this.navigation.hasPrev = photoIndex > 0;
this.navigation.hasNext = photoIndex + 1 < pcList.length;
const to = this.activePhoto.getDimension();
// if target image out of screen -> scroll to there
if (PageHelper.ScrollY > to.top || PageHelper.ScrollY + this.getPhotoFrameHeight() < to.top) {
PageHelper.ScrollY = to.top;
2016-04-30 21:36:24 +02:00
}
}
@HostListener('mousemove')
onMouseMove() {
this.showControls();
}
private showControls() {
this.controllersDimmed = false;
if (this.visibilityTimer != null) {
clearTimeout(this.visibilityTimer);
}
2018-11-28 23:49:33 +01:00
this.visibilityTimer = window.setTimeout(this.hideControls, 2000);
}
private hideControls = () => {
this.controllersDimmed = true;
};
2017-07-16 20:49:12 +02:00
public pause() {
if (this.timerSub != null) {
this.timerSub.unsubscribe();
}
2018-12-18 20:12:51 +01:00
this.playBackState = PlayBackStates.Paused;
}
2018-11-04 19:28:32 +01:00
private findPhotoComponent(media: MediaDTO): GalleryPhotoComponent {
2018-05-03 18:23:48 -04:00
const galleryPhotoComponents = this.gridPhotoQL.toArray();
for (let i = 0; i < galleryPhotoComponents.length; i++) {
2018-11-04 19:28:32 +01:00
if (galleryPhotoComponents[i].gridPhoto.media === media) {
2018-05-03 18:23:48 -04:00
return galleryPhotoComponents[i];
}
2018-05-03 18:23:48 -04:00
}
return null;
}
2018-11-04 19:28:32 +01:00
private calcLightBoxPhotoDimension(photo: MediaDTO): Dimension {
2018-05-03 18:23:48 -04:00
let width = 0;
let height = 0;
const photoAspect = photo.metadata.size.width / photo.metadata.size.height;
const windowAspect = this.getPhotoFrameWidth() / this.getPhotoFrameHeight();
if (photoAspect < windowAspect) {
width = Math.round(photo.metadata.size.width * (this.getPhotoFrameHeight() / photo.metadata.size.height));
height = this.getPhotoFrameHeight();
} else {
width = this.getPhotoFrameWidth();
height = Math.round(photo.metadata.size.height * (this.getPhotoFrameWidth() / photo.metadata.size.width));
}
const top = (this.getPhotoFrameHeight() / 2 - height / 2);
const left = (this.getPhotoFrameWidth() / 2 - width / 2);
return <Dimension>{top: top, left: left, width: width, height: height};
}
public isVisible(): boolean {
return this.status !== LightboxStates.Closed;
}
2018-12-05 19:15:24 +01:00
get Title(): string {
if (!this.activePhoto) {
return null;
}
return (<PhotoDTO>this.activePhoto.gridPhoto.media).metadata.caption;
}
}