/* * Jellyfin Slideshow by M0RPH3US v4.0.1 * Modified by CodeDevMLH * * New features: * - optional Trailer background video support * - option to make video backdrops full width * - SponsorBlock support to skip intro/outro segments * - option to always show arrows * - option to disable/enable keyboard controls * - option to show/hide trailer button if trailer as backdrop is disabled (opens in a modal) * - option to wait for trailer to end before loading next slide * - option to set a maximum for the pagination dots (will turn into a counter style if exceeded) * - option to disable loading screen * - option to put collection (boxsets) IDs into the slideshow to display their items * - option to enable client-side settings (allow users to override settings locally on their device) * - option to enable seasonal content (only show items that are relevant to the current season/holiday) * - option to prefer local trailers (from the media item) over online sources * - options to sort the content by various criteria (PremiereDate, ProductionYear, Random, Original order, etc.) */ //Core Module Configuration const CONFIG = { IMAGE_SVG: { freshTomato: 'image/svg+xml', rottenTomato: '', }, shuffleInterval: 7000, retryInterval: 500, minSwipeDistance: 50, loadingCheckInterval: 100, maxPlotLength: 360, maxMovies: 15, maxTvShows: 15, maxItems: 500, preloadCount: 3, fadeTransitionDuration: 500, maxPaginationDots: 15, slideAnimationEnabled: true, enableVideoBackdrop: true, useSponsorBlock: true, preferLocalTrailers: false, randomizeLocalTrailers: false, preferLocalBackdrops: false, randomizeThemeVideos: false, includeWatchedContent: false, waitForTrailerToEnd: true, startMuted: true, fullWidthVideo: true, enableMobileVideo: false, showTrailerButton: true, preferredVideoQuality: "Auto", enableKeyboardControls: true, alwaysShowArrows: false, enableCustomMediaIds: true, enableSeasonalContent: false, customMediaIds: "", enableLoadingScreen: true, enableClientSideSettings: false, sortBy: "Random", sortOrder: "Ascending", applyLimitsToCustomIds: false, seasonalSections: "[]", isEnabled: true, }; // State management const STATE = { jellyfinData: { userId: null, appName: null, appVersion: null, deviceName: null, deviceId: null, accessToken: null, serverAddress: null, }, slideshow: { hasInitialized: false, isTransitioning: false, isPaused: false, currentSlideIndex: 0, focusedSlide: null, containerFocused: false, slideInterval: null, itemIds: [], loadedItems: {}, createdSlides: {}, totalItems: 0, isLoading: false, videoPlayers: {}, sponsorBlockInterval: null, isMuted: CONFIG.startMuted, customTrailerUrls: {}, ytPromise: null, autoplayTimeouts: [], }, }; // Request throttling system const requestQueue = []; let isProcessingQueue = false; /** * Process the next request in the queue with throttling */ const processNextRequest = () => { if (requestQueue.length === 0) { isProcessingQueue = false; return; } isProcessingQueue = true; const { url, callback } = requestQueue.shift(); fetch(url) .then((response) => { if (response.ok) { return response; } throw new Error(`Failed to fetch: ${response.status}`); }) .then(callback) .catch((error) => { console.error("Error in throttled request:", error); }) .finally(() => { setTimeout(processNextRequest, 100); }); }; /** * Add a request to the throttled queue * @param {string} url - URL to fetch * @param {Function} callback - Callback to run on successful fetch */ const addThrottledRequest = (url, callback) => { requestQueue.push({ url, callback }); if (!isProcessingQueue) { processNextRequest(); } }; /** * Checks if the user is currently logged in * @returns {boolean} True if logged in, false otherwise */ const isUserLoggedIn = () => { try { return ( window.ApiClient && window.ApiClient._currentUser && window.ApiClient._currentUser.Id && window.ApiClient._serverInfo && window.ApiClient._serverInfo.AccessToken ); } catch (error) { console.error("Error checking login status:", error); return false; } }; /** * Initializes Jellyfin data from ApiClient * @param {Function} callback - Function to call once data is initialized */ const initJellyfinData = (callback) => { if (!window.ApiClient) { console.warn("⏳ window.ApiClient is not available yet. Retrying..."); setTimeout(() => initJellyfinData(callback), CONFIG.retryInterval); return; } try { const apiClient = window.ApiClient; STATE.jellyfinData = { userId: apiClient.getCurrentUserId() || "Not Found", appName: apiClient._appName || "Not Found", appVersion: apiClient._appVersion || "Not Found", deviceName: apiClient._deviceName || "Not Found", deviceId: apiClient._deviceId || "Not Found", accessToken: apiClient._serverInfo.AccessToken || "Not Found", serverId: apiClient._serverInfo.Id || "Not Found", serverAddress: apiClient._serverAddress || "Not Found", }; if (callback && typeof callback === "function") { callback(); } } catch (error) { console.error("Error initializing Jellyfin data:", error); setTimeout(() => initJellyfinData(callback), CONFIG.retryInterval); } }; /** * Initializes localization by loading translation chunks */ const initLocalization = async () => { try { const locale = await LocalizationUtils.getCurrentLocale(); await LocalizationUtils.loadTranslations(locale); console.log("✅ Localization initialized"); } catch (error) { console.error("Error initializing localization:", error); } }; /** * Creates and displays loading screen */ const initLoadingScreen = () => { const currentPath = window.location.href.toLowerCase().replace(window.location.origin, ""); const isHomePage = currentPath.includes("/web/#/home.html") || currentPath.includes("/web/#/home") || currentPath.includes("/web/index.html#/home.html") || currentPath === "/web/index.html#/home" || currentPath.endsWith("/web/"); if (!isHomePage) return; // Check LocalStorage for cached preference to avoid flash const cachedSetting = localStorage.getItem('mediaBarEnhanced-enableLoadingScreen'); if (cachedSetting === 'false') { return; } const loadingDiv = document.createElement("div"); loadingDiv.className = "bar-loading"; loadingDiv.id = "page-loader"; loadingDiv.innerHTML = `

