From 1f17792bc63909dfd23c4030120c9e11ebdc6fe8 Mon Sep 17 00:00:00 2001 From: rlauuzo <46294892+rlauuzo@users.noreply.github.com> Date: Tue, 3 Sep 2024 22:55:06 +0200 Subject: [PATCH] Add user settings to configure skip behavior on client site (alternative to #261) (#263) * Update inject.js * Update inject.js * Update inject.js * Update inject.js * Update inject.js * Update inject.js * Update inject.js * update Update inject.js --------- Co-authored-by: rlauu <46294892+rlauu@users.noreply.github.com> --- .../Configuration/inject.js | 129 ++++++++++++++++-- 1 file changed, 115 insertions(+), 14 deletions(-) diff --git a/ConfusedPolarBear.Plugin.IntroSkipper/Configuration/inject.js b/ConfusedPolarBear.Plugin.IntroSkipper/Configuration/inject.js index 335f68d..efd6363 100644 --- a/ConfusedPolarBear.Plugin.IntroSkipper/Configuration/inject.js +++ b/ConfusedPolarBear.Plugin.IntroSkipper/Configuration/inject.js @@ -3,14 +3,23 @@ const introSkipper = { d: msg => console.debug("[intro skipper] ", msg), setup() { this.initializeState(); + this.initializeObserver(); + this.currentOption = localStorage.getItem('introskipperOption') || 'Show Button'; document.addEventListener("viewshow", this.viewShow.bind(this)); window.fetch = this.fetchWrapper.bind(this); this.videoPositionChanged = this.videoPositionChanged.bind(this); + this.handleEscapeKey = this.handleEscapeKey.bind(this); this.d("Registered hooks"); }, initializeState() { Object.assign(this, { allowEnter: true, skipSegments: {}, videoPlayer: null, skipButton: null, osdElement: null, skipperData: null, currentEpisodeId: null, injectMetadata: false }); }, + initializeObserver() { + this.observer = new MutationObserver(mutations => { + const actionSheet = mutations[mutations.length - 1].target.querySelector('.actionSheet'); + if (actionSheet && !actionSheet.querySelector(`[data-id="${'introskipperMenu'}"]`)) this.injectIntroSkipperOptions(actionSheet); + }); + }, /** Wrapper around fetch() that retrieves skip segments for the currently playing item or metadata. */ async fetchWrapper(resource, options) { const response = await this.originalFetch(resource, options); @@ -60,8 +69,12 @@ const introSkipper = { this.d("Hooking video timeupdate"); this.videoPlayer.addEventListener("timeupdate", this.videoPositionChanged); this.osdElement = document.querySelector("div.videoOsdBottom") + this.observer.observe(document.body, { childList: true, subtree: false }); } - } + } + else { + this.observer.disconnect(); + } }, /** * Injects the CSS used by the skip intro button. @@ -167,7 +180,7 @@ const introSkipper = { /** Get the currently playing skippable segment. */ getCurrentSegment(position) { for (const [key, segment] of Object.entries(this.skipSegments)) { - if ((position > segment.ShowSkipPromptAt && position < segment.HideSkipPromptAt - 1) || + if ((position > segment.ShowSkipPromptAt && position < segment.HideSkipPromptAt - 1) || (this.osdVisible() && position > segment.IntroStart && position < segment.IntroEnd - 1)) { segment.SegmentType = key; return segment; @@ -190,18 +203,22 @@ const introSkipper = { if (!this.skipButton) return; const embyButton = this.skipButton.querySelector(".emby-button"); const segmentType = this.getCurrentSegment(this.videoPlayer.currentTime).SegmentType; - if (segmentType === "None") { - if (!this.skipButton.classList.contains('show')) return; - this.skipButton.classList.remove('show'); - embyButton.addEventListener("transitionend", () => { - this.skipButton.classList.add("hide"); - this.allowEnter = true; - if (this.osdVisible()) { - this.osdElement.querySelector('button.btnPause').focus(); - } else { - embyButton.originalBlur(); - } - }, { once: true }); + if (segmentType === "None" || this.currentOption === "Off" || !this.allowEnter) { + if (this.skipButton.classList.contains('show')) { + this.skipButton.classList.remove('show'); + embyButton.addEventListener("transitionend", () => { + this.skipButton.classList.add("hide"); + if (this.osdVisible()) { + this.osdElement.querySelector('button.btnPause').focus(); + } else { + embyButton.originalBlur(); + } + }, { once: true }); + } + return; + } + if (this.currentOption === "Automatically Skip" || (this.currentOption === "Button w/ auto PiP" && document.pictureInPictureElement)) { + this.doSkip(); return; } this.skipButton.querySelector("#btnSkipSegmentText").textContent = this.skipButton.dataset[segmentType]; @@ -228,10 +245,88 @@ const introSkipper = { } this.d(`Skipping ${segment.SegmentType}`); this.allowEnter = false; + const seekedHandler = () => { + this.videoPlayer.removeEventListener('seeked', seekedHandler); + setTimeout(() => { + this.allowEnter = true; + }, 500); + }; + this.videoPlayer.addEventListener('seeked', seekedHandler); this.videoPlayer.currentTime = segment.SegmentType === "Credits" && this.videoPlayer.duration - segment.IntroEnd < 3 ? this.videoPlayer.duration + 10 : segment.IntroEnd; }, + createButton(ref, id, innerHTML, clickHandler) { + const button = ref.cloneNode(true); + button.setAttribute('data-id', id); + button.innerHTML = innerHTML; + button.addEventListener('click', clickHandler); + return button; + }, + closeSubmenu(fullscreen) { + document.querySelector('.dialogContainer').remove(); + document.querySelector('.dialogBackdrop').remove() + document.dispatchEvent(new KeyboardEvent('keydown', { key: 'Control' })); + if (!fullscreen) return; + document.removeEventListener('keydown', this.handleEscapeKey); + document.querySelector('.btnVideoOsdSettings').focus(); + }, + openSubmenu(ref, menu) { + const options = ['Show Button', 'Button w/ auto PiP', 'Automatically Skip', 'Off']; + const submenu = menu.cloneNode(true); + const scroller = submenu.querySelector('.actionSheetScroller'); + scroller.innerHTML = ''; + options.forEach(option => { + if (option !== 'Button w/ auto PiP' || document.pictureInPictureEnabled) { + const button = this.createButton(ref, `introskipper-${option.toLowerCase().replace(' ', '-')}`, + `