1
0
Fork 0
mirror of https://github.com/jellyfin/jellyfin-web synced 2025-03-30 19:56:21 +00:00
jellyfin-web/src/components/slideshow/slideshow.js

790 lines
27 KiB
JavaScript
Raw Normal View History

2020-03-22 15:34:34 +01:00
/**
* Image viewer component
* @module components/slideshow/slideshow
*/
2020-08-14 08:46:34 +02:00
import dialogHelper from '../dialogHelper/dialogHelper';
import inputManager from '../../scripts/inputManager';
import layoutManager from '../layoutManager';
import focusManager from '../focusManager';
import browser from '../../scripts/browser';
2020-08-16 20:24:45 +02:00
import { appHost } from '../apphost';
2020-08-14 08:46:34 +02:00
import dom from '../../scripts/dom';
2021-01-26 16:25:38 -05:00
import './style.scss';
2020-08-14 08:46:34 +02:00
import 'material-design-icons-iconfont';
import '../../elements/emby-button/paper-icon-button-light';
import ServerConnections from '../ServerConnections';
2020-11-04 11:26:37 +01:00
import screenfull from 'screenfull';
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Name of transition event.
*/
const transitionEndEventName = dom.whichTransitionEvent();
2020-05-07 23:11:19 +03:00
2020-08-06 08:44:04 +01:00
/**
* Flag to use fake image to fix blurry zoomed image.
* At least WebKit doesn't restore quality for zoomed images.
*/
const useFakeZoomImage = browser.safari;
2020-08-06 08:44:04 +01:00
/**
* Retrieves an item's image URL from the API.
* @param {object|string} item - Item used to generate the image URL.
* @param {object} options - Options of the image.
* @param {object} apiClient - API client instance used to retrieve the image.
* @returns {null|string} URL of the item's image.
*/
function getImageUrl(item, options, apiClient) {
options = options || {};
options.type = options.type || 'Primary';
2020-08-06 08:44:04 +01:00
if (typeof (item) === 'string') {
return apiClient.getScaledImageUrl(item, options);
}
2020-08-06 08:44:04 +01:00
if (item.ImageTags && item.ImageTags[options.type]) {
options.tag = item.ImageTags[options.type];
return apiClient.getScaledImageUrl(item.Id, options);
}
2022-10-03 14:22:02 -04:00
if (options.type === 'Primary' && item.AlbumId && item.AlbumPrimaryImageTag) {
options.tag = item.AlbumPrimaryImageTag;
return apiClient.getScaledImageUrl(item.AlbumId, options);
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
return null;
}
2020-08-06 08:44:04 +01:00
/**
* Retrieves a backdrop's image URL from the API.
* @param {object} item - Item used to generate the image URL.
* @param {object} options - Options of the image.
* @param {object} apiClient - API client instance used to retrieve the image.
* @returns {null|string} URL of the item's backdrop.
*/
function getBackdropImageUrl(item, options, apiClient) {
options = options || {};
options.type = options.type || 'Backdrop';
2020-08-06 08:44:04 +01:00
// If not resizing, get the original image
if (!options.maxWidth && !options.width && !options.maxHeight && !options.height && !options.fillWidth && !options.fillHeight) {
2020-08-06 08:44:04 +01:00
options.quality = 100;
}
2020-08-06 08:44:04 +01:00
if (item.BackdropImageTags && item.BackdropImageTags.length) {
options.tag = item.BackdropImageTags[0];
return apiClient.getScaledImageUrl(item.Id, options);
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
return null;
}
2020-08-06 08:44:04 +01:00
/**
* Dispatches a request for an item's image to its respective handler.
* @param {object} item - Item used to generate the image URL.
* @returns {string} URL of the item's image.
*/
function getImgUrl(item, user) {
const apiClient = ServerConnections.getApiClient(item.ServerId);
2020-08-06 08:44:04 +01:00
const imageOptions = {};
if (item.BackdropImageTags && item.BackdropImageTags.length) {
return getBackdropImageUrl(item, imageOptions, apiClient);
} else {
if (item.MediaType === 'Photo' && user && user.Policy.EnableContentDownloading) {
return apiClient.getItemDownloadUrl(item.Id);
}
2020-08-06 08:44:04 +01:00
imageOptions.type = 'Primary';
return getImageUrl(item, imageOptions, apiClient);
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Generates a button using the specified icon, classes and properties.
* @param {string} icon - Name of the material icon on the button
* @param {string} cssClass - CSS classes to assign to the button
* @param {boolean} canFocus - Flag to set the tabindex attribute on the button to -1.
* @param {boolean} autoFocus - Flag to set the autofocus attribute on the button.
* @returns {string} The HTML markup of the button.
*/
function getIcon(icon, cssClass, canFocus, autoFocus) {
const tabIndex = canFocus ? '' : ' tabindex="-1"';
autoFocus = autoFocus ? ' autofocus' : '';
2022-02-24 20:15:24 +03:00
return '<button is="paper-icon-button-light" class="autoSize ' + cssClass + '"' + tabIndex + autoFocus + '><span class="material-icons slideshowButtonIcon ' + icon + '" aria-hidden="true"></span></button>';
2020-08-06 08:44:04 +01:00
}
/**
* Sets the viewport meta tag to enable or disable scaling by the user.
* @param {boolean} scalable - Flag to set the scalability of the viewport.
*/
function setUserScalable(scalable) {
try {
appHost.setUserScalable(scalable);
} catch (err) {
console.error('error in appHost.setUserScalable: ' + err);
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
export default function (options) {
const self = this;
/** Initialized instance of Swiper. */
let swiperInstance;
/** Initialized instance of the dialog containing the Swiper instance. */
let dialog;
/** Options of the slideshow components */
let currentOptions;
/** ID of the timeout used to hide the OSD. */
let hideTimeout;
/** Last coordinates of the mouse pointer. */
let lastMouseMoveData;
2018-10-23 01:05:09 +03:00
2020-03-22 15:34:34 +01:00
/**
2020-08-06 08:44:04 +01:00
* Creates the HTML markup for the dialog and the OSD.
* @param {Object} options - Options used to create the dialog and slideshow.
2020-03-22 15:34:34 +01:00
*/
2022-10-16 16:04:37 +02:00
function createElements(slideshowOptions) {
currentOptions = slideshowOptions;
2020-08-06 08:44:04 +01:00
dialog = dialogHelper.createDialog({
2022-10-16 16:04:37 +02:00
exitAnimationDuration: slideshowOptions.interactive ? 400 : 800,
2020-08-06 08:44:04 +01:00
size: 'fullscreen',
autoFocus: false,
scrollY: false,
exitAnimation: 'fadeout',
removeOnClose: true
});
2020-08-06 08:44:04 +01:00
dialog.classList.add('slideshowDialog');
2020-08-06 08:44:04 +01:00
let html = '';
2020-08-06 08:44:04 +01:00
html += '<div class="slideshowSwiperContainer"><div class="swiper-wrapper"></div></div>';
2020-04-04 00:22:31 +02:00
2022-10-16 16:04:37 +02:00
if (slideshowOptions.interactive && !layoutManager.tv) {
2020-08-06 08:44:04 +01:00
const actionButtonsOnTop = layoutManager.mobile;
2020-08-06 08:44:04 +01:00
html += getIcon('keyboard_arrow_left', 'btnSlideshowPrevious slideshowButton hide-mouse-idle-tv', false);
html += getIcon('keyboard_arrow_right', 'btnSlideshowNext slideshowButton hide-mouse-idle-tv', false);
2020-08-06 08:44:04 +01:00
html += '<div class="topActionButtons">';
if (actionButtonsOnTop) {
2021-09-09 00:03:38 +03:00
html += getIcon('play_arrow', 'btnSlideshowPause slideshowButton', true);
2022-10-16 16:04:37 +02:00
if (appHost.supports('filedownload') && slideshowOptions.user && slideshowOptions.user.Policy.EnableContentDownloading) {
2020-08-06 08:44:04 +01:00
html += getIcon('file_download', 'btnDownload slideshowButton', true);
}
2020-08-06 08:44:04 +01:00
if (appHost.supports('sharing')) {
html += getIcon('share', 'btnShare slideshowButton', true);
}
2020-11-04 11:26:37 +01:00
if (screenfull.isEnabled) {
html += getIcon('fullscreen', 'btnFullscreen', true);
html += getIcon('fullscreen_exit', 'btnFullscreenExit hide', true);
}
}
2020-08-06 08:44:04 +01:00
html += getIcon('close', 'slideshowButton btnSlideshowExit hide-mouse-idle-tv', false);
html += '</div>';
2020-08-06 08:44:04 +01:00
if (!actionButtonsOnTop) {
html += '<div class="slideshowBottomBar hide">';
2020-08-06 08:44:04 +01:00
html += getIcon('play_arrow', 'btnSlideshowPause slideshowButton', true, true);
2022-10-16 16:04:37 +02:00
if (appHost.supports('filedownload') && slideshowOptions.user && slideshowOptions.user.Policy.EnableContentDownloading) {
2020-08-06 08:44:04 +01:00
html += getIcon('file_download', 'btnDownload slideshowButton', true);
}
2020-08-06 08:44:04 +01:00
if (appHost.supports('sharing')) {
html += getIcon('share', 'btnShare slideshowButton', true);
}
2020-11-04 11:26:37 +01:00
if (screenfull.isEnabled) {
html += getIcon('fullscreen', 'btnFullscreen', true);
html += getIcon('fullscreen_exit', 'btnFullscreenExit hide', true);
}
2020-08-06 08:44:04 +01:00
html += '</div>';
}
2020-08-06 08:44:04 +01:00
} else {
html += '<div class="slideshowImage"></div><h1 class="slideshowImageText"></h1>';
}
2020-08-06 08:44:04 +01:00
dialog.innerHTML = html;
2022-10-16 16:04:37 +02:00
if (slideshowOptions.interactive && !layoutManager.tv) {
2021-01-26 22:20:12 -05:00
dialog.querySelector('.btnSlideshowExit').addEventListener('click', function () {
2020-08-06 08:44:04 +01:00
dialogHelper.close(dialog);
});
2022-02-26 12:15:03 +03:00
dialog.querySelector('.btnSlideshowPrevious')?.addEventListener('click', getClickHandler(null));
dialog.querySelector('.btnSlideshowNext')?.addEventListener('click', getClickHandler(null));
2020-08-06 08:44:04 +01:00
const btnPause = dialog.querySelector('.btnSlideshowPause');
if (btnPause) {
2022-02-26 12:15:03 +03:00
btnPause.addEventListener('click', getClickHandler(playPause));
2020-08-06 08:44:04 +01:00
}
2020-08-06 08:44:04 +01:00
const btnDownload = dialog.querySelector('.btnDownload');
if (btnDownload) {
2022-02-26 12:15:03 +03:00
btnDownload.addEventListener('click', getClickHandler(download));
2020-08-06 08:44:04 +01:00
}
2020-08-06 08:44:04 +01:00
const btnShare = dialog.querySelector('.btnShare');
if (btnShare) {
2022-02-26 12:15:03 +03:00
btnShare.addEventListener('click', getClickHandler(share));
2020-08-06 08:44:04 +01:00
}
2020-11-04 11:26:37 +01:00
const btnFullscreen = dialog.querySelector('.btnFullscreen');
if (btnFullscreen) {
2022-02-26 12:15:03 +03:00
btnFullscreen.addEventListener('click', getClickHandler(fullscreen));
2020-11-04 11:26:37 +01:00
}
const btnFullscreenExit = dialog.querySelector('.btnFullscreenExit');
if (btnFullscreenExit) {
2022-02-26 12:15:03 +03:00
btnFullscreenExit.addEventListener('click', getClickHandler(fullscreenExit));
2020-11-04 11:26:37 +01:00
}
if (screenfull.isEnabled) {
screenfull.on('change', function () {
toggleFullscreenButtons(screenfull.isFullscreen);
});
}
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
setUserScalable(true);
dialogHelper.open(dialog).then(function () {
setUserScalable(false);
});
inputManager.on(window, onInputCommand);
/* eslint-disable-next-line compat/compat */
document.addEventListener((window.PointerEvent ? 'pointermove' : 'mousemove'), onPointerMove);
dialog.addEventListener('close', onDialogClosed);
loadSwiper(dialog, options);
2020-11-16 15:15:03 +01:00
if (layoutManager.desktop) {
const topActionButtons = dialog.querySelector('.topActionButtons');
if (topActionButtons) topActionButtons.classList.add('hide');
}
2020-11-16 15:15:03 +01:00
const btnSlideshowPrevious = dialog.querySelector('.btnSlideshowPrevious');
if (btnSlideshowPrevious) btnSlideshowPrevious.classList.add('hide');
const btnSlideshowNext = dialog.querySelector('.btnSlideshowNext');
if (btnSlideshowNext) btnSlideshowNext.classList.add('hide');
2020-08-06 08:44:04 +01:00
}
/**
* Handles OSD changes when the autoplay is started.
*/
function onAutoplayStart() {
const btnSlideshowPause = dialog.querySelector('.btnSlideshowPause .material-icons');
if (btnSlideshowPause) {
btnSlideshowPause.classList.replace('play_arrow', 'pause');
}
2020-08-06 08:44:04 +01:00
}
2020-08-06 08:44:04 +01:00
/**
* Handles OSD changes when the autoplay is stopped.
*/
function onAutoplayStop() {
const btnSlideshowPause = dialog.querySelector('.btnSlideshowPause .material-icons');
if (btnSlideshowPause) {
btnSlideshowPause.classList.replace('pause', 'play_arrow');
}
2020-08-06 08:44:04 +01:00
}
2020-08-06 08:44:04 +01:00
/**
* Handles zoom changes.
*/
2020-08-29 19:43:05 +03:00
function onZoomChange(swiper, scale, imageEl, slideEl) {
2020-08-06 08:44:04 +01:00
const zoomImage = slideEl.querySelector('.swiper-zoom-fakeimg');
if (zoomImage) {
zoomImage.style.width = zoomImage.style.height = scale * 100 + '%';
if (scale > 1) {
if (zoomImage.classList.contains('swiper-zoom-fakeimg-hidden')) {
// Await for Swiper style changes
setTimeout(() => {
const callback = () => {
imageEl.removeEventListener(transitionEndEventName, callback);
zoomImage.classList.remove('swiper-zoom-fakeimg-hidden');
};
// Swiper set 'transition-duration: 300ms' for auto zoom
// and 'transition-duration: 0s' for touch zoom
const transitionDuration = parseFloat(imageEl.style.transitionDuration.replace(/[a-z]/i, ''));
if (transitionDuration > 0) {
imageEl.addEventListener(transitionEndEventName, callback);
} else {
callback();
}
}, 0);
2020-05-07 23:11:19 +03:00
}
2020-08-06 08:44:04 +01:00
} else {
zoomImage.classList.add('swiper-zoom-fakeimg-hidden');
2020-05-07 23:11:19 +03:00
}
}
2020-08-06 08:44:04 +01:00
}
2020-05-07 23:11:19 +03:00
2020-08-06 08:44:04 +01:00
/**
* Initializes the Swiper instance and binds the relevant events.
* @param {HTMLElement} dialog - Element containing the dialog.
* @param {Object} options - Options used to initialize the Swiper instance.
*/
2022-10-16 16:04:37 +02:00
function loadSwiper(dialogElement, swiperOptions) {
2020-08-06 08:44:04 +01:00
let slides;
if (currentOptions.slides) {
slides = currentOptions.slides;
} else {
slides = currentOptions.items;
}
2022-09-22 10:23:23 -04:00
//eslint-disable-next-line import/no-unresolved
import('swiper/css/bundle');
// eslint-disable-next-line import/no-unresolved
import('swiper/bundle').then(({ Swiper }) => {
2022-10-16 16:04:37 +02:00
swiperInstance = new Swiper(dialogElement.querySelector('.slideshowSwiperContainer'), {
2022-09-22 10:23:23 -04:00
direction: 'horizontal',
// Loop is disabled due to the virtual slides option not supporting it.
loop: false,
zoom: {
minRatio: 1,
toggle: true
},
2022-10-16 16:04:37 +02:00
autoplay: !swiperOptions.interactive || !!swiperOptions.autoplay,
2022-09-22 10:23:23 -04:00
keyboard: {
enabled: true
},
preloadImages: true,
slidesPerView: 1,
slidesPerColumn: 1,
2022-10-16 16:04:37 +02:00
initialSlide: swiperOptions.startIndex || 0,
2022-09-22 10:23:23 -04:00
speed: 240,
navigation: {
nextEl: '.btnSlideshowNext',
prevEl: '.btnSlideshowPrevious'
},
// Virtual slides reduce memory consumption for large libraries while allowing preloading of images;
virtual: {
slides: slides,
cache: true,
renderSlide: getSwiperSlideHtml,
addSlidesBefore: 1,
addSlidesAfter: 1
}
});
2020-11-08 12:37:53 +00:00
2022-09-22 10:23:23 -04:00
swiperInstance.on('autoplayStart', onAutoplayStart);
swiperInstance.on('autoplayStop', onAutoplayStop);
2020-11-08 12:37:53 +00:00
2022-09-22 10:23:23 -04:00
if (useFakeZoomImage) {
swiperInstance.on('zoomChange', onZoomChange);
}
2021-09-09 00:06:09 +03:00
2022-09-22 10:23:23 -04:00
if (swiperInstance.autoplay?.running) onAutoplayStart();
});
2020-08-06 08:44:04 +01:00
}
2020-08-06 08:44:04 +01:00
/**
* Renders the HTML markup of a slide for an item or a slide.
* @param {Object} item - The item used to render the slide.
* @returns {string} The HTML markup of the slide.
*/
2021-01-26 22:20:12 -05:00
function getSwiperSlideHtml(item) {
2020-08-06 08:44:04 +01:00
if (currentOptions.slides) {
return getSwiperSlideHtmlFromSlide(item);
} else {
return getSwiperSlideHtmlFromItem(item);
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Renders the HTML markup of a slide for an item.
* @param {Object} item - Item used to generate the slide.
* @returns {string} The HTML markup of the slide.
*/
function getSwiperSlideHtmlFromItem(item) {
return getSwiperSlideHtmlFromSlide({
originalImage: getImgUrl(item, currentOptions.user),
Id: item.Id,
ServerId: item.ServerId
});
}
/**
* Renders the HTML markup of a slide for a slide object.
* @param {Object} item - Slide object used to generate the slide.
* @returns {string} The HTML markup of the slide.
*/
function getSwiperSlideHtmlFromSlide(item) {
let html = '';
html += '<div class="swiper-slide" data-original="' + item.originalImage + '" data-itemid="' + item.Id + '" data-serverid="' + item.ServerId + '">';
html += '<div class="swiper-zoom-container">';
if (useFakeZoomImage) {
html += `<div class="swiper-zoom-fakeimg swiper-zoom-fakeimg-hidden" style="background-image: url('${item.originalImage}')"></div>`;
}
html += '<img src="' + item.originalImage + '" class="swiper-slide-img">';
html += '</div>';
if (item.title || item.subtitle) {
html += '<div class="slideText">';
html += '<div class="slideTextInner">';
if (item.title) {
html += '<h1 class="slideTitle">';
html += item.title;
html += '</h1>';
}
2020-08-06 08:44:04 +01:00
if (item.description) {
html += '<div class="slideSubtitle">';
html += item.description;
html += '</div>';
}
html += '</div>';
2020-08-06 08:44:04 +01:00
html += '</div>';
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
html += '</div>';
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
return html;
}
/**
* Fetches the information of the currently displayed slide.
* @returns {null|{itemId: string, shareUrl: string, serverId: string, url: string}} Object containing the information of the currently displayed slide.
*/
function getCurrentImageInfo() {
if (swiperInstance) {
const slide = document.querySelector('.swiper-slide-active');
if (slide) {
return {
url: slide.getAttribute('data-original'),
shareUrl: slide.getAttribute('data-original'),
itemId: slide.getAttribute('data-itemid'),
serverId: slide.getAttribute('data-serverid')
};
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
return null;
} else {
return null;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Starts a download for the currently displayed slide.
*/
function download() {
const imageInfo = getCurrentImageInfo();
2020-08-14 08:46:34 +02:00
import('../../scripts/fileDownloader').then((fileDownloader) => {
2020-08-06 08:44:04 +01:00
fileDownloader.download([imageInfo]);
});
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Shares the currently displayed slide using the browser's built-in sharing feature.
*/
function share() {
const imageInfo = getCurrentImageInfo();
2020-08-06 08:44:04 +01:00
navigator.share({
url: imageInfo.shareUrl
});
}
2020-11-04 11:26:37 +01:00
/**
* Goes to fullscreen using screenfull plugin
*/
function fullscreen() {
if (!screenfull.isFullscreen) screenfull.request();
toggleFullscreenButtons(true);
}
/**
* Exits fullscreen using screenfull plugin
*/
function fullscreenExit() {
if (screenfull.isFullscreen) screenfull.exit();
toggleFullscreenButtons(false);
}
/**
* Updates the display of fullscreen buttons
* @param {boolean} isFullscreen - Whether the wanted state of buttons is fullscreen or not
*/
function toggleFullscreenButtons(isFullscreen) {
const btnFullscreen = dialog.querySelector('.btnFullscreen');
const btnFullscreenExit = dialog.querySelector('.btnFullscreenExit');
if (btnFullscreen)
btnFullscreen.classList.toggle('hide', isFullscreen);
if (btnFullscreenExit)
btnFullscreenExit.classList.toggle('hide', !isFullscreen);
}
2020-08-06 08:44:04 +01:00
/**
* Starts the autoplay feature of the Swiper instance.
*/
function play() {
if (swiperInstance.autoplay) {
swiperInstance.autoplay.start();
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Pauses the autoplay feature of the Swiper instance;
*/
function pause() {
if (swiperInstance.autoplay) {
swiperInstance.autoplay.stop();
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Toggles the autoplay feature of the Swiper instance.
*/
function playPause() {
const paused = !dialog.querySelector('.btnSlideshowPause .material-icons').classList.contains('pause');
if (paused) {
play();
} else {
pause();
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Closes the dialog and destroys the Swiper instance.
*/
function onDialogClosed() {
2020-11-04 11:26:37 +01:00
// Exits fullscreen
fullscreenExit();
2020-08-06 08:44:04 +01:00
const swiper = swiperInstance;
if (swiper) {
swiper.destroy(true, true);
swiperInstance = null;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
inputManager.off(window, onInputCommand);
/* eslint-disable-next-line compat/compat */
document.removeEventListener((window.PointerEvent ? 'pointermove' : 'mousemove'), onPointerMove);
}
2020-08-06 08:44:04 +01:00
/**
* Shows the OSD.
*/
function showOsd() {
const bottom = dialog.querySelector('.slideshowBottomBar');
if (bottom) {
slideToShow(bottom, 'down');
2018-10-23 01:05:09 +03:00
}
const topActionButtons = dialog.querySelector('.topActionButtons');
if (topActionButtons) slideToShow(topActionButtons, 'up');
const left = dialog.querySelector('.btnSlideshowPrevious');
if (left) slideToShow(left, 'left');
const right = dialog.querySelector('.btnSlideshowNext');
if (right) slideToShow(right, 'right');
startHideTimer();
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Hides the OSD.
*/
function hideOsd() {
const bottom = dialog.querySelector('.slideshowBottomBar');
if (bottom) {
slideToHide(bottom, 'down');
2018-10-23 01:05:09 +03:00
}
const topActionButtons = dialog.querySelector('.topActionButtons');
if (topActionButtons) slideToHide(topActionButtons, 'up');
const left = dialog.querySelector('.btnSlideshowPrevious');
if (left) slideToHide(left, 'left');
const right = dialog.querySelector('.btnSlideshowNext');
if (right) slideToHide(right, 'right');
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Starts the timer used to automatically hide the OSD.
*/
function startHideTimer() {
stopHideTimer();
hideTimeout = setTimeout(hideOsd, 3000);
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Stops the timer used to automatically hide the OSD.
*/
function stopHideTimer() {
if (hideTimeout) {
clearTimeout(hideTimeout);
hideTimeout = null;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
*
* @param {string} hiddenPosition - Position of the hidden element compared to when it's visible ('down', 'up', 'left', 'right')
* @param {*} fadingOut - Whether it is fading out or in
* @param {HTMLElement} element - Element to fade.
* @returns {Array} Array of keyframes
*/
function keyframesSlide(hiddenPosition, fadingOut, element) {
const visible = { transform: 'translate(0,0)', opacity: '1' };
const invisible = { opacity: '.3' };
if (hiddenPosition === 'up' || hiddenPosition === 'down') {
invisible['transform'] = 'translate3d(0,' + element.offsetHeight * (hiddenPosition === 'down' ? 1 : -1) + 'px,0)';
} else if (hiddenPosition === 'left' || hiddenPosition === 'right') {
invisible['transform'] = 'translate3d(' + element.offsetWidth * (hiddenPosition === 'right' ? 1 : -1) + 'px,0,0)';
}
return fadingOut ? [visible, invisible] : [invisible, visible];
}
/**
* Shows the element by sliding it into view.
* @param {HTMLElement} element - Element to show.
* @param {string} slideFrom - Direction to slide from ('down', 'up', 'left', 'right')
2020-08-06 08:44:04 +01:00
*/
function slideToShow(element, slideFrom) {
2020-08-06 08:44:04 +01:00
if (!element.classList.contains('hide')) {
return;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
element.classList.remove('hide');
2020-08-06 08:44:04 +01:00
const onFinish = function () {
2020-11-16 15:15:03 +01:00
const btnSlideshowPause = element.querySelector('.btnSlideshowPause');
if (btnSlideshowPause) focusManager.focus(btnSlideshowPause);
2020-08-06 08:44:04 +01:00
};
2020-08-06 08:44:04 +01:00
if (!element.animate) {
onFinish();
return;
}
2020-08-06 08:44:04 +01:00
requestAnimationFrame(function () {
const keyframes = keyframesSlide(slideFrom, false, element);
2020-08-06 08:44:04 +01:00
const timing = { duration: 300, iterations: 1, easing: 'ease-out' };
element.animate(keyframes, timing).onfinish = onFinish;
});
}
2020-08-06 08:44:04 +01:00
/**
* Hides the element by sliding it out of view.
* @param {HTMLElement} element - Element to hide.
* @param {string} slideInto - Direction to slide into ('down', 'up', 'left', 'right')
2020-08-06 08:44:04 +01:00
*/
function slideToHide(element, slideInto) {
2020-08-06 08:44:04 +01:00
if (element.classList.contains('hide')) {
return;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
const onFinish = function () {
element.classList.add('hide');
};
2020-08-06 08:44:04 +01:00
if (!element.animate) {
onFinish();
return;
}
2020-08-06 08:44:04 +01:00
requestAnimationFrame(function () {
const keyframes = keyframesSlide(slideInto, true, element);
2020-08-06 08:44:04 +01:00
const timing = { duration: 300, iterations: 1, easing: 'ease-out' };
element.animate(keyframes, timing).onfinish = onFinish;
});
}
/**
* Shows the OSD when moving the mouse pointer or touching the screen.
* @param {Event} event - Pointer movement event.
*/
function onPointerMove(event) {
const pointerType = event.pointerType || (layoutManager.mobile ? 'touch' : 'mouse');
if (pointerType === 'mouse') {
const eventX = event.screenX || 0;
const eventY = event.screenY || 0;
const obj = lastMouseMoveData;
if (!obj) {
lastMouseMoveData = {
x: eventX,
y: eventY
};
return;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
// if coord are same, it didn't move
if (Math.abs(eventX - obj.x) < 10 && Math.abs(eventY - obj.y) < 10) {
return;
}
2020-08-06 08:44:04 +01:00
obj.x = eventX;
obj.y = eventY;
2020-08-06 08:44:04 +01:00
showOsd();
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2020-08-06 08:44:04 +01:00
/**
* Dispatches keyboard inputs to their proper handlers.
* @param {Event} event - Keyboard input event.
*/
function onInputCommand(event) {
switch (event.detail.command) {
case 'up':
case 'down':
case 'select':
case 'menu':
case 'info':
showOsd();
break;
case 'play':
play();
break;
case 'pause':
pause();
break;
case 'playpause':
playPause();
break;
default:
break;
2018-10-23 01:05:09 +03:00
}
2020-08-06 08:44:04 +01:00
}
2018-10-23 01:05:09 +03:00
2022-02-26 12:15:03 +03:00
/**
* Constructs click event handler.
* @param {function|null|undefined} callback - Click event handler.
*/
function getClickHandler(callback) {
return (e) => {
showOsd();
callback?.(e);
};
}
2020-08-06 08:44:04 +01:00
/**
* Shows the slideshow component.
*/
self.show = function () {
createElements(options);
};
2020-08-06 08:44:04 +01:00
/**
* Hides the slideshow element.
*/
self.hide = function () {
if (dialog) {
dialogHelper.close(dialog);
}
};
2020-08-06 08:44:04 +01:00
}