`; document.body.appendChild(loadingDiv); requestAnimationFrame(() => { document.querySelector(".bar-loading h1 div").style.opacity = "1"; }); const progressBar = document.getElementById("progress-bar"); const unfilledBar = document.getElementById("unfilled-bar"); let progress = 0; let lastIncrement = 5; const progressInterval = setInterval(() => { if (progress < 95) { lastIncrement = Math.max(0.5, lastIncrement * 0.98); const randomFactor = 0.8 + Math.random() * 0.4; const increment = lastIncrement * randomFactor; progress += increment; progress = Math.min(progress, 95); progressBar.style.width = `${progress}%`; unfilledBar.style.width = `${100 - progress}%`; } }, 150); const checkInterval = setInterval(() => { const loginFormLoaded = document.querySelector(".manualLoginForm"); const activeTab = document.querySelector(".pageTabContent.is-active"); if (loginFormLoaded) { finishLoading(); return; } if (activeTab) { const tabIndex = activeTab.getAttribute("data-index"); if (tabIndex === "0") { const homeSections = document.querySelector(".homeSectionsContainer"); const slidesContainer = document.querySelector("#slides-container"); if (homeSections && slidesContainer) { finishLoading(); } } else { if ( activeTab.children.length > 0 || activeTab.innerText.trim().length > 0 ) { finishLoading(); } } } }, CONFIG.loadingCheckInterval); const finishLoading = () => { clearInterval(progressInterval); clearInterval(checkInterval); progressBar.style.transition = "width 300ms ease-in-out"; progressBar.style.width = "100%"; unfilledBar.style.width = "0%"; progressBar.addEventListener("transitionend", () => { requestAnimationFrame(() => { const loader = document.querySelector(".bar-loading"); if (loader) { loader.style.opacity = "0"; setTimeout(() => { loader.remove(); }, 300); } }); }); }; }; /** * Resets the slideshow state completely */ const resetSlideshowState = () => { console.log("🔄 Resetting slideshow state..."); if (STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.stop(); } // Destroy all video players if (STATE.slideshow.videoPlayers) { Object.values(STATE.slideshow.videoPlayers).forEach(player => { if (player && typeof player.destroy === 'function') { player.destroy(); } }); STATE.slideshow.videoPlayers = {}; } if (STATE.slideshow.sponsorBlockInterval) { clearInterval(STATE.slideshow.sponsorBlockInterval); STATE.slideshow.sponsorBlockInterval = null; } const container = document.getElementById("slides-container"); if (container) { while (container.firstChild) { container.removeChild(container.firstChild); } } STATE.slideshow.hasInitialized = false; STATE.slideshow.isTransitioning = false; STATE.slideshow.isPaused = false; STATE.slideshow.currentSlideIndex = 0; STATE.slideshow.focusedSlide = null; STATE.slideshow.containerFocused = false; STATE.slideshow.slideInterval = null; STATE.slideshow.itemIds = []; STATE.slideshow.loadedItems = {}; STATE.slideshow.createdSlides = {}; STATE.slideshow.customTrailerUrls = {}; STATE.slideshow.totalItems = 0; STATE.slideshow.isLoading = false; }; /** * Watches for login status changes */ const startLoginStatusWatcher = () => { let wasLoggedIn = false; setInterval(() => { const isLoggedIn = isUserLoggedIn(); if (isLoggedIn !== wasLoggedIn) { if (isLoggedIn) { console.log("👤 User logged in. Initializing slideshow..."); if (!STATE.slideshow.hasInitialized) { waitForApiClientAndInitialize(); } else { console.log("🔄 Slideshow already initialized, skipping"); } } else { console.log("👋 User logged out. Stopping slideshow..."); resetSlideshowState(); } wasLoggedIn = isLoggedIn; } }, 2000); }; /** * Wait for ApiClient to initialize before starting the slideshow */ const waitForApiClientAndInitialize = () => { if (window.slideshowCheckInterval) { clearInterval(window.slideshowCheckInterval); } window.slideshowCheckInterval = setInterval(() => { if (!window.ApiClient) { console.log("⏳ ApiClient not available yet. Waiting..."); return; } if ( window.ApiClient._currentUser && window.ApiClient._currentUser.Id && window.ApiClient._serverInfo && window.ApiClient._serverInfo.AccessToken ) { console.log( "🔓 User is fully logged in. Starting slideshow initialization..." ); clearInterval(window.slideshowCheckInterval); if (!STATE.slideshow.hasInitialized) { initJellyfinData(async () => { console.log("✅ Jellyfin API client initialized successfully"); await initLocalization(); await fetchPluginConfig(); slidesInit(); }); } else { console.log("🔄 Slideshow already initialized, skipping"); } } else { console.log( "🔒 Authentication incomplete. Waiting for complete login..." ); } }, CONFIG.retryInterval); }; const fetchPluginConfig = async () => { try { const response = await fetch('../MediaBarEnhanced/Config'); if (response.ok) { const pluginConfig = await response.json(); if (pluginConfig) { for (const key in pluginConfig) { const camelKey = key.charAt(0).toLowerCase() + key.slice(1); if (CONFIG.hasOwnProperty(camelKey)) { CONFIG[camelKey] = pluginConfig[key]; } } STATE.slideshow.isMuted = CONFIG.startMuted; if (!CONFIG.enableLoadingScreen) { const loader = document.querySelector(".bar-loading"); if (loader) { loader.remove(); } } // Sync to LocalStorage for next load localStorage.setItem('mediaBarEnhanced-enableLoadingScreen', CONFIG.enableLoadingScreen); console.log("✅ MediaBarEnhanced config loaded", CONFIG); } } } catch (e) { console.error("Failed to load MediaBarEnhanced config", e); } }; waitForApiClientAndInitialize(); /** * Utility functions for slide creation and management */ const SlideUtils = { /** * Sorts items based on configuration * @param {Array} items - Array of item objects * @param {string} sortBy - Sort criteria * @param {string} sortOrder - Sort order 'Ascending' or 'Descending' * @returns {Array} Sorted array of items */ sortItems(items, sortBy, sortOrder) { if (sortBy === 'Random' || sortBy === 'Original') { return items; } const simpleCompare = (a, b) => { if (a < b) return -1; if (a > b) return 1; return 0; }; const sorted = [...items].sort((a, b) => { let valA, valB; switch (sortBy) { case 'PremiereDate': valA = new Date(a.PremiereDate).getTime(); valB = new Date(b.PremiereDate).getTime(); break; case 'ProductionYear': valA = a.ProductionYear || 0; valB = b.ProductionYear || 0; break; case 'CriticRating': valA = a.CriticRating || 0; valB = b.CriticRating || 0; break; case 'CommunityRating': valA = a.CommunityRating || 0; valB = b.CommunityRating || 0; break; case 'Runtime': valA = a.RunTimeTicks || 0; valB = b.RunTimeTicks || 0; break; case 'Name': valA = (a.Name || '').toLowerCase(); valB = (b.Name || '').toLowerCase(); break; default: return 0; } return simpleCompare(valA, valB); }); if (sortOrder === 'Descending') { sorted.reverse(); } return sorted; }, /** * Shuffles array elements randomly * @param {Array} array - Array to shuffle * @returns {Array} Shuffled array */ shuffleArray(array) { const newArray = [...array]; for (let i = newArray.length - 1; i > 0; i--) { const j = Math.floor(Math.random() * (i + 1)); [newArray[i], newArray[j]] = [newArray[j], newArray[i]]; } return newArray; }, /** * Truncates text to specified length and adds ellipsis * @param {HTMLElement} element - Element containing text to truncate * @param {number} maxLength - Maximum length before truncation */ truncateText(element, maxLength) { if (!element) return; const text = element.innerText || element.textContent; if (text && text.length > maxLength) { element.innerText = text.substring(0, maxLength) + "..."; } }, /** * Creates a separator icon element * @returns {HTMLElement} Separator element */ createSeparator() { const separator = document.createElement("i"); separator.className = "material-icons fiber_manual_record separator-icon"; //material-icons radio_button_off return separator; }, /** * Creates a DOM element with attributes and properties * @param {string} tag - Element tag name * @param {Object} attributes - Element attributes * @param {string|HTMLElement} [content] - Element content * @returns {HTMLElement} Created element */ createElement(tag, attributes = {}, content = null) { const element = document.createElement(tag); Object.entries(attributes).forEach(([key, value]) => { if (key === "style" && typeof value === "object") { Object.entries(value).forEach(([prop, val]) => { element.style[prop] = val; }); } else if (key === "className") { element.className = value; } else if (key === "innerHTML") { element.innerHTML = value; } else if (key === "onclick" && typeof value === "function") { element.addEventListener("click", value); } else { element.setAttribute(key, value); } }); if (content) { if (typeof content === "string") { element.textContent = content; } else { element.appendChild(content); } } return element; }, /** * Find or create the slides container * @returns {HTMLElement} Slides container element */ getOrCreateSlidesContainer() { let container = document.getElementById("slides-container"); if (!container) { container = this.createElement("div", { id: "slides-container", className: "noautofocus", tabIndex: "-1" }); document.body.appendChild(container); } return container; }, /** * Formats genres into a readable string * @param {Array} genresArray - Array of genre strings * @returns {string} Formatted genres string */ parseGenres(genresArray) { if (Array.isArray(genresArray) && genresArray.length > 0) { return genresArray.slice(0, 3).join(this.createSeparator().outerHTML); } return "No Genre Available"; }, /** * Creates a loading indicator * @returns {HTMLElement} Loading indicator element */ createLoadingIndicator() { const loadingIndicator = this.createElement("div", { className: "slide-loading-indicator", innerHTML: `
`, }); return loadingIndicator; }, /** * Loads the YouTube IFrame API if not already loaded * @returns {Promise} */ loadYouTubeIframeAPI() { if (STATE.slideshow.ytPromise) return STATE.slideshow.ytPromise; STATE.slideshow.ytPromise = new Promise((resolve) => { if (window.YT && window.YT.Player) { resolve(window.YT); return; } window.onYouTubeIframeAPIReady = () => resolve(window.YT); if (!document.querySelector('script[src*="youtube.com/iframe_api"]')) { const tag = document.createElement('script'); tag.src = "https://www.youtube.com/iframe_api"; const firstScriptTag = document.getElementsByTagName('script')[0]; firstScriptTag.parentNode.insertBefore(tag, firstScriptTag); } }); return STATE.slideshow.ytPromise; }, /** * Opens a modal video player * @param {string} url - Video URL */ openVideoModal(url) { const existingModal = document.getElementById('video-modal-overlay'); if (existingModal) existingModal.remove(); if (STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.stop(); } STATE.slideshow.isPaused = true; const overlay = this.createElement('div', { id: 'video-modal-overlay' }); const closeModal = () => { overlay.remove(); STATE.slideshow.isPaused = false; if (STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.start(); } }; const closeButton = this.createElement('button', { className: 'modal-close-button', innerHTML: 'close', onclick: closeModal }); const contentContainer = this.createElement('div', { className: 'video-modal-content' }); let videoId = null; let isYoutube = false; try { const urlObj = new URL(url); if (urlObj.hostname.includes('youtube.com') || urlObj.hostname.includes('youtu.be')) { isYoutube = true; videoId = urlObj.searchParams.get('v'); if (!videoId && urlObj.hostname.includes('youtu.be')) { videoId = urlObj.pathname.substring(1); } } } catch (e) { console.warn("Invalid URL for modal:", url); } if (isYoutube && videoId) { const playerDiv = this.createElement('div', { id: 'modal-yt-player' }); contentContainer.appendChild(playerDiv); overlay.append(closeButton, contentContainer); document.body.appendChild(overlay); this.loadYouTubeIframeAPI().then(() => { new YT.Player('modal-yt-player', { height: '100%', width: '100%', videoId: videoId, playerVars: { autoplay: 1, controls: 1, iv_load_policy: 3, rel: 0, playsinline: 1, origin: window.location.origin, widget_referrer: window.location.href, enablejsapi: 1 } }); }); } else { const video = this.createElement('video', { src: url, controls: true, autoplay: true, className: 'video-modal-player' }); contentContainer.appendChild(video); overlay.append(closeButton, contentContainer); document.body.appendChild(overlay); } overlay.addEventListener('click', (e) => { if (e.target === overlay) { closeModal(); } }); }, }; /** * Localization utilities for fetching and using Jellyfin translations */ const LocalizationUtils = { translations: {}, locale: null, isLoading: {}, cachedLocale: null, chunkUrlCache: {}, /** * Gets the current locale from user preference, server config, or HTML tag * @returns {Promise} Locale code (e.g., "de", "en-us") */ async getCurrentLocale() { if (this.cachedLocale) { return this.cachedLocale; } let locale = null; try { if (window.ApiClient && typeof window.ApiClient.deviceId === 'function') { const deviceId = window.ApiClient.deviceId(); if (deviceId) { const deviceKey = `${deviceId}-language`; locale = localStorage.getItem(deviceKey).toLowerCase(); } } if (!locale) { locale = localStorage.getItem("language").toLowerCase(); } } catch (e) { console.warn("Could not access localStorage for language:", e); } if (!locale) { const langAttr = document.documentElement.getAttribute("lang"); if (langAttr) { locale = langAttr.toLowerCase(); } } if (window.ApiClient && STATE.jellyfinData?.accessToken) { try { const userId = window.ApiClient.getCurrentUserId(); if (userId) { const userUrl = window.ApiClient.getUrl(`Users/${userId}`); const userResponse = await fetch(userUrl, { headers: ApiUtils.getAuthHeaders(), }); if (userResponse.ok) { const userData = await userResponse.json(); if (userData.Configuration?.AudioLanguagePreference) { locale = userData.Configuration.AudioLanguagePreference.toLowerCase(); } } } } catch (error) { console.warn("Could not fetch user audio language preference:", error); } } if (!locale && window.ApiClient && STATE.jellyfinData?.accessToken) { try { const configUrl = window.ApiClient.getUrl('System/Configuration'); const configResponse = await fetch(configUrl, { headers: ApiUtils.getAuthHeaders(), }); if (configResponse.ok) { const configData = await configResponse.json(); if (configData.PreferredMetadataLanguage) { locale = configData.PreferredMetadataLanguage.toLowerCase(); if (configData.MetadataCountryCode) { locale = `${locale}-${configData.MetadataCountryCode.toLowerCase()}`; } } } } catch (error) { console.warn("Could not fetch server metadata language preference:", error); } } if (!locale) { const navLang = navigator.language || navigator.userLanguage; locale = navLang ? navLang.toLowerCase() : "en-us"; } // Convert 3-letter country codes to 2-letter if necessary if (locale.length === 3) { const countriesData = await window.ApiClient.getCountries(); const countryData = Object.values(countriesData).find(countryData => countryData.ThreeLetterISORegionName === locale.toUpperCase()); if (countryData) { locale = countryData.TwoLetterISORegionName.toLowerCase(); } } this.cachedLocale = locale; return locale; }, /** * Finds the translation chunk URL from performance entries * @param {string} locale - Locale code * @returns {string|null} URL to translation chunk or null */ findTranslationChunkUrl(locale) { const localePrefix = locale.split('-')[0]; if (this.chunkUrlCache[localePrefix]) { return this.chunkUrlCache[localePrefix]; } if (window.performance && window.performance.getEntriesByType) { try { const resources = window.performance.getEntriesByType('resource'); for (const resource of resources) { const url = resource.name || resource.url; if (url && url.includes(`${localePrefix}-json`) && url.includes('.chunk.js')) { this.chunkUrlCache[localePrefix] = url; return url; } } } catch (e) { console.warn("Error checking performance entries:", e); } } this.chunkUrlCache[localePrefix] = null; return null; }, /** * Fetches and loads translations from the chunk JSON * @param {string} locale - Locale code * @returns {Promise} */ async loadTranslations(locale) { if (this.translations[locale]) return; if (this.isLoading[locale]) { await this.isLoading[locale]; return; } const loadPromise = (async () => { try { const chunkUrl = this.findTranslationChunkUrl(locale); if (!chunkUrl) { return; } const response = await fetch(chunkUrl); if (!response.ok) { throw new Error(`Failed to fetch translations: ${response.statusText}`); } /** * @example * Standard version * ```js * "use strict"; * (self.webpackChunk = self.webpackChunk || []).push([[62634], { * 30985: function(e) { * e.exports = JSON.parse('{"Absolute":"..."}') * } * }]); * ``` * * Minified version * ```js * "use strict";(self.webpackChunk=self.webpackChunk||[]).push([[24072],{60715:function(e){e.exports=JSON.parse('{"Absolute":"..."}')}}]); * ``` */ const chunkText = await response.text(); const replaceEscaped = (text) => text.replace(/\\"/g, '"').replace(/\\n/g, '\n').replace(/\\\\/g, '\\').replace(/\\'/g, "'"); // 1. Try to remove start and end wrappers first try { // Matches from start of file to the beginning of JSON.parse(' const START = /^(.*)JSON\.parse\(['"]/gms; // Matches from the end of the JSON string to the end of the file const END = /['"]?\)?\s*}?(\r\n|\r|\n)?}?]?\)?;(\r\n|\r|\n)?$/gms; const jsonString = replaceEscaped(chunkText.replace(START, '').replace(END, '')); this.translations[locale] = JSON.parse(jsonString); return; } catch (e) { console.error('Failed to parse JSON from standard extraction.'); // Try alternative extraction below } // 2. Try to extract only the JSON string directly let jsonMatch = chunkText.match(/JSON\.parse\(['"](.*?)['"]\)/); if (jsonMatch) { try { const jsonString = replaceEscaped(jsonMatch[1]); this.translations[locale] = JSON.parse(jsonString); return; } catch (e) { console.error('Failed to parse JSON from direct extraction.'); // Try direct extraction } } // 3. Fallback: extract everything between the first { and the last } const jsonStart = chunkText.indexOf('{'); const jsonEnd = chunkText.lastIndexOf('}') + 1; if (jsonStart !== -1 && jsonEnd > jsonStart) { const jsonString = chunkText.substring(jsonStart, jsonEnd); try { this.translations[locale] = JSON.parse(jsonString); return; } catch (e) { console.error("Failed to parse JSON from chunk:", e); } } } catch (error) { console.error("Error loading translations:", error); } finally { delete this.isLoading[locale]; } })(); this.isLoading[locale] = loadPromise; await loadPromise; }, /** * Gets a localized string (synchronous - translations must be loaded first) * @param {string} key - Localization key (e.g., "EndsAtValue", "Play") * @param {string} fallback - Fallback English string * @param {...any} args - Optional arguments for placeholders (e.g., {0}, {1}) * @returns {string} Localized string or fallback */ getLocalizedString(key, fallback, ...args) { const locale = this.cachedLocale || 'en-us'; let translated = this.translations[locale]?.[key] || fallback; if (args.length > 0) { for (let i = 0; i < args.length; i++) { translated = translated.replace(new RegExp(`\\{${i}\\}`, 'g'), args[i]); } } return translated; } }; /** * API utilities for fetching data from Jellyfin server */ const ApiUtils = { /** * Fetches details for a specific item by ID * @param {string} itemId - Item ID * @returns {Promise} Item details */ async fetchItemDetails(itemId) { try { if (STATE.slideshow.loadedItems[itemId]) { return STATE.slideshow.loadedItems[itemId]; } const response = await fetch( // `${STATE.jellyfinData.serverAddress}/Items/${itemId}`, `${STATE.jellyfinData.serverAddress}/Items/${itemId}?Fields=Overview,RemoteTrailers,Genres,CommunityRating,CriticRating,OfficialRating,PremiereDate,ProductionYear,MediaSources,RunTimeTicks,LocalTrailerCount`, { headers: this.getAuthHeaders(), } ); if (!response.ok) { throw new Error(`Failed to fetch item details: ${response.statusText}`); } const itemData = await response.json(); STATE.slideshow.loadedItems[itemId] = itemData; return itemData; } catch (error) { console.error(`Error fetching details for item ${itemId}:`, error); return null; } }, /** * Fetch item IDs from the list file * @returns {Promise} Array of item IDs */ // MARK: LIST FILE async fetchItemIdsFromList() { try { const listFileName = `${STATE.jellyfinData.serverAddress}/web/avatars/list.txt?userId=${STATE.jellyfinData.userId}`; const response = await fetch(listFileName); if (!response.ok) { console.warn("list.txt not found or inaccessible. Using random items."); return []; } const text = await response.text(); return text .split("\n") .map((id) => id.trim()) .filter((id) => id) .slice(1); } catch (error) { console.error("Error fetching list.txt:", error); return []; } }, /** * Fetches random items from the server * @returns {Promise} Array of item objects */ async fetchItemIdsFromServer() { try { if ( !STATE.jellyfinData.accessToken || STATE.jellyfinData.accessToken === "Not Found" ) { console.warn("Access token not available. Delaying API request..."); return []; } if ( !STATE.jellyfinData.serverAddress || STATE.jellyfinData.serverAddress === "Not Found" ) { console.warn("Server address not available. Delaying API request..."); return []; } console.log("Fetching random items from server..."); let sortParams = `sortBy=${CONFIG.sortBy}`; if (CONFIG.sortBy === 'Random' || CONFIG.sortBy === 'Original') { sortParams = 'sortBy=Random'; } else { sortParams += `&sortOrder=${CONFIG.sortOrder}`; } // Filter by isPlayed=False unless IncludeWatchedContent is enabled const playedFilter = CONFIG.includeWatchedContent ? '' : '&isPlayed=False'; const response = await fetch( `${STATE.jellyfinData.serverAddress}/Items?IncludeItemTypes=Movie,Series&Recursive=true&hasOverview=true&imageTypes=Logo,Backdrop&${sortParams}${playedFilter}&enableUserData=true&Limit=${CONFIG.maxItems}&fields=Id`, { headers: this.getAuthHeaders(), } ); if (!response.ok) { console.error( `Failed to fetch items: ${response.status} ${response.statusText}` ); return []; } const data = await response.json(); const items = data.Items || []; console.log( `Successfully fetched ${items.length} random items from server` ); return items.map((item) => item.Id); } catch (error) { console.error("Error fetching item IDs:", error); return []; } }, /** * Get authentication headers for API requests * @returns {Object} Headers object */ getAuthHeaders() { return { Authorization: `MediaBrowser Client="${STATE.jellyfinData.appName}", Device="${STATE.jellyfinData.deviceName}", DeviceId="${STATE.jellyfinData.deviceId}", Version="${STATE.jellyfinData.appVersion}", Token="${STATE.jellyfinData.accessToken}"`, }; }, /** * Send a command to play an item * @param {string} itemId - Item ID to play * @returns {Promise} Success status */ async playItem(itemId) { try { const sessionId = await this.getSessionId(); if (!sessionId) { console.error("Session ID not found."); return false; } const playUrl = `${STATE.jellyfinData.serverAddress}/Sessions/${sessionId}/Playing?playCommand=PlayNow&itemIds=${itemId}`; const playResponse = await fetch(playUrl, { method: "POST", headers: this.getAuthHeaders(), }); if (!playResponse.ok) { throw new Error( `Failed to send play command: ${playResponse.statusText}` ); } console.log("Play command sent successfully to session:", sessionId); return true; } catch (error) { console.error("Error sending play command:", error); return false; } }, /** * Gets current session ID * @returns {Promise} Session ID or null */ async getSessionId() { try { const response = await fetch( `${STATE.jellyfinData.serverAddress }/Sessions?deviceId=${encodeURIComponent(STATE.jellyfinData.deviceId)}`, { headers: this.getAuthHeaders(), } ); if (!response.ok) { throw new Error(`Failed to fetch session data: ${response.statusText}`); } const sessions = await response.json(); if (!sessions || sessions.length === 0) { console.warn( "No sessions found for deviceId:", STATE.jellyfinData.deviceId ); return null; } return sessions[0].Id; } catch (error) { console.error("Error fetching session data:", error); return null; } }, //Favorites async toggleFavorite(itemId, button) { try { const userId = STATE.jellyfinData.userId; const isFavorite = button.classList.contains("favorited"); const url = `${STATE.jellyfinData.serverAddress}/Users/${userId}/FavoriteItems/${itemId}`; const method = isFavorite ? "DELETE" : "POST"; const response = await fetch(url, { method, headers: { ...ApiUtils.getAuthHeaders(), "Content-Type": "application/json", }, }); if (!response.ok) { throw new Error(`Failed to toggle favorite: ${response.statusText}`); } button.classList.toggle("favorited", !isFavorite); } catch (error) { console.error("Error toggling favorite:", error); } }, /** * Fetches SponsorBlock segments for a YouTube video * @param {string} videoId - YouTube Video ID * @returns {Promise} Object containing intro and outro segments */ async fetchSponsorBlockData(videoId) { if (!CONFIG.useSponsorBlock) return { intro: null, outro: null }; // Return cached result if available if (!this._sponsorBlockCache) this._sponsorBlockCache = {}; if (this._sponsorBlockCache[videoId]) { return this._sponsorBlockCache[videoId]; } try { const response = await fetch(`https://sponsor.ajay.app/api/skipSegments?videoID=${videoId}&categories=["intro","outro"]`); if (!response.ok) { const result = { intro: null, outro: null }; this._sponsorBlockCache[videoId] = result; return result; } const segments = await response.json(); let intro = null; let outro = null; segments.forEach(segment => { if (segment.category === "intro" && Array.isArray(segment.segment)) { intro = segment.segment; } else if (segment.category === "outro" && Array.isArray(segment.segment)) { outro = segment.segment; } }); const result = { intro, outro }; this._sponsorBlockCache[videoId] = result; return result; } catch (error) { console.warn('Error fetching SponsorBlock data:', error); return { intro: null, outro: null }; } }, /** * Searches for a Collection or Playlist by name * @param {string} name - Name to search for * @returns {Promise} ID of the first match or null */ async findCollectionOrPlaylistByName(name) { try { const response = await fetch( `${STATE.jellyfinData.serverAddress}/Items?IncludeItemTypes=BoxSet,Playlist&Recursive=true&searchTerm=${encodeURIComponent(name)}&Limit=1&fields=Id&userId=${STATE.jellyfinData.userId}`, { headers: this.getAuthHeaders(), } ); if (!response.ok) { console.warn(`Failed to search for '${name}'`); return null; } const data = await response.json(); if (data.Items && data.Items.length > 0) { return data.Items[0].Id; } return null; } catch (error) { console.error(`Error searching for '${name}':`, error); return null; } }, /** * Fetches items belonging to a collection (BoxSet) * @param {string} collectionId - ID of the collection * @returns {Promise} Array of item IDs */ async fetchCollectionItems(collectionId) { try { const response = await fetch( `${STATE.jellyfinData.serverAddress}/Items?ParentId=${collectionId}&Recursive=true&IncludeItemTypes=Movie,Series&fields=Id,Type&userId=${STATE.jellyfinData.userId}`, { headers: this.getAuthHeaders(), } ); if (!response.ok) { console.warn(`Failed to fetch collection items for ${collectionId}`); return []; } const data = await response.json(); const items = data.Items || []; console.log(`Resolved collection ${collectionId} to ${items.length} items`); return items.map(i => ({ Id: i.Id, Type: i.Type })); } catch (error) { console.error(`Error fetching collection items for ${collectionId}:`, error); return []; } }, /** * Fetches the first local trailer for an item * @param {string} itemId - Item ID * @returns {Promise} Trailer data object {id, url} or null */ async fetchLocalTrailer(itemId) { try { const response = await fetch( `${STATE.jellyfinData.serverAddress}/Users/${STATE.jellyfinData.userId}/Items/${itemId}/LocalTrailers`, { headers: this.getAuthHeaders(), } ); if (!response.ok) { return null; } const trailers = await response.json(); if (trailers && trailers.length > 0) { let trailer; if (CONFIG.randomizeLocalTrailers && trailers.length > 1) { const randomIndex = Math.floor(Math.random() * trailers.length); trailer = trailers[randomIndex]; console.log(`Using random local trailer (${randomIndex + 1}/${trailers.length}) for ${itemId}: ${trailer.Name}`); } else { trailer = trailers[0]; } const mediaSourceId = trailer.MediaSources && trailer.MediaSources[0] ? trailer.MediaSources[0].Id : trailer.Id; return { id: trailer.Id, url: `${STATE.jellyfinData.serverAddress}/Videos/${trailer.Id}/stream.mp4?mediaSourceId=${mediaSourceId}&api_key=${STATE.jellyfinData.accessToken}` }; } return null; } catch (error) { console.error(`Error fetching local trailer for ${itemId}:`, error); return null; } }, /** * Fetches theme videos for an item * @param {string} itemId - Item ID * @returns {Promise} Theme video data object {id, url} or null */ async fetchThemeVideos(itemId) { try { const response = await fetch( `${STATE.jellyfinData.serverAddress}/Items/${itemId}/ThemeVideos?userId=${STATE.jellyfinData.userId}`, { headers: this.getAuthHeaders() } ); if (response.ok) { const data = await response.json(); const items = Array.isArray(data) ? data : (data.Items || []); if (items.length > 0) { let video; if (CONFIG.randomizeThemeVideos && items.length > 1) { const randomIndex = Math.floor(Math.random() * items.length); video = items[randomIndex]; console.log(`Found Theme Video (Random ${randomIndex + 1}/${items.length}) via ThemeVideos endpoint: ${video.Name} (${video.Id})`); } else { video = items[0]; console.log(`Found Theme Video (First) via ThemeVideos endpoint: ${video.Name} (${video.Id})`); } return { id: video.Id, url: `${STATE.jellyfinData.serverAddress}/Videos/${video.Id}/stream.mp4?api_key=${STATE.jellyfinData.accessToken}` }; } } return null; } catch (error) { console.error(`Error fetching theme videos for ${itemId}:`, error); return null; } } }; /** * Class for managing slide timing */ class SlideTimer { /** * Creates a new slide timer * @param {Function} callback - Function to call on interval * @param {number} interval - Interval in milliseconds */ constructor(callback, interval) { this.callback = callback; this.interval = interval; this.timerId = null; this.start(); } /** * Stops the timer * @returns {SlideTimer} This instance for chaining */ stop() { if (this.timerId) { clearInterval(this.timerId); this.timerId = null; } return this; } /** * Starts the timer * @returns {SlideTimer} This instance for chaining */ start() { if (!this.timerId) { this.timerId = setInterval(this.callback, this.interval); } return this; } /** * Restarts the timer * @returns {SlideTimer} This instance for chaining */ restart() { return this.stop().start(); } } /** * Observer for handling slideshow visibility based on current page */ const VisibilityObserver = { updateVisibility() { const videoPlayer = document.querySelector('.videoPlayerContainer'); const trailerPlayer = document.querySelector('.youtubePlayerContainer'); // If a full screen video player is active, hide slideshow and stop playback if ((videoPlayer && !videoPlayer.classList.contains('hide')) || (trailerPlayer && !trailerPlayer.classList.contains('hide'))) { const container = document.getElementById("slides-container"); if (container) { container.style.display = "none"; container.style.visibility = "hidden"; container.style.pointerEvents = "none"; } if (STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.stop(); } SlideshowManager.stopAllPlayback(); return; } const activeTab = document.querySelector(".emby-tab-button-active"); const container = document.getElementById("slides-container"); if (!container) return; const isVisible = (window.location.hash === "#/home.html" || window.location.hash === "#/home") && activeTab && activeTab.getAttribute("data-index") === "0"; container.style.display = isVisible ? "block" : "none"; container.style.visibility = isVisible ? "visible" : "hidden"; container.style.pointerEvents = isVisible ? "auto" : "none"; if (isVisible) { if (STATE.slideshow.slideInterval && !STATE.slideshow.isPaused) { STATE.slideshow.slideInterval.start(); SlideshowManager.resumeActivePlayback(); } } else { if (STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.stop(); } SlideshowManager.stopAllPlayback(); } }, /** * Initializes visibility observer */ init() { const observer = new MutationObserver(() => this.updateVisibility()); observer.observe(document.body, { childList: true, subtree: true }); document.body.addEventListener("click", () => this.updateVisibility()); window.addEventListener("hashchange", () => this.updateVisibility()); this.updateVisibility(); }, }; /** * Slideshow UI creation and management */ const SlideCreator = { /** * Builds a tag-based image URL for cache-friendly image requests * @param {Object} item - Item data containing ImageTags * @param {string} imageType - Image type (Backdrop, Logo, Primary, etc.) * @param {number} [index] - Image index (for Backdrop, Primary, etc.) * @param {string} serverAddress - Server address * @param {number} [quality] - Image quality (0-100). If tag is available, both tag and quality are used. * @returns {string} Image URL with tag parameter (and quality if tag available), or quality-only fallback */ buildImageUrl(item, imageType, index, serverAddress, quality) { const itemId = item.Id; let tag = null; // Handle Backdrop images if (imageType === "Backdrop") { // Check BackdropImageTags array first if (item.BackdropImageTags && Array.isArray(item.BackdropImageTags) && item.BackdropImageTags.length > 0) { const backdropIndex = index !== undefined ? index : 0; if (backdropIndex < item.BackdropImageTags.length) { tag = item.BackdropImageTags[backdropIndex]; } } // Fallback to ImageTags.Backdrop if BackdropImageTags not available if (!tag && item.ImageTags && item.ImageTags.Backdrop) { tag = item.ImageTags.Backdrop; } } else { // For other image types (Logo, Primary, etc.), use ImageTags if (item.ImageTags && item.ImageTags[imageType]) { tag = item.ImageTags[imageType]; } } // Build base URL path let baseUrl; if (index !== undefined) { baseUrl = `${serverAddress}/Items/${itemId}/Images/${imageType}/${index}`; } else { baseUrl = `${serverAddress}/Items/${itemId}/Images/${imageType}`; } // Build URL with tag and quality if tag is available, otherwise quality-only fallback if (tag) { // Use both tag and quality for cacheable, quality-controlled images const qualityParam = quality !== undefined ? `&quality=${quality}` : ''; return `${baseUrl}?tag=${tag}${qualityParam}`; } else { // Fallback to quality-only URL if no tag is available const qualityParam = quality !== undefined ? quality : 90; return `${baseUrl}?quality=${qualityParam}`; } }, /** * Creates a slide element for an item * @param {Object} item - Item data * @param {string} title - Title type (Movie/TV Show) * @returns {HTMLElement} Slide element */ createSlideElement(item, title) { if (!item || !item.Id) { console.error("Invalid item data:", item); return null; } const itemId = item.Id; const serverAddress = STATE.jellyfinData.serverAddress; const slide = SlideUtils.createElement("a", { className: "slide", target: "_top", rel: "noreferrer", tabIndex: 0, "data-item-id": itemId, }); let videoBackdrop; let backdrop; let isVideo = false; let trailerUrl = null; // 1. Check for Remote/Local Trailers // Priority: Custom Config URL > (PreferLocal -> Local) > Metadata RemoteTrailer // 1a. Custom URL override if (STATE.slideshow.customTrailerUrls && STATE.slideshow.customTrailerUrls[itemId]) { const customValue = STATE.slideshow.customTrailerUrls[itemId]; // Check if the custom value is a Jellyfin Item ID (GUID) const guidMatch = customValue.match(/^([0-9a-f]{32})$/i); if (guidMatch) { const videoId = guidMatch[1]; console.log(`Using custom local video ID for ${itemId}: ${videoId}`); trailerUrl = { id: videoId, url: `${STATE.jellyfinData.serverAddress}/Videos/${videoId}/stream.mp4?api_key=${STATE.jellyfinData.accessToken}` }; } else { // Assume it's a standard URL (YouTube, etc.) trailerUrl = customValue; console.log(`Using custom trailer URL for ${itemId}: ${trailerUrl}`); } } // 1b. Check Theme Video if preferred (Local Backdrop) else if (CONFIG.preferLocalBackdrops && item.themeVideoUrl) { trailerUrl = item.themeVideoUrl; console.log(`Using theme video (local backdrop) for ${itemId}: ${trailerUrl.url || trailerUrl}`); } // 1c. Check Local Trailer if preferred else if (CONFIG.preferLocalTrailers && item.LocalTrailerCount > 0 && item.localTrailerUrl) { trailerUrl = item.localTrailerUrl; console.log(`Using local trailer for ${itemId}: ${trailerUrl}`); } // 1d. Fallback to Remote Trailer else if (item.RemoteTrailers && item.RemoteTrailers.length > 0) { trailerUrl = item.RemoteTrailers[0].Url; } // 1e. Final Fallback to Local Trailer (even if not preferred) else if (item.LocalTrailerCount > 0 && item.localTrailerUrl) { trailerUrl = item.localTrailerUrl; console.log(`Using local trailer fallback for ${itemId}: ${trailerUrl}`); } const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent); // Client Setting Overrides const enableVideo = MediaBarEnhancedSettingsManager.getSetting('videoBackdrops', CONFIG.enableVideoBackdrop); const enableMobileVideo = MediaBarEnhancedSettingsManager.getSetting('mobileVideo', CONFIG.enableMobileVideo); const shouldPlayVideo = enableVideo && (!isMobile || enableMobileVideo); if (trailerUrl && shouldPlayVideo) { let isYoutube = false; let videoId = null; try { let urlToCheck = trailerUrl; if (typeof trailerUrl === 'object' && trailerUrl.url) { urlToCheck = trailerUrl.url; } const urlObjChecked = new URL(urlToCheck); if (urlObjChecked.hostname.includes('youtube.com') || urlObjChecked.hostname.includes('youtu.be')) { isYoutube = true; videoId = urlObjChecked.searchParams.get('v'); if (!videoId && urlObjChecked.hostname.includes('youtu.be')) { videoId = urlObjChecked.pathname.substring(1); } } } catch (e) { console.warn("Invalid trailer URL:", trailerUrl); } if (isYoutube && videoId) { isVideo = true; // Create container for YouTube API const videoClass = CONFIG.fullWidthVideo ? "video-backdrop-full" : "video-backdrop-default"; // Create a wrapper for opacity transition videoBackdrop = SlideUtils.createElement("div", { className: `backdrop video-backdrop ${videoClass}`, style: "opacity: 0; transition: opacity 1.2s ease-in-out;" // Start interrupted/transparent }); const ytPlayerDiv = SlideUtils.createElement("div", { id: `youtube-player-${itemId}`, style: "width: 100%; height: 100%;" }); videoBackdrop.appendChild(ytPlayerDiv); // Initialize YouTube Player SlideUtils.loadYouTubeIframeAPI().then(() => { // Fetch SponsorBlock data ApiUtils.fetchSponsorBlockData(videoId).then(segments => { const playerVars = { autoplay: 0, mute: STATE.slideshow.isMuted ? 1 : 0, controls: 0, disablekb: 1, fs: 0, iv_load_policy: 3, rel: 0, loop: 0, playsinline: 1, origin: window.location.origin, widget_referrer: window.location.href, enablejsapi: 1 }; // Determine video quality let quality = 'hd1080'; if (CONFIG.preferredVideoQuality === 'Maximum') { quality = 'highres'; } else if (CONFIG.preferredVideoQuality === '720p') { quality = 'hd720'; } else if (CONFIG.preferredVideoQuality === '1080p') { quality = 'hd1080'; } else { // Auto or fallback // If screen is wider than 1920, prefer highres, otherwise 1080p quality = window.screen.width > 1920 ? 'highres' : 'hd1080'; } playerVars.suggestedQuality = quality; // Apply SponsorBlock start/end times if (segments.intro) { playerVars.start = Math.ceil(segments.intro[1]); console.info(`SponsorBlock intro detected for video ${videoId}: skipping to ${playerVars.start}s`); } if (segments.outro) { playerVars.end = Math.floor(segments.outro[0]); console.info(`SponsorBlock outro detected for video ${videoId}: ending at ${playerVars.end}s`); } STATE.slideshow.videoPlayers[itemId] = new YT.Player(`youtube-player-${itemId}`, { height: '100%', width: '100%', videoId: videoId, playerVars: playerVars, events: { 'onReady': (event) => { // Store start/end time and videoId for later use event.target._startTime = playerVars.start || 0; event.target._endTime = playerVars.end || undefined; event.target._videoId = videoId; // Store reference to wrapper for fading event.target._wrapperDiv = videoBackdrop; if (STATE.slideshow.isMuted) { event.target.mute(); } else { event.target.unMute(); event.target.setVolume(40); } if (typeof event.target.setPlaybackQuality === 'function') { event.target.setPlaybackQuality(quality); } // Only play if this is the active slide const slide = document.querySelector(`.slide[data-item-id="${itemId}"]`); const isVideoPlayerOpen = document.querySelector('.videoPlayerContainer') || document.querySelector('.youtubePlayerContainer'); if (slide && slide.classList.contains('active') && !document.hidden && (!isVideoPlayerOpen || isVideoPlayerOpen.classList.contains('hide'))) { event.target.playVideo(); // Check if it actually started playing after a short delay (handling autoplay blocks) const timeoutId = setTimeout(() => { // Re-check conditions before processing fallback const isVideoPlayerOpenNow = document.querySelector('.videoPlayerContainer') || document.querySelector('.youtubePlayerContainer'); if (document.hidden || (isVideoPlayerOpenNow && !isVideoPlayerOpenNow.classList.contains('hide')) || !slide.classList.contains('active')) { console.log(`Navigation detected during autoplay check for ${itemId}, stopping video.`); try { event.target.stopVideo(); } catch (e) { console.warn("Error stopping video in timeout:", e); } return; } if (event.target.getPlayerState() !== YT.PlayerState.PLAYING && event.target.getPlayerState() !== YT.PlayerState.BUFFERING) { console.warn(`Autoplay blocked for ${itemId}, attempting muted fallback`); event.target.mute(); event.target.playVideo(); } }, 1000); if (!STATE.slideshow.autoplayTimeouts) STATE.slideshow.autoplayTimeouts = []; STATE.slideshow.autoplayTimeouts.push(timeoutId); // Pause slideshow timer when video starts if configured if (CONFIG.waitForTrailerToEnd && STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.stop(); } } }, 'onStateChange': (event) => { // Fade in when playing if (event.data === YT.PlayerState.PLAYING) { if (event.target._wrapperDiv) { event.target._wrapperDiv.style.opacity = "1"; } } if (event.data === YT.PlayerState.ENDED) { const slide = document.querySelector(`.slide[data-item-id="${itemId}"]`); if (slide && slide.classList.contains('active')) { SlideshowManager.nextSlide(); } } }, 'onError': (event) => { console.warn(`YouTube player error ${event.data} for video ${videoId}`); // Fallback to next slide on error if (CONFIG.waitForTrailerToEnd) { SlideshowManager.nextSlide(); } } } }); }); }); // 2. Check for local video trailers in MediaSources if yt is not available } else if (!isYoutube) { isVideo = true; const videoSrc = (typeof trailerUrl === 'object' ? trailerUrl.url : trailerUrl); const videoAttributes = { className: "backdrop video-backdrop", preload: "none", disablePictureInPicture: true, "data-src": videoSrc, style: "object-fit: cover; object-position: center center; width: 100%; height: 100%; position: absolute; top: 0; left: 0; pointer-events: none; opacity: 0; transition: opacity 1.2s ease-in-out;" }; videoAttributes.muted = ""; videoBackdrop = SlideUtils.createElement("video", videoAttributes); videoBackdrop.volume = 0.4; STATE.slideshow.videoPlayers[itemId] = videoBackdrop; videoBackdrop.addEventListener('play', (event) => { const slide = document.querySelector(`.slide[data-item-id="${itemId}"]`); if (!slide || !slide.classList.contains('active')) { console.log(`Local video ${itemId} started playing but slide is not active, pausing.`); event.target.pause(); event.target.currentTime = 0; return; } // Fade in event.target.style.opacity = "1"; if (CONFIG.waitForTrailerToEnd && STATE.slideshow.slideInterval) { STATE.slideshow.slideInterval.stop(); } }); videoBackdrop.addEventListener('ended', (event) => { const slide = event.target.closest('.slide'); if (slide && slide.classList.contains('active')) { SlideshowManager.nextSlide(); } }); videoBackdrop.addEventListener('error', (event) => { console.warn(`Local video error for item ${itemId}`); const slide = event.target.closest('.slide'); if (slide && slide.classList.contains('active')) { SlideshowManager.nextSlide(); } }); } } // Always create a static backdrop image (to show while video loads or if no video) backdrop = SlideUtils.createElement("img", { className: "backdrop high-quality", src: this.buildImageUrl(item, "Backdrop", 0, serverAddress, 60), alt: LocalizationUtils.getLocalizedString('Backdrop', 'Backdrop'), loading: "eager", }); // If video, static backdrop should be strictly a background (no animation) if (isVideo) { backdrop.style.animation = "none"; backdrop.style.transition = "none"; } const backdropOverlay = SlideUtils.createElement("div", { className: "backdrop-overlay", }); const backdropContainer = SlideUtils.createElement("div", { className: "backdrop-container" + (isVideo && CONFIG.fullWidthVideo ? " full-width-video" : ""), }); backdropContainer.append(backdrop, backdropOverlay); // If video exists, append on top of static backdrop if (isVideo && videoBackdrop) { backdropContainer.appendChild(videoBackdrop); } const logo = SlideUtils.createElement("img", { className: "logo high-quality", src: this.buildImageUrl(item, "Logo", undefined, serverAddress, 40), alt: item.Name, loading: "eager", }); const logoContainer = SlideUtils.createElement("div", { className: "logo-container", }); logoContainer.appendChild(logo); const featuredContent = SlideUtils.createElement( "div", { className: "featured-content", }, title ); const plot = item.Overview || "No overview available"; const plotElement = SlideUtils.createElement( "div", { className: "plot", }, plot ); SlideUtils.truncateText(plotElement, CONFIG.maxPlotLength); const plotContainer = SlideUtils.createElement("div", { className: "plot-container", }); plotContainer.appendChild(plotElement); const gradientOverlay = SlideUtils.createElement("div", { className: "gradient-overlay" + (isVideo && CONFIG.fullWidthVideo ? " full-width-video" : ""), }); const infoContainer = SlideUtils.createElement("div", { className: "info-container", }); const ratingInfo = this.createRatingInfo(item); infoContainer.appendChild(ratingInfo); const genreElement = SlideUtils.createElement("div", { className: "genre", innerHTML: SlideUtils.parseGenres(item.Genres) }); const buttonContainer = SlideUtils.createElement("div", { className: "button-container", }); const playButton = this.createPlayButton(itemId); const detailButton = this.createDetailButton(itemId); const favoriteButton = this.createFavoriteButton(item); if (trailerUrl && !isVideo && CONFIG.showTrailerButton) { const trailerButton = this.createTrailerButton(trailerUrl); buttonContainer.append(detailButton, playButton, trailerButton, favoriteButton); } else { buttonContainer.append(detailButton, playButton, favoriteButton); } slide.append( logoContainer, backdropContainer, gradientOverlay, featuredContent, plotContainer, infoContainer, genreElement, buttonContainer ); return slide; }, /** * Creates the rating information element * @param {Object} item - Item data * @returns {HTMLElement} Rating information element */ createRatingInfo(item) { const { CommunityRating: communityRating, CriticRating: criticRating, OfficialRating: ageRating, PremiereDate: premiereDate, RunTimeTicks: runtime, ChildCount: seasonCount, } = item; const miscInfo = SlideUtils.createElement("div", { className: "misc-info", }); // Community Rating Section (IMDb) if (typeof communityRating === "number") { const container = SlideUtils.createElement("div", { className: "star-rating-container", innerHTML: `${communityRating.toFixed(1)}`, }); miscInfo.appendChild(container); miscInfo.appendChild(SlideUtils.createSeparator()); } // Critic Rating Section (Rotten Tomatoes) if (typeof criticRating === "number") { const svgIcon = criticRating < 60 ? CONFIG.IMAGE_SVG.rottenTomato : CONFIG.IMAGE_SVG.freshTomato; const container = SlideUtils.createElement("div", { className: "critic-rating", innerHTML: `${svgIcon}${criticRating.toFixed(0)}%`, }) miscInfo.appendChild(container); miscInfo.appendChild(SlideUtils.createSeparator()); }; // Year Section if (typeof premiereDate === "string" && !isNaN(new Date(premiereDate))) { const container = SlideUtils.createElement("div", { className: "date", innerHTML: new Date(premiereDate).getFullYear(), }); miscInfo.appendChild(container); miscInfo.appendChild(SlideUtils.createSeparator()); }; // Age Rating Section if (typeof ageRating === "string") { const container = SlideUtils.createElement("div", { className: "age-rating mediaInfoOfficialRating", rating: ageRating, ariaLabel: `Content rated ${ageRating}`, title: `Rating: ${ageRating}`, innerHTML: ageRating, }); miscInfo.appendChild(container); miscInfo.appendChild(SlideUtils.createSeparator()); }; // Runtime / Seasons Section if (seasonCount !== undefined || runtime !== undefined) { const container = SlideUtils.createElement("div", { className: "runTime", }); if (seasonCount) { const seasonText = seasonCount <= 1 ? LocalizationUtils.getLocalizedString('Season', 'Season') : LocalizationUtils.getLocalizedString('TypeOptionPluralSeason', 'Seasons'); container.innerHTML = `${seasonCount} ${seasonText}`; } else { const milliseconds = runtime / 10000; const currentTime = new Date(); const endTime = new Date(currentTime.getTime() + milliseconds); const options = { hour: "2-digit", minute: "2-digit", hour12: false }; const formattedEndTime = endTime.toLocaleTimeString([], options); const endsAtText = LocalizationUtils.getLocalizedString('EndsAtValue', 'Ends at {0}', formattedEndTime); container.innerText = endsAtText; } miscInfo.appendChild(container); } return miscInfo; }, /** * Creates a play button for an item * @param {string} itemId - Item ID * @returns {HTMLElement} Play button element */ createPlayButton(itemId) { const playText = LocalizationUtils.getLocalizedString('Play', 'Play'); return SlideUtils.createElement("button", { className: "detailButton btnPlay play-button", innerHTML: ` ${playText} `, tabIndex: "0", onclick: (e) => { e.preventDefault(); e.stopPropagation(); ApiUtils.playItem(itemId); }, }); }, /** * Creates a detail button for an item * @param {string} itemId - Item ID * @returns {HTMLElement} Detail button element */ createDetailButton(itemId) { return SlideUtils.createElement("button", { className: "detailButton detail-button", tabIndex: "0", onclick: (e) => { e.preventDefault(); e.stopPropagation(); if (window.Emby && window.Emby.Page) { Emby.Page.show( `/details?id=${itemId}&serverId=${STATE.jellyfinData.serverId}` ); } else { window.location.href = `#/details?id=${itemId}&serverId=${STATE.jellyfinData.serverId}`; } }, }); }, /** * Creates a favorite button for an item * @param {string} itemId - Item ID * @returns {HTMLElement} Favorite button element */ createFavoriteButton(item) { const isFavorite = item.UserData && item.UserData.IsFavorite === true; const button = SlideUtils.createElement("button", { className: `favorite-button ${isFavorite ? "favorited" : ""}`, tabIndex: "0", onclick: async (e) => { e.preventDefault(); e.stopPropagation(); await ApiUtils.toggleFavorite(item.Id, button); }, }); return button; }, /** * Creates a trailer button * @param {string|Object} trailerInfo - Trailer URL string or object {id, url} * @returns {HTMLElement} Trailer button element */ createTrailerButton(trailerInfo) { const trailerText = LocalizationUtils.getLocalizedString('Trailer', 'Trailer'); let url = trailerInfo; let localTrailerId = null; if (typeof trailerInfo === 'object' && trailerInfo !== null) { url = trailerInfo.url; localTrailerId = trailerInfo.id; } return SlideUtils.createElement("button", { className: "detailButton trailer-button", innerHTML: `movie ${trailerText}`, tabIndex: "0", onclick: (e) => { e.preventDefault(); e.stopPropagation(); if (localTrailerId) { // Play local trailer using native player ApiUtils.playItem(localTrailerId); } else { SlideUtils.openVideoModal(url); } }, }); }, /** * Creates a placeholder slide for loading * @param {string} itemId - Item ID to load * @returns {HTMLElement} Placeholder slide element */ createLoadingPlaceholder(itemId) { const placeholder = SlideUtils.createElement("a", { className: "slide placeholder", "data-item-id": itemId, style: { display: "none", opacity: "0", transition: `opacity ${CONFIG.fadeTransitionDuration}ms ease-in-out`, }, }); const loadingIndicator = SlideUtils.createLoadingIndicator(); placeholder.appendChild(loadingIndicator); return placeholder; }, /** * Creates a slide for an item and adds it to the container * @param {string} itemId - Item ID * @returns {Promise} Created slide element */ async createSlideForItemId(itemId) { try { if (STATE.slideshow.createdSlides[itemId]) { return document.querySelector(`.slide[data-item-id="${itemId}"]`); } const container = SlideUtils.getOrCreateSlidesContainer(); const item = await ApiUtils.fetchItemDetails(itemId); // Pre-fetch local trailer URL if needed if (CONFIG.preferLocalTrailers && item.LocalTrailerCount > 0) { item.localTrailerUrl = await ApiUtils.fetchLocalTrailer(itemId); } // Pre-fetch theme video URL if needed if (CONFIG.preferLocalBackdrops) { item.themeVideoUrl = await ApiUtils.fetchThemeVideos(itemId); } const slideElement = this.createSlideElement( item, item.Type === "Movie" ? "Movie" : "TV Show" ); container.appendChild(slideElement); STATE.slideshow.createdSlides[itemId] = true; return slideElement; } catch (error) { console.error("Error creating slide for item:", error, itemId); return null; } }, }; /** * Manages slideshow functionality */ const SlideshowManager = { createPaginationDots() { let dotsContainer = document.querySelector(".dots-container"); if (!dotsContainer) { dotsContainer = document.createElement("div"); dotsContainer.className = "dots-container"; document.getElementById("slides-container").appendChild(dotsContainer); } const totalItems = STATE.slideshow.totalItems || 0; // Switch to counter style if too many items if (totalItems > CONFIG.maxPaginationDots) { const counter = document.createElement("span"); counter.className = "slide-counter"; counter.id = "slide-counter"; dotsContainer.appendChild(counter); } else { // Create dots for all items for (let i = 0; i < totalItems; i++) { const dot = document.createElement("span"); dot.className = "dot"; dot.setAttribute("data-index", i); dotsContainer.appendChild(dot); } } this.updateDots(); }, /** * Updates active dot based on current slide * Maps current slide to one of the 5 dots */ updateDots() { const currentIndex = STATE.slideshow.currentSlideIndex; const totalItems = STATE.slideshow.totalItems || 0; // Handle Large List Counter const counter = document.getElementById("slide-counter"); if (counter) { counter.textContent = `${currentIndex + 1} / ${totalItems}`; return; } // Handle Dots const container = SlideUtils.getOrCreateSlidesContainer(); const dots = container.querySelectorAll(".dot"); // Fallback if dots exist but totalItems matched counter mode if (dots.length === 0) return; dots.forEach((dot, index) => { if (index === currentIndex) { dot.classList.add("active"); } else { dot.classList.remove("active"); } }); }, /** * Updates current slide to the specified index * @param {number} index - Slide index to display */ async updateCurrentSlide(index) { if (STATE.slideshow.isTransitioning) { return; } STATE.slideshow.isTransitioning = true; let previousVisibleSlide; try { const container = SlideUtils.getOrCreateSlidesContainer(); const totalItems = STATE.slideshow.totalItems; index = Math.max(0, Math.min(index, totalItems - 1)); const currentItemId = STATE.slideshow.itemIds[index]; let currentSlide = document.querySelector( `.slide[data-item-id="${currentItemId}"]` ); if (!currentSlide) { currentSlide = await SlideCreator.createSlideForItemId(currentItemId); this.upgradeSlideImageQuality(currentSlide); if (!currentSlide) { console.error(`Failed to create slide for item ${currentItemId}`); STATE.slideshow.isTransitioning = false; setTimeout(() => this.nextSlide(), 500); return; } } previousVisibleSlide = container.querySelector(".slide.active"); if (previousVisibleSlide) { previousVisibleSlide.classList.remove("active"); } currentSlide.classList.add("active"); // Manage Video Playback: Stop others, Play current // 1. Stop all other YouTube players and local video elements, release connections setTimeout(() => { if (STATE.slideshow.videoPlayers) { Object.keys(STATE.slideshow.videoPlayers).forEach(id => { if (id !== currentItemId) { const p = STATE.slideshow.videoPlayers[id]; if (!p) return; // YouTube player if (typeof p.pauseVideo === 'function') { p.pauseVideo(); } // HTML5