YouTube PiP

Smart Picture-in-Picture mode with all YouTube controls and functions

Du musst eine Erweiterung wie Tampermonkey, Greasemonkey oder Violentmonkey installieren, um dieses Skript zu installieren.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

Sie müssten eine Skript Manager Erweiterung installieren damit sie dieses Skript installieren können

(Ich habe schon ein Skript Manager, Lass mich es installieren!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         YouTube PiP
// @namespace    https://github.com/dmitroderkach/youtube-pip
// @version      2.2.11
// @description  Smart Picture-in-Picture mode with all YouTube controls and functions
// @author       Dmytro Derkach
// @match        https://www.youtube.com/*
// @grant        none
// @homepageURL  https://github.com/dmitroderkach/youtube-pip
// @supportURL   https://github.com/dmitroderkach/youtube-pip/issues
// @license      MIT
// ==/UserScript==


!function(){"use strict";const e=5e3,t=0,i=500,n=480,r=270,o=600,s=400,a=225,l="https://youtu.be",d="list",c="t",u=-1,h=1,g="yt-action",p="yt-navigate",y="yt-activate-miniplayer",m="yt-activate-miniplayer-from-watch-action",b="LIKE",v="DISLIKE",w="INDIFFERENT",f="WEB_PAGE_TYPE_WATCH",P=2,M=3,E=4,S=5,W=2,k=new WeakMap,x=new WeakSet;function C(e){return(t,i,n)=>{!function(e,t,i){const n=k.get(e)??[];n[t]=i,k.set(e,n)}(t,n,e)}}function O(){return e=>{!function(e){x.add(e)}(e)}}var T=Object.getOwnPropertyDescriptor;const A=new Intl.DateTimeFormat("en-CA",{year:"numeric",month:"2-digit",day:"2-digit",hour:"2-digit",minute:"2-digit",second:"2-digit",hour12:!1});const D=class e{constructor(e){this.scope=e}static getInstance(t){e.storageListenerAdded||(e.enabled=e.checkDebugFlag(),window.addEventListener("storage",()=>{e.enabled=e.checkDebugFlag()}),e.storageListenerAdded=!0);let i=e.instances.get(t);return i||(i=new e(t),e.instances.set(t,i)),i}static checkDebugFlag(){try{return"true"===localStorage.getItem("YOUTUBE_PIP_DEBUG")}catch{return!1}}static setGlobalMetadata(t){e.globalMetadata={...t}}styled(t,i,n,r){const o=function(e){const t=String(e.getMilliseconds()).padStart(3,"0");return A.format(e).replace(", ",":")+"."+t}(new Date),s=n.replace(/%/g,"%%");var a;const l=[`%c${o}%c [YouTube PiP]%c${a=this.scope,`[${a}]`}%c ${s}`,"color: #3b82f6; font-weight: 500;","color: #6366f1; font-weight: 600;","color: #f59e0b; font-weight: 500;",i];void 0!==r&&l.push(r),Object.keys(e.globalMetadata).length>0&&l.push(e.globalMetadata),t(...l)}log(t,i){e.enabled&&this.styled(console.log.bind(console),"color: #86efac;",t,i)}warn(t,i){e.enabled&&this.styled(console.warn.bind(console),"color: #fde047;",t,i)}error(e,t){this.styled(console.error.bind(console),"color: #f87171;",e,t)}debug(t,i){e.enabled&&this.styled(console.debug.bind(console),"color: #a1a1aa;",t,i)}};D.instances=new Map,D.enabled=!1,D.storageListenerAdded=!1,D.globalMetadata={};let I=D,$=class{create(e){return I.getInstance(e)}};$=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?T(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O()],$);class z extends Error{constructor(e,t){super(e),this.cause=t,Error.captureStackTrace&&Error.captureStackTrace(this,this.constructor)}}class F extends z{constructor(e,t){super(e,t),this.name="AppRuntimeError"}}class N{constructor(){this.bindings=new Map}bind(e){return{to:t=>{this.bindings.set(e,{token:e,implementation:t,scope:"singleton"})},toSelf:()=>{if("function"!=typeof e)throw new F("toSelf() requires a class constructor as token");this.bindings.set(e,{token:e,implementation:e,scope:"singleton"})},toTransient:t=>{this.bindings.set(e,{token:e,implementation:t,scope:"transient"})},toInstance:t=>{this.bindings.set(e,{token:e,implementation:()=>t,scope:"singleton",instance:t})}}}get(e,t=new Set){const i=this.tokenName(e),n=this.bindings.get(e);if(!n)throw new F(`No binding for ${i}`);if(void 0!==n.instance)return n.instance;if(t.has(e)){const i=[...t,e].map(e=>this.tokenName(e)).join(" → ");throw new F(`Circular dependency detected: ${i}`)}t.add(e);try{const i=n.implementation;if(r=i,!x.has(r))throw new F(`${this.tokenName(e)} must be decorated with @injectable()`);const o=function(e){return k.get(e)}(i)??[],s=i.length;for(let t=0;t<s;t++)if(void 0===o[t])throw new F(`${this.tokenName(e)}: constructor parameter at index ${t} must be decorated with @inject(token)`);const a=new i(...o.map(e=>this.get(e,t)));return"singleton"===n.scope&&(n.instance=a),a}finally{t.delete(e)}var r}tokenName(e){return"function"==typeof e?e.name||"anonymous":String(e)}unbind(e){this.bindings.delete(e)}}const L=I.getInstance("DOMUtils");class H{static createPlaceholder(e){return document.createComment(e)}static insertPlaceholderBefore(e,t){const i=e.parentNode;return i?(i.insertBefore(t,e),L.debug("Placeholder inserted"),!0):(L.warn("insertPlaceholderBefore: element has no parent"),!1)}static restoreElementFromPlaceholder(e,t){const i=t.parentNode;i?(i.insertBefore(e,t),t.remove(),L.debug("Element restored from placeholder")):L.warn("restoreElementFromPlaceholder: placeholder has no parent")}static copyAttributes(e,t){if(e&&t)try{Array.from(e.attributes).forEach(e=>{try{t.setAttribute(e.nodeName,e.nodeValue||"")}catch(i){L.warn(`Failed to copy attribute ${e.nodeName}:`,i)}}),L.debug("Attributes copied successfully")}catch(i){L.error("Error copying attributes:",i)}else L.warn("copyAttributes: source or target is null")}static copyViaTextarea(e,t){const i=e.createElement("textarea");i.value=t,i.style.cssText="position:fixed;left:-9999px;top:0;opacity:0;pointer-events:none;",e.body.appendChild(i),i.focus(),i.select();let n=!1;try{n=e.execCommand("copy")}catch{}return i.remove(),n}static unwrap(e){if(!e)return void L.warn("unwrap: wrapper is null");const t=e.parentNode;if(t)try{for(;e.firstChild;)t.insertBefore(e.firstChild,e);t.removeChild(e),L.debug("Element unwrapped successfully")}catch(i){L.error("Error unwrapping element:",i)}else L.warn("unwrap: Element has no parent node")}static waitForElementSelector(t,i=document,n=e,r){return new Promise((e,o)=>{const s=i.querySelector(t);if(s)return L.debug(`Element found immediately: ${t}`),e(s);L.debug(`Waiting for element: ${t}`);const a=new MutationObserver((n,r)=>{const o=i.querySelector(t);o&&(L.debug(`Element appeared: ${t}`),r.disconnect(),e(o))}),l=i===document?document.body:i;
/* v8 ignore next -- @preserve */l?(a.observe(l,{childList:!0,subtree:!0}),n>0&&setTimeout(()=>{a.disconnect(),o(new F(`Timeout: ${t} not found`))},n),0===n&&r&&!r.closed&&r.addEventListener("pagehide",()=>{a.disconnect(),o(new F("Wait aborted: target window closed"))},{once:!0})):o(new F(`Target element not found for selector: ${t}`))})}}const B="ytd-miniplayer",V="ytd-miniplayer-player-container",R=".ytdMiniplayerComponentHost.ytdMiniplayerComponentVisible",_="#movie_player",q="video",j="ytd-app",U="yt-draggable",Y=".ytp-popup.ytp-contextmenu",G=".ytp-contextmenu",X=".ytp-menuitem",K=".ytp-panel-menu > .ytp-menuitem",J=".yt-spec-button-shape-next[aria-expanded]",Q=".ytdMiniplayerComponentPlaylistPanel",Z=".ytp-progress-bar",ee="ytd-slim-metadata-toggle-button-renderer",te=".yt-spec-button-shape-next",ie=".yt-simple-endpoint",ne="button",re='style, link[rel="stylesheet"]',oe="ytd-notification-topbar-button-renderer";class se extends z{constructor(e,t){super(e,t),this.name="AppInitializationError"}}var ae=Object.getOwnPropertyDescriptor;let le=class{constructor(e){this.player=null,this.wasPlaying=!1,this.wasMiniPlayerActiveBeforePiP=!1,this.logger=e.create("PlayerManager")}async initialize(){try{const t=await H.waitForElementSelector(_,document,e);this.player=t,this.logger.debug("Player initialized")}catch(t){throw new se(`${_} element not found`,t)}}getPlayer(){return this.player}getPlayerState(e){return"function"!=typeof e.getPlayerState?(this.logger.error("getPlayerState method not found"),u):e.getPlayerState()}isPlaying(e){return this.getPlayerState(e)===h}savePlayingState(e){this.wasPlaying=this.isPlaying(e),this.logger.debug(`Player state saved: wasPlaying = ${this.wasPlaying}`)}setWasMiniPlayerActiveBeforePiP(e){this.wasMiniPlayerActiveBeforePiP=e}getWasMiniPlayerActiveBeforePiP(){return this.wasMiniPlayerActiveBeforePiP}restorePlayingState(e){if(this.wasPlaying)try{"function"==typeof e.playVideo?(e.playVideo(),this.logger.log("Playback restored after return to main window")):this.logger.error("player.playVideo method not found, cannot restore playback")}catch(t){this.logger.error("Error restoring playback:",t)}else this.logger.debug("No need to restore playing state")}getVideoDataFromPlayer(e){return"function"!=typeof e.getVideoData?null:e.getVideoData()||null}getVideoId(){const e=this.getVideoDataFromPlayer(this.getPlayer()),t=null==e?void 0:e.video_id;return t||(this.logger.error("Video ID not found, cannot navigate",{player:this.getPlayer()}),null)}getVideoData(){return this.getVideoDataFromPlayer(this.getPlayer())}getCurrentTime(){const e=this.getPlayer();if("function"!=typeof e.getCurrentTime)return 0;const t=e.getCurrentTime();return"number"!=typeof t||Number.isNaN(t)?0:Math.floor(t)}getPlayerSize(){const e=this.getPlayer();return"function"!=typeof e.getPlayerSize?null:e.getPlayerSize()}getDebugInfo(){const e=this.getPlayer();if("function"!=typeof e.getDebugText)return null;const t=e.getDebugText(!0);return"string"==typeof t&&t.length>0?t:null}async waitForMainPlayer(){try{const t=await H.waitForElementSelector(_,document,e);return this.player=t,this.logger.debug("Main player is ready"),t}catch(t){return this.logger.error("Error waiting for main player:",t),null}}async waitForMiniPlayer(){try{const t=await H.waitForElementSelector(R,document,e);return this.logger.debug("Miniplayer is ready"),t}catch(t){return this.logger.error("Error waiting for miniplayer:",t),null}}resetState(){this.wasPlaying=!1,this.logger.debug("Player state reset")}};var de,ce;le=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ae(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),(de=0,ce=C($),(e,t)=>ce(e,t,de))],le);var ue=Object.getOwnPropertyDescriptor;let he=class{constructor(e){this.app=null,this.notifyRenderer=null,this.logger=e.create("YtdAppProvider")}async initialize(){try{const i=await H.waitForElementSelector(j,document,e);this.app=i;try{this.notifyRenderer=await H.waitForElementSelector(oe,document,e)}catch(t){this.logger.warn(`${oe} not found, notification count will be omitted from title`,t),this.notifyRenderer=null}this.logger.debug("ytd-app initialized")}catch(i){throw new se(`${j} element not found`,i)}}getApp(){return this.app}getNotifyRenderer(){return this.notifyRenderer}};he=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ue(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),((e,t)=>(i,n)=>t(i,n,e))(0,C($))],he);var ge=Object.getOwnPropertyDescriptor;let pe=class{constructor(e){this.pipWindow=null,this.logger=e.create("PipWindowProvider")}setWindow(e){this.pipWindow=e,this.logger.debug("PipWindowProvider.setWindow",{hasWindow:null!==e})}getWindow(){return this.pipWindow}};pe=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ge(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),((e,t)=>(i,n)=>t(i,n,e))(0,C($))],pe);var ye=Object.getOwnPropertyDescriptor,me=(e,t)=>(i,n)=>t(i,n,e);let be=class{constructor(e,t,i){this.playerManager=t,this.ytdAppProvider=i,this.miniplayer=null,this.logger=e.create("MiniPlayerController")}async initialize(){try{const t=await H.waitForElementSelector(B,document,e);this.miniplayer=t,this.logger.debug("Miniplayer initialized")}catch(t){throw new se(`${B} element not found`,t)}}getMiniplayer(){return this.miniplayer}isVisible(){const e=!!document.querySelector(R);return this.logger.debug(`Mini player visible: ${e}`),e}activateMiniPlayer(){this.logger.debug("Activating mini player via YouTube API");const e=this.ytdAppProvider.getApp();if("function"==typeof e.fire)try{e.fire(g,{actionName:y,args:[!1],optionalAction:!1,returnValue:[void 0]}),this.logger.debug("Mini player activation event dispatched")}catch(t){this.logger.error("Error activating mini player:",t)}else this.logger.error("ytd-app fire method not found")}toggleMiniPlayer(){const e=this.ytdAppProvider.getApp();if("function"==typeof e.fire)try{if(e.miniplayerIsActive){this.logger.debug("Returning to full player via YouTube API");const t=this.playerManager.getVideoId();if(!t)return void this.logger.error("Video ID not found, cannot navigate to full player");e.fire(p,{endpoint:{watchEndpoint:{videoId:t}}}),this.logger.debug(`Navigation to full player dispatched for video ${t}`)}else this.logger.debug("Activating miniplayer via YouTube API"),e.fire(g,{actionName:m,args:null,optionalAction:!1,returnValue:[void 0]}),this.logger.debug("Miniplayer activation event dispatched")}catch(t){this.logger.error("Error toggling mini player:",t)}else this.logger.error("ytd-app fire method not found")}};be=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ye(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),me(0,C($)),me(1,C(le)),me(2,C(he))],be);var ve=Object.getOwnPropertyDescriptor,we=(e,t)=>(i,n)=>t(i,n,e);let fe=class{constructor(e,t){this.pipWindowProvider=t,this.pipWindow=null,this.logger=e.create("NavigationHandler")}initialize(){this.pipWindow=this.pipWindowProvider.getWindow(),this.setupClickHandler(),this.logger.debug("Navigation handler initialized")}setupClickHandler(){this.pipWindow?this.pipWindow.document.addEventListener("click",e=>{var t,i;const n=null==(t=e.target)?void 0:t.closest(ie);if((null==(i=e.target)?void 0:i.closest(ne))||!n)return;const r=n.href;if(r){this.logger.log("Navigation click detected");try{const t=new URL(r),i=Object.fromEntries(t.searchParams);e.preventDefault();const n={endpoint:{commandMetadata:{webCommandMetadata:{url:r,webPageType:f,rootVe:3832}},watchEndpoint:{videoId:i.v,playlistId:i.list||null,index:i.index?parseInt(i.index)-1:0,params:"OAE%3D",playerParams:i.pp}},entryTime:performance.now()};this.logger.log(`SPA navigation via ${p}: ${r}`),window.dispatchEvent(new PopStateEvent("popstate",{state:n}))}catch(o){this.logger.error("Error handling navigation:",o)}}else this.logger.warn("Navigation endpoint has no href")},!0):this.logger.error("PiP window not available for navigation handler")}cleanup(){this.pipWindow=null,this.logger.debug("Navigation handler cleaned up")}};fe=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ve(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),we(0,C($)),we(1,C(pe))],fe);var Pe=Object.getOwnPropertyDescriptor,Me=(e,t)=>(i,n)=>t(i,n,e);let Ee=class{constructor(e,t){this.playerManager=t,this.observer=null,this.logger=e.create("ResizeTracker")}start(e){"undefined"!=typeof ResizeObserver?(this.logger.debug("Starting resize tracking"),this.observer=new ResizeObserver(e=>{var t,i;for(const n of e){const e=n.contentRect.width;this.logger.debug(`New size: ${e}px`);const r=this.playerManager.getPlayer();"function"==typeof r.setInternalSize||"function"==typeof r.setSize||this.logger.warn("Player resize methods (setInternalSize, setSize) not found"),null==(t=r.setInternalSize)||t.call(r),null==(i=r.setSize)||i.call(r),r.dispatchEvent(new Event("resize",{bubbles:!0})),this.logger.debug("Player size updated")}}),this.observer.observe(e)):this.logger.error("ResizeObserver not available")}stop(){this.observer&&(this.observer.disconnect(),this.observer=null,this.logger.debug("Resize tracking stopped"))}};Ee=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?Pe(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),Me(0,C($)),Me(1,C(le))],Ee);var Se=Object.getOwnPropertyDescriptor,We=(e,t)=>(i,n)=>t(i,n,e);let ke=class{constructor(e,t){this.pipWindowProvider=t,this.observer=null,this.removalObserver=null,this.logger=e.create("MenuObserver")}async start(){const e=this.pipWindowProvider.getWindow();e&&await this.runObservation(e)}async runObservation(e){if(e.closed)return void this.logger.debug("PiP window already closed, skipping menu observation");let i;try{i=await H.waitForElementSelector(J,e.document,t,e)}catch(n){return void this.logger.warn("Wait for menu button aborted",n)}this.logger.debug("Starting menu observation"),this.observer=new MutationObserver(t=>{for(const n of t)if("aria-expanded"===n.attributeName){const t="true"===i.getAttribute("aria-expanded"),n=e.document.querySelector(Q),r=e.outerHeight;this.logger.debug(`Menu state changed: expanded = ${t}`),t?(r<o&&(e.resizeTo(e.outerWidth,o),this.logger.debug("PiP window expanded")),n&&(n.style.display="block")):n&&(n.style.display="none")}}),this.observer.observe(i,{attributes:!0,attributeFilter:["aria-expanded"]}),this.removalObserver=new MutationObserver(()=>{var t,n;i.isConnected||(null==(t=this.observer)||t.disconnect(),null==(n=this.removalObserver)||n.disconnect(),this.observer=null,this.removalObserver=null,this.logger.debug("Menu button removed from DOM, re-waiting"),this.runObservation(e))}),this.removalObserver.observe(e.document.body,{childList:!0,subtree:!0})}stop(){this.observer&&(this.observer.disconnect(),this.observer=null),this.removalObserver&&(this.removalObserver.disconnect(),this.removalObserver=null),this.logger.debug("Menu observation stopped")}};ke=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?Se(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),We(0,C($)),We(1,C(pe))],ke);var xe=(e=>(e.VIDEO_URL="video_url",e.URL_AT_TIME="url_at_time",e.EMBED="embed",e.DEBUG_INFO="debug_info",e))(xe||{});function Ce(e){const{videoId:t,playlistId:i,currentTime:n,title:r,copyType:o,embedSize:u}=e;switch(o){case xe.VIDEO_URL:return function(e,t){const i=`${l}/${e}`,n=t?`?${d}=${t}`:"";return n?`${i}${n}`:i}(t,i);case xe.URL_AT_TIME:return function(e,t,i){const n=t?`?${d}=${t}`:"";return`${l}/${e}${n}${i>0?n?`&${c}=${i}s`:`?${c}=${i}s`:""}`}(t,i,n);case xe.EMBED:return function(e,t,i,n){return`<iframe width="${(null==n?void 0:n.width)??s}" height="${(null==n?void 0:n.height)??a}" src="https://www.youtube.com/embed/${e}${t?`?${d}=${t}`:""}" title="${i.replace(/"/g,"&quot;")}" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe>`}(t,i,r,u);default:return""}}var Oe=Object.getOwnPropertyDescriptor,Te=(e,t)=>(i,n)=>t(i,n,e);let Ae=class{constructor(e,t,i,n){this.playerManager=t,this.ytdAppProvider=i,this.pipWindowProvider=n,this.visibilityObserver=null,this.pipWindow=null,this.contextMenu=null,this.contextMenuPlaceholder=null,this.visibilitySubscribers=new Set,this.handleCopyClick=e=>{var t;if(!this.pipWindow)return;const i=this.pipWindow.document,n=null==(t=e.target)?void 0:t.closest(X);if(!(null==n?void 0:n.parentElement))return void this.logger.debug("Copy click: not a menu item or no parent",{item:n});const r=i.querySelectorAll(K),o=Array.prototype.indexOf.call(r,n);if(-1===o)return void this.logger.warn("Copy click: menu item index not found");const s=this.getCopyTypeForIndex(o);if(!s)return void this.logger.debug("Copy click: not a copy action",{index:o});let a;switch(s){case xe.DEBUG_INFO:if(a=this.playerManager.getDebugInfo()??"",!a)return void this.logger.warn("Debug info not available, cannot copy");break;default:{const e=this.playerManager.getVideoData(),t=null==e?void 0:e.video_id;if(!t)return void this.logger.warn("Video ID not found, cannot copy");const i=(null==e?void 0:e.list)??null,n=this.playerManager.getCurrentTime(),r=(null==e?void 0:e.title)??"",o=s===xe.EMBED?this.playerManager.getPlayerSize():null;if(a=this.getCopyPayload({videoId:t,playlistId:i,currentTime:n,title:r,copyType:s,embedSize:o}),!a)return void this.logger.warn("Copy click: empty payload",{copyType:s});break}}H.copyViaTextarea(i,a)&&this.logger.debug(`Copied ${s} to clipboard`)},this.logger=e.create("ContextMenuHandler")}subscribeContextMenu(e){return this.visibilitySubscribers.add(e),()=>{this.visibilitySubscribers.delete(e)}}notifyVisibility(e){this.visibilitySubscribers.forEach(t=>t(e))}async initialize(){this.pipWindow=this.pipWindowProvider.getWindow(),this.contextMenuPlaceholder=H.createPlaceholder("context_menu_placeholder");try{this.contextMenu=await H.waitForElementSelector(Y,document,t,this.pipWindow),this.logger.log("Context menu element found, starting visibility monitoring"),this.startMonitoring(),this.setupDismissalHandler(),this.setupCopyHandler()}catch(e){this.logger.warn("Error initializing context menu handler:",e)}}startMonitoring(){this.contextMenu&&this.pipWindow&&(this.visibilityObserver=new MutationObserver(()=>{var e;if(!this.contextMenu||!this.pipWindow)return;const t="none"!==this.contextMenu.style.display,i=this.contextMenu.parentNode!==this.pipWindow.document.body;t&&i?(this.logger.log("Context menu opened in main window. Intercepting..."),this.contextMenuPlaceholder&&H.insertPlaceholderBefore(this.contextMenu,this.contextMenuPlaceholder),this.pipWindow.document.body.appendChild(this.contextMenu),this.notifyVisibility(!0)):t||this.contextMenu.parentNode!==this.pipWindow.document.body||(this.notifyVisibility(!1),(null==(e=this.contextMenuPlaceholder)?void 0:e.parentNode)&&(this.logger.log("Context menu closed in PiP window. Returning to main..."),H.restoreElementFromPlaceholder(this.contextMenu,this.contextMenuPlaceholder),this.simulateMainContextMenu()))}),this.visibilityObserver.observe(this.contextMenu,{attributes:!0,attributeFilter:["style"]}),"none"!==this.contextMenu.style.display&&(this.contextMenuPlaceholder&&H.insertPlaceholderBefore(this.contextMenu,this.contextMenuPlaceholder),this.pipWindow.document.body.appendChild(this.contextMenu),this.notifyVisibility(!0)))}setupDismissalHandler(){if(!this.pipWindow)return;const e=e=>{var t;const i=this.pipWindow.document.querySelector(Y);i&&"none"!==i.style.display&&!(null==(t=e.target)?void 0:t.closest(G))&&(e.stopPropagation(),i.style.display="none",this.notifyVisibility(!1),this.logger.debug("Context menu dismissed"))};this.pipWindow.document.addEventListener("click",e,!0),this.pipWindow.document.addEventListener("contextmenu",e,!0)}setupCopyHandler(){this.pipWindow&&this.pipWindow.document.addEventListener("click",this.handleCopyClick,!0)}getCopyTypeForIndex(e){return e===P?xe.VIDEO_URL:e===M?xe.URL_AT_TIME:e===E?xe.EMBED:e===S?xe.DEBUG_INFO:null}getCopyPayload(e){return Ce(e)}simulateMainContextMenu(){const e=this.ytdAppProvider.getApp(),t=new MouseEvent("contextmenu",{bubbles:!0,cancelable:!0,clientX:0,clientY:0,button:W});e.dispatchEvent(t),this.logger.debug("Synthetic contextmenu event sent to main window")}stop(){var e;this.visibilityObserver&&(this.visibilityObserver.disconnect(),this.visibilityObserver=null),this.pipWindow&&this.pipWindow.document.removeEventListener("click",this.handleCopyClick,!0),this.contextMenu&&this.pipWindow&&this.contextMenu.parentNode===this.pipWindow.document.body&&(null==(e=this.contextMenuPlaceholder)?void 0:e.parentNode)&&(this.logger.log("Returning context menu to main window"),H.restoreElementFromPlaceholder(this.contextMenu,this.contextMenuPlaceholder),this.simulateMainContextMenu()),this.pipWindow=null,this.logger.debug("Context menu handler stopped")}};Ae=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?Oe(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),Te(0,C($)),Te(1,C(le)),Te(2,C(he)),Te(3,C(pe))],Ae);var De=Object.getOwnPropertyDescriptor,Ie=(e,t)=>(i,n)=>t(i,n,e);let $e=class{constructor(e,t,i){this.playerManager=t,this.pipWindowProvider=i,this.pipWindow=null,this.logger=e.create("SeekHandler")}initialize(){this.pipWindow=this.pipWindowProvider.getWindow(),this.setupSeekHandler(),this.logger.debug("Seek handler initialized")}setupSeekHandler(){if(!this.pipWindow)return;const e=this.pipWindow.document;e.addEventListener("mousedown",t=>{var i;const n=null==(i=t.target)?void 0:i.closest(Z);if(!n)return;const r=this.playerManager.getPlayer();if("function"!=typeof r.getDuration)return void this.logger.error("player.getDuration method not found");if("function"!=typeof r.seekTo)return void this.logger.error("player.seekTo method not found");this.logger.debug("Progress bar clicked, initiating seek"),t.preventDefault(),t.stopPropagation();const o=e=>{var t,i;const o=n.getBoundingClientRect(),s=e.clientX-o.left,a=Math.max(0,Math.min(1,s/o.width)),l=null==(t=r.getDuration)?void 0:t.call(r);if(l){const e=a*l;null==(i=r.seekTo)||i.call(r,e,!0),this.logger.debug(`Seeking to ${e.toFixed(2)}s (${(100*a).toFixed(1)}%)`)}};o(t);const s=e=>o(e),a=()=>{e.removeEventListener("mousemove",s),e.removeEventListener("mouseup",a),this.logger.debug("Seek drag ended")};e.addEventListener("mousemove",s),e.addEventListener("mouseup",a)},!0)}cleanup(){this.pipWindow=null,this.logger.debug("Seek handler cleaned up")}};$e=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?De(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),Ie(0,C($)),Ie(1,C(le)),Ie(2,C(pe))],$e);var ze=Object.getOwnPropertyDescriptor,Fe=(e,t)=>(i,n)=>t(i,n,e);let Ne=class{constructor(e,t,i,n){this.playerManager=t,this.ytdAppProvider=i,this.pipWindowProvider=n,this.logger=e.create("YtActionSender")}sendLikeAction(e){if(!this.pipWindowProvider.getWindow())return;const t=this.playerManager.getVideoId();if(!t)return void this.logger.error("Video ID not found, cannot send like action");const i=this.ytdAppProvider.getApp();if("function"!=typeof i.resolveCommand)return void this.logger.error("Failed to find resolveCommand in main window");const n={likeEndpoint:{status:e,target:{videoId:t}}};try{i.resolveCommand(n),this.logger.log(`Sent ${e} for video ${t}`)}catch(r){this.logger.error("Error sending YouTube action:",r)}}};Ne=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ze(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),Fe(0,C($)),Fe(1,C(le)),Fe(2,C(he)),Fe(3,C(pe))],Ne);var Le=Object.getOwnPropertyDescriptor,He=(e,t)=>(i,n)=>t(i,n,e);let Be=class{constructor(e,t,i){this.pipWindowProvider=t,this.ytActionSender=i,this.pipWindow=null,this.logger=e.create("LikeButtonHandler")}initialize(){this.pipWindow=this.pipWindowProvider.getWindow(),this.setupClickHandler(),this.logger.debug("Like button handler initialized")}setupClickHandler(){this.pipWindow&&this.pipWindow.document.addEventListener("click",e=>{var t,i;const n=null==(t=e.target)?void 0:t.closest(ee);if(!n)return;const r=n.parentElement;if(!r)return;const o=r.childNodes[0]===n,s=r.childNodes[1]===n;if(!o&&!s)return;const a=null==(i=e.target)?void 0:i.closest(te);if(!a)return;const l="true"===a.getAttribute("aria-pressed"),d=l?w:o?b:v;this.logger.log(`${d} button clicked (currently pressed: ${l})`),this.ytActionSender.sendLikeAction(d)},!0)}cleanup(){this.pipWindow=null,this.logger.debug("Like button handler cleaned up")}};Be=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?Le(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),He(0,C($)),He(1,C(pe)),He(2,C(Ne))],Be);var Ve=Object.getOwnPropertyDescriptor,Re=(e,t)=>(i,n)=>t(i,n,e);let _e=class{constructor(e,t,i,n){this.playerManager=t,this.pipWindowProvider=i,this.contextMenuHandler=n,this.pipWindow=null,this.isContextMenuOpen=!1,this.unsubscribeContextMenu=null,this.onBodyClick=()=>this.returnFocusToPlayerIfNeeded(),this.onKeyUp=e=>{"Tab"!==e.key&&this.returnFocusToPlayerIfNeeded()},this.logger=e.create("DocumentFocusHandler")}returnFocusToPlayerIfNeeded(){if(!this.pipWindow||this.isContextMenuOpen)return;const e=this.pipWindow.document.activeElement,t=this.playerManager.getPlayer();e&&e!==t&&"function"==typeof t.focus&&(this.logger.debug("Returning focus to player"),setTimeout(()=>t.focus(),0))}initialize(){this.pipWindow=this.pipWindowProvider.getWindow(),this.pipWindow?(this.unsubscribeContextMenu=this.contextMenuHandler.subscribeContextMenu(e=>{this.isContextMenuOpen=e,e||this.returnFocusToPlayerIfNeeded()}),this.pipWindow.document.body.addEventListener("click",this.onBodyClick,!0),this.pipWindow.document.addEventListener("keyup",this.onKeyUp,!0),this.logger.debug("Document focus handler initialized")):this.logger.error("PiP window not available for document focus handler")}cleanup(){var e,t,i;null==(e=this.unsubscribeContextMenu)||e.call(this),this.unsubscribeContextMenu=null,(null==(i=null==(t=this.pipWindow)?void 0:t.document)?void 0:i.body)&&(this.pipWindow.document.body.removeEventListener("click",this.onBodyClick,!0),this.pipWindow.document.removeEventListener("keyup",this.onKeyUp,!0)),this.pipWindow=null,this.isContextMenuOpen=!1,this.logger.debug("Document focus handler cleaned up")}};_e=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?Ve(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),Re(0,C($)),Re(1,C(le)),Re(2,C(pe)),Re(3,C(Ae))],_e);var qe=Object.getOwnPropertyDescriptor,je=(e,t)=>(i,n)=>t(i,n,e);let Ue=class{constructor(e,t,i,n){this.pipWindowProvider=t,this.ytdAppProvider=i,this.playerManager=n,this.mutationObserver=null,this.logger=e.create("TitleSyncHandler")}initialize(){if(this.playerManager.getWasMiniPlayerActiveBeforePiP())return void this.logger.debug("Title sync skipped (e.g. PiP opened from mini player)");if(!this.pipWindowProvider.getWindow())return void this.logger.debug("PiP not open, skipping title sync init");const e=this.playerManager.getPlayer(),t=e.querySelector(q);if(!t)return void this.logger.warn("Video element not found inside player");const i=()=>{var t;const i=null==(t=e.getVideoData)?void 0:t.call(e);(null==i?void 0:i.title)&&this.setWindowsTitle(i.title)};i(),this.mutationObserver=new MutationObserver(()=>{i()}),this.mutationObserver.observe(t,{attributes:!0,attributeFilter:["src"]});const n=this.ytdAppProvider.getNotifyRenderer();n&&this.mutationObserver.observe(n,{childList:!0,subtree:!0}),this.logger.debug("Title sync observing video src and notify renderer subtree")}setWindowsTitle(e){const t=this.pipWindowProvider.getWindow();if(!t)return;const i=this.ytdAppProvider.getNotifyRenderer(),n=`${(null==i?void 0:i.showNotificationCount)?`(${i.showNotificationCount}) `:""}${e} - YouTube`;document.title=n,t.document.title=n,this.logger.debug(`Title synced: ${e}`)}cleanup(){this.mutationObserver&&(this.mutationObserver.disconnect(),this.mutationObserver=null,this.logger.debug("Title sync observer disconnected"))}};Ue=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?qe(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),je(0,C($)),je(1,C(pe)),je(2,C(he)),je(3,C(le))],Ue);var Ye=Object.getOwnPropertyDescriptor,Ge=(e,t)=>(i,n)=>t(i,n,e);let Xe=class{constructor(e,t,i,n,r,o,s,a){this.resizeTracker=e,this.menuObserver=t,this.contextMenuHandler=i,this.seekHandler=n,this.likeButtonHandler=r,this.navigationHandler=o,this.documentFocusHandler=s,this.titleSyncHandler=a}async initialize(e){return this.navigationHandler.initialize(),this.resizeTracker.start(e),this.menuObserver.start(),this.contextMenuHandler.initialize(),this.seekHandler.initialize(),this.likeButtonHandler.initialize(),this.documentFocusHandler.initialize(),this.titleSyncHandler.initialize(),()=>{this.titleSyncHandler.cleanup(),this.documentFocusHandler.cleanup(),this.seekHandler.cleanup(),this.likeButtonHandler.cleanup(),this.contextMenuHandler.stop(),this.menuObserver.stop(),this.resizeTracker.stop(),this.navigationHandler.cleanup()}}};Xe=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?Ye(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),Ge(0,C(Ee)),Ge(1,C(ke)),Ge(2,C(Ae)),Ge(3,C($e)),Ge(4,C(Be)),Ge(5,C(fe)),Ge(6,C(_e)),Ge(7,C(Ue))],Xe);const Ke=I.getInstance("StyleUtils");class Je{static copyStyles(e,t){try{const i=e.querySelectorAll(re);Ke.debug(`Copying ${i.length} style elements`),i.forEach(e=>{try{t.head.appendChild(e.cloneNode(!0))}catch(i){Ke.warn("Failed to copy style node:",i)}}),Ke.debug("Styles copied successfully")}catch(i){Ke.error("Error copying styles:",i)}}static injectCSSFixes(e){try{const t=e.createElement("style");t.textContent="/**\n * CSS styles for PiP window fixes and customizations\n */\n\nbody {\n  overflow: auto !important;\n}\n\nytd-miniplayer {\n  left: 0 !important;\n  right: 0 !important;\n  top: 0 !important;\n  bottom: 0 !important;\n  width: 100% !important;\n  height: 100% !important;\n  max-height: 100% !important;\n  display: flex !important;\n}\n\nytd-miniplayer-player-container {\n  width: 100% !important;\n  height: 100% !important;\n}\n\n.html5-video-container {\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n}\n\n.html5-video-container .html5-main-video {\n  width: 100% !important;\n  height: 100% !important;\n  object-fit: contain;\n}\n\nvideo {\n  left: 0 !important;\n  top: 0 !important;\n  right: 0 !important;\n  bottom: 0 !important;\n}\n\n.ytDraggableComponentHost.ytdMiniplayerComponentDraggable {\n  pointer-events: auto !important;\n}\n\n.ytp-miniplayer-expand-watch-page-button,\n.ytp-miniplayer-close-button,\n#header-contents,\n.ytdMiniplayerComponentResizers,\n/* TODO: remove when context menu interception is implemented */\n.dropdown-trigger {\n  display: none !important;\n}\n\n/* Stretch progress bar to full width */\n.ytp-progress-bar-container,\n.ytp-progress-bar {\n  width: 100% !important;\n}\n\n.ytp-prev-button,\n.ytp-next-button {\n  display: inline-flex !important;\n}\n",e.head.appendChild(t),Ke.debug("CSS fixes injected successfully")}catch(t){Ke.error("Error injecting CSS fixes:",t)}}}const Qe=I.getInstance("AsyncLock");class Ze{constructor(){this.locked=!1,this.queue=[]}async withLock(e){Qe.debug("withLock: waiting for lock"),await this.acquire(),Qe.debug("withLock: lock acquired, running fn");try{return await e()}finally{this.release(),Qe.debug("withLock: lock released")}}acquire(){return this.locked?(Qe.debug("acquire: lock busy, joining queue",{queueSize:this.queue.length}),new Promise(e=>{this.queue.push(()=>{this.locked=!0,Qe.debug("acquire: woke from queue"),e()})})):(this.locked=!0,Qe.debug("acquire: lock taken immediately"),Promise.resolve())}release(){if(this.queue.length>0){const e=this.queue.shift();Qe.debug("release: passing lock to next in queue",{remaining:this.queue.length}),e()}else this.locked=!1,Qe.debug("release: lock freed")}}class et extends z{constructor(e,t){super(e,t),this.name="PiPError"}}class tt extends z{constructor(e,t){super(e,t),this.name="PiPCriticalError"}}var it=Object.getOwnPropertyDescriptor,nt=(e,t)=>(i,n)=>t(i,n,e);let rt=class{constructor(e,t,i,n,r,o){this.miniPlayerController=t,this.playerManager=i,this.ytdAppProvider=n,this.pipWindowProvider=r,this.pipWindowHandlers=o,this.miniPlayerContainer=null,this.placeholder=null,this.onBeforeReturn=null,this.asyncLock=new Ze,this.close=async()=>this.asyncLock.withLock(()=>this.returnPlayerToMain()).catch(e=>{this.logger.error("Unhandled error in returnPlayerToMain:",e)}),this.logger=e.create("PiPManager")}isOpen(){return null!==this.pipWindowProvider.getWindow()}getWindow(){return this.pipWindowProvider.getWindow()}open(){return this.asyncLock.withLock(async()=>{if(this.isOpen())this.logger.warn("PiP window already open");else{this.logger.log("Opening PiP window");try{await this.movePlayerToPIP();if(this.pipWindowProvider.getWindow()){const e=await this.pipWindowHandlers.initialize(this.miniPlayerController.getMiniplayer());"function"==typeof e&&(this.onBeforeReturn=e)}}catch(e){if(e instanceof tt)throw e;throw new et("Error opening PiP",e)}}})}async movePlayerToPIP(){const e=this.miniPlayerController.getMiniplayer();this.playerManager.setWasMiniPlayerActiveBeforePiP(this.miniPlayerController.isVisible()),this.playerManager.getWasMiniPlayerActiveBeforePiP()||(this.miniPlayerController.toggleMiniPlayer(),await H.waitForElementSelector(V),this.logger.debug("Mini player container ready"));const t=e.offsetWidth||n,o=e.offsetHeight||r;this.logger.debug(`Requesting PiP window: ${t}x${o}`);const s=window.documentPictureInPicture;if(!s)throw new et("Document Picture-in-Picture API not available");const a=this.ytdAppProvider.getApp();if(this.miniPlayerContainer=document.querySelector(V),!this.miniPlayerContainer)throw new et("miniplayer-container element not found");const l=await s.requestWindow({width:t,height:o});this.pipWindowProvider.setWindow(l),setTimeout(()=>{this.asyncLock.withLock(async()=>{const e=this.pipWindowProvider.getWindow();if(null==e?void 0:e.closed)return this.logger.warn("phantom window detected, closing"),this.pipWindowProvider.setWindow(null),void this.close()})},i),l.addEventListener("pagehide",this.close),this.logger.log("PiP window opened");const d=l.document;H.copyAttributes(document.documentElement,d.documentElement),H.copyAttributes(document.body,d.body),Je.copyStyles(document,d),Je.injectCSSFixes(d);const c=d.createElement(j);H.copyAttributes(a,c),d.body.appendChild(c),this.placeholder=H.createPlaceholder("mini_player_placeholder"),H.insertPlaceholderBefore(e,this.placeholder),c.appendChild(e);const u=d.querySelector(U);if(!u)throw new tt("yt-draggable element not found");u.prepend(this.miniPlayerContainer),H.unwrap(u)}async returnPlayerToMain(){if(this.logger.log("Returning player to main window"),this.placeholder&&this.miniPlayerContainer){if(this.onBeforeReturn){try{await this.onBeforeReturn()}catch(e){this.logger.error("Error in onBeforeReturn:",e)}this.onBeforeReturn=null}try{await this.movePlayerToMain()}catch(t){this.logger.error("Error returning player to main window:",t)}this.pipWindowProvider.setWindow(null)}else this.logger.warn("Placeholder or miniPlayerContainer not found",{placeholder:this.placeholder,miniPlayerContainer:this.miniPlayerContainer})}async movePlayerToMain(){const e=this.miniPlayerController.getMiniplayer();if(!this.placeholder||!this.miniPlayerContainer)return;const t=this.playerManager.getPlayer();this.playerManager.savePlayingState(t),H.restoreElementFromPlaceholder(e,this.placeholder),this.placeholder=null;const i=document.querySelector(U);if(!i)throw new tt("yt-draggable element not found");i.prepend(this.miniPlayerContainer),this.playerManager.getWasMiniPlayerActiveBeforePiP()||(this.miniPlayerController.toggleMiniPlayer(),await this.playerManager.waitForMainPlayer());const n=document.querySelector(V);n&&n.remove(),await new Promise(e=>{setTimeout(async()=>{this.playerManager.restorePlayingState(t),this.playerManager.getWasMiniPlayerActiveBeforePiP()&&(this.miniPlayerController.activateMiniPlayer(),await this.playerManager.waitForMiniPlayer().catch(()=>{})),e()})})}};rt=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?it(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),nt(0,C($)),nt(1,C(be)),nt(2,C(le)),nt(3,C(he)),nt(4,C(pe)),nt(5,C(Xe))],rt);var ot=Object.getOwnPropertyDescriptor,st=(e,t)=>(i,n)=>t(i,n,e);let at=class{constructor(e,t){this.pipManager=t,this.logger=e.create("MediaSessionHandler")}initialize(){"mediaSession"in navigator?(this.registerActionHandler(),this.logger.debug("Media session handler initialized")):this.logger.warn("Media Session API not available")}registerActionHandler(){if("mediaSession"in navigator)try{navigator.mediaSession.setActionHandler("enterpictureinpicture",()=>{this.logger.log("Media session enterpictureinpicture action triggered"),this.pipManager.open().catch(e=>{this.logger.error("Error opening PiP from media session:",e)})}),this.logger.debug("Media session action handler registered")}catch(e){throw new se("Error registering media session action handler",e)}}};function lt(){var e,t,i;try{const n=window.ytcfg;if(null==(i=null==(t=null==(e=null==n?void 0:n.data_)?void 0:e.INNERTUBE_CONTEXT)?void 0:t.client)?void 0:i.clientVersion)return n.data_.INNERTUBE_CONTEXT.client.clientVersion}catch{}return"unknown"}function dt(){try{const e=navigator.userAgent;if(!e)return"unknown";const t=e.match(/Chrome\/([\d.]+)/);if(t&&!e.includes("Edg/"))return`Chrome/${t[1]}`;const i=e.match(/Edg\/([\d.]+)/);if(i)return`Edge/${i[1]}`;const n=e.match(/Firefox\/([\d.]+)/);if(n)return`Firefox/${n[1]}`;const r=e.match(/Version\/([\d.]+).*Safari/);return r&&!e.includes("Chrome")?`Safari/${r[1]}`:e}catch{}return"unknown"}function ct(){const e={youtubeVersion:lt(),scriptVersion:"2.2.11",browserVersion:dt()},t=function(){var e;try{const t=window.ytcfg;return(null==(e=null==t?void 0:t.data_)?void 0:e.EXPERIMENT_FLAGS)?t.data_.EXPERIMENT_FLAGS:{}}catch{}return{}}();return Object.keys(t).length>0&&(e.youtubeFeatureFlags=t),e}at=((e,t,i,n)=>{for(var r,o=n>1?void 0:n?ot(t,i):t,s=e.length-1;s>=0;s--)(r=e[s])&&(o=r(o)||o);return o})([O(),st(0,C($)),st(1,C(rt))],at);const ut=I.getInstance("Main");async function ht(){I.setGlobalMetadata(ct());const e=function(){const e=new N;return e.bind($).toSelf(),e.bind(le).toSelf(),e.bind(he).toSelf(),e.bind(pe).toSelf(),e.bind(be).toSelf(),e.bind(fe).toSelf(),e.bind(Ee).toSelf(),e.bind(ke).toSelf(),e.bind(Ae).toSelf(),e.bind($e).toSelf(),e.bind(Ne).toSelf(),e.bind(Be).toSelf(),e.bind(_e).toSelf(),e.bind(Ue).toSelf(),e.bind(Xe).toSelf(),e.bind(rt).toSelf(),e.bind(at).toSelf(),e}();ut.log("Initializing YouTube PiP application");try{const t=e.get(he);await t.initialize();const i=e.get(le);await i.initialize();const n=e.get(be);await n.initialize();e.get(at).initialize(),ut.log("YouTube PiP application initialized")}catch(t){ut.error("YouTube PiP initialization failed",t)}}"loading"===document.readyState?document.addEventListener("DOMContentLoaded",()=>{ht()}):ht()}();
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"userscript.js","sources":["../src/constants/app.ts","../src/constants/copyPayload.ts","../src/constants/youtube.ts","../src/constants/ui.ts","../src/di/metadata.ts","../src/di/decorators.ts","../src/logger.ts","../src/errors/AppError.ts","../src/errors/AppRuntimeError.ts","../src/di/container.ts","../src/utils/DOMUtils.ts","../src/selectors.ts","../src/errors/AppInitializationError.ts","../src/core/PlayerManager.ts","../src/core/YtdAppProvider.ts","../src/core/PipWindowProvider.ts","../src/ui/MiniPlayerController.ts","../src/core/NavigationHandler.ts","../src/ui/ResizeTracker.ts","../src/ui/MenuObserver.ts","../src/types/app.ts","../src/utils/copyPayload.ts","../src/ui/ContextMenuHandler.ts","../src/handlers/SeekHandler.ts","../src/core/YtActionSender.ts","../src/handlers/LikeButtonHandler.ts","../src/handlers/DocumentFocusHandler.ts","../src/handlers/TitleSyncHandler.ts","../src/core/PiPWindowHandlers.ts","../src/styles.css?raw","../src/utils/StyleUtils.ts","../src/utils/AsyncLock.ts","../src/errors/PiPError.ts","../src/errors/PiPCriticalError.ts","../src/core/PiPManager.ts","../src/handlers/MediaSessionHandler.ts","../src/utils/VersionDetector.ts","../src/main.ts","../src/di/container-config.ts"],"sourcesContent":["/**\n * Application-level constants\n */\n\n/** LocalStorage flag for enabling debug logs */\nexport const DEBUG_FLAG = 'YOUTUBE_PIP_DEBUG';\n\n/** Timeout values (in milliseconds) */\nexport const TIMEOUTS = {\n  ACTIVE_ELEMENT_POLL: 100,\n  ELEMENT_WAIT: 5000,\n  ELEMENT_WAIT_INFINITE: 0,\n  MENU_RETRY_DELAY: 100,\n  PHANTOM_WINDOW_CHECK: 500,\n} as const;\n\n/** Retry counts */\nexport const RETRY_LIMITS = {\n  MINIPLAYER_SWITCH: 5,\n} as const;\n\n/** Default dimensions for PiP window */\nexport const DEFAULT_DIMENSIONS = {\n  PIP_WIDTH: 480,\n  PIP_HEIGHT: 270,\n  PIP_EXPANDED_HEIGHT: 600,\n} as const;\n\n/** Default dimensions for embed iframe (fallback when player size unknown) */\nexport const EMBED_IFRAME_DEFAULTS = {\n  WIDTH: 400,\n  HEIGHT: 225,\n} as const;\n","/**\n * Constants for context menu \"Copy\" payloads (URLs, iframe, query params).\n */\n\n/** Base URL for short YouTube links */\nexport const YOUTUBE_SHORT_BASE = 'https://youtu.be';\n\n/** Base URL for embed player */\nexport const YOUTUBE_EMBED_BASE = 'https://www.youtube.com/embed';\n\n/** Query param names for copy payload URLs */\nexport const COPY_PAYLOAD_QUERY = {\n  LIST: 'list',\n  TIME: 't',\n} as const;\n\n/** Escaped double quote for iframe title attribute */\nexport const IFRAME_TITLE_QUOT = '&quot;';\n\n/** iframe allow attribute value */\nexport const IFRAME_ALLOW =\n  'accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share';\n\n/** iframe referrerpolicy value */\nexport const IFRAME_REFERRER_POLICY = 'strict-origin-when-cross-origin';\n","/**\n * YouTube API constants\n */\n\n/** YouTube player states */\nexport const PLAYER_STATES = {\n  UNSTARTED: -1,\n  ENDED: 0,\n  PLAYING: 1,\n  PAUSED: 2,\n  BUFFERING: 3,\n  CUED: 5,\n} as const;\n\n/** YouTube event names */\nexport const YT_EVENTS = {\n  ACTION: 'yt-action',\n  NAVIGATE: 'yt-navigate',\n} as const;\n\n/** YouTube action names for fire() */\nexport const YT_ACTION_NAMES = {\n  ACTIVATE_MINIPLAYER: 'yt-activate-miniplayer',\n  ACTIVATE_MINIPLAYER_FROM_WATCH: 'yt-activate-miniplayer-from-watch-action',\n} as const;\n\n/** YouTube like/dislike action types */\nexport const YT_LIKE_ACTIONS = {\n  LIKE: 'LIKE',\n  DISLIKE: 'DISLIKE',\n  REMOVE: 'INDIFFERENT',\n} as const;\n\n/** YouTube action types (general-purpose, extensible) */\nexport const YT_ACTIONS = {\n  ...YT_LIKE_ACTIONS,\n} as const;\n\n/** YouTube web page types */\nexport const WEB_PAGE_TYPES = {\n  WATCH: 'WEB_PAGE_TYPE_WATCH',\n} as const;\n\n/** YouTube root visual element ID */\nexport const ROOT_VE = 3832;\n","/**\n * UI-related constants\n */\n\n/** Player context menu copy-item indices (.ytp-panel-menu .ytp-menuitem) */\nexport const COPY_MENU_INDICES = {\n  VIDEO_URL: 2,\n  URL_AT_TIME: 3,\n  EMBED: 4,\n  DEBUG_INFO: 5,\n} as const;\n\n/** Mouse button codes (MouseEvent.button) */\nexport const MOUSE_BUTTONS = {\n  PRIMARY: 0, // Left button\n  AUXILIARY: 1, // Middle button (wheel)\n  SECONDARY: 2, // Right button\n  FOURTH: 3, // Browser back\n  FIFTH: 4, // Browser forward\n} as const;\n","/**\n * Metadata storage for DI decorators (no reflect-metadata dependency)\n */\n\nimport type { ServiceId } from './types';\n\nconst paramMetadata = new WeakMap<object, ServiceId[]>();\nconst injectableClasses = new WeakSet<object>();\n\nexport function setParamMetadata(target: object, index: number, token: ServiceId): void {\n  const existing = paramMetadata.get(target) ?? [];\n  existing[index] = token;\n  paramMetadata.set(target, existing);\n}\n\nexport function getParamMetadata(target: object): ServiceId[] | undefined {\n  return paramMetadata.get(target);\n}\n\nexport function setInjectable(target: object): void {\n  injectableClasses.add(target);\n}\n\nexport function isInjectable(target: object): boolean {\n  return injectableClasses.has(target);\n}\n","import { setParamMetadata, setInjectable } from './metadata';\nimport type { ServiceId } from './types';\n\n/**\n * Marks a constructor parameter for injection.\n * Accepts Symbol/string token or class constructor (Inversify-style).\n */\nexport function inject(token: ServiceId) {\n  return (target: object, _key: string | symbol | undefined, index: number): void => {\n    setParamMetadata(target, index, token);\n  };\n}\n\n/**\n * Marks a class as injectable (can be resolved by container)\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport function injectable(): (target: any) => void {\n  return (target: object): void => {\n    setInjectable(target);\n  };\n}\n","import { DEBUG_FLAG } from './constants';\nimport { injectable } from './di/decorators';\n\n/**\n * Format date as YYYY-MM-DD:HH:mm:ss.SSS using Intl.DateTimeFormat\n */\nconst timestampFormatter = new Intl.DateTimeFormat('en-CA', {\n  year: 'numeric',\n  month: '2-digit',\n  day: '2-digit',\n  hour: '2-digit',\n  minute: '2-digit',\n  second: '2-digit',\n  hour12: false,\n});\n\nfunction formatTimestamp(date: Date): string {\n  // Intl.DateTimeFormat outputs: \"2026-01-27, 12:34:56\"\n  // Add milliseconds manually: \"2026-01-27:12:34:56.789\"\n  const ms = String(date.getMilliseconds()).padStart(3, '0');\n  return timestampFormatter.format(date).replace(', ', ':') + '.' + ms;\n}\n\nconst PREFIX_BRAND = '[YouTube PiP]';\nconst PREFIX_SCOPE = (scope: string) => `[${scope}]` as const;\n\n/** Timestamp YYYY-MM-DD:HH:mm:ss.SSS – blue */\nconst STYLE_TIMESTAMP = 'color: #3b82f6; font-weight: 500;';\n/** [YouTube PiP] – indigo, bold */\nconst STYLE_BRAND = 'color: #6366f1; font-weight: 600;';\n/** [scope] – amber */\nconst STYLE_SCOPE = 'color: #f59e0b; font-weight: 500;';\n/** Info (log) message – light green */\nconst STYLE_MSG_INFO = 'color: #86efac;';\n/** Warn message – light yellow */\nconst STYLE_MSG_WARN = 'color: #fde047;';\n/** Error message – light red */\nconst STYLE_MSG_ERROR = 'color: #f87171;';\n/** Debug message – muted gray */\nconst STYLE_MSG_DEBUG = 'color: #a1a1aa;';\n\n/**\n * Logger class for conditional console logging with module scope.\n * Only logs when DEBUG_FLAG is set in localStorage (except error, always logged).\n * Uses %c styling: [YouTube PiP] / [scope] / message in separate colors.\n * Message and optional metadata are passed to console as-is (objects stay expandable).\n */\nexport class Logger {\n  private static instances = new Map<string, Logger>();\n  private static enabled = false;\n  private static storageListenerAdded = false;\n  private static globalMetadata: Record<string, unknown> = {};\n\n  private readonly scope: string;\n\n  private constructor(scope: string) {\n    this.scope = scope;\n  }\n\n  /**\n   * Get logger instance for the given module scope.\n   */\n  public static getInstance(scope: string): Logger {\n    if (!Logger.storageListenerAdded) {\n      Logger.enabled = Logger.checkDebugFlag();\n      window.addEventListener('storage', () => {\n        Logger.enabled = Logger.checkDebugFlag();\n      });\n      Logger.storageListenerAdded = true;\n    }\n\n    let instance = Logger.instances.get(scope);\n    if (!instance) {\n      instance = new Logger(scope);\n      Logger.instances.set(scope, instance);\n    }\n    return instance;\n  }\n\n  private static checkDebugFlag(): boolean {\n    try {\n      return localStorage.getItem(DEBUG_FLAG) === 'true';\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Set global metadata that will be included in all log messages.\n   * Should be called before initializing the application.\n   */\n  public static setGlobalMetadata(metadata: Record<string, unknown>): void {\n    Logger.globalMetadata = { ...metadata };\n  }\n\n  private styled(\n    fn: (template: string, ...rest: unknown[]) => void,\n    msgStyle: string,\n    message: string,\n    meta?: unknown\n  ): void {\n    const ts = formatTimestamp(new Date());\n    const escaped = message.replace(/%/g, '%%');\n    const template = `%c${ts}%c ${PREFIX_BRAND}%c${PREFIX_SCOPE(this.scope)}%c ${escaped}`;\n\n    // Build arguments: template, styles, meta (if provided), global metadata (if exists)\n    const args: [string, ...unknown[]] = [\n      template,\n      STYLE_TIMESTAMP,\n      STYLE_BRAND,\n      STYLE_SCOPE,\n      msgStyle,\n    ];\n\n    // Add user-provided metadata first\n    if (meta !== undefined) {\n      args.push(meta);\n    }\n\n    // Add global metadata after user metadata (if exists)\n    if (Object.keys(Logger.globalMetadata).length > 0) {\n      args.push(Logger.globalMetadata);\n    }\n\n    fn(...args);\n  }\n\n  /**\n   * Log message if debug is enabled. Info level – light green. Optional metadata as second arg.\n   */\n  public log(message: string, meta?: unknown): void {\n    if (Logger.enabled) {\n      this.styled(console.log.bind(console), STYLE_MSG_INFO, message, meta);\n    }\n  }\n\n  /**\n   * Log warning if debug is enabled. Optional metadata as second arg.\n   */\n  public warn(message: string, meta?: unknown): void {\n    if (Logger.enabled) {\n      this.styled(console.warn.bind(console), STYLE_MSG_WARN, message, meta);\n    }\n  }\n\n  /**\n   * Log error (always logged, regardless of debug flag). Optional metadata as second arg.\n   */\n  public error(message: string, meta?: unknown): void {\n    this.styled(console.error.bind(console), STYLE_MSG_ERROR, message, meta);\n  }\n\n  /**\n   * Log debug message if debug is enabled. Optional metadata as second arg.\n   */\n  public debug(message: string, meta?: unknown): void {\n    if (Logger.enabled) {\n      this.styled(console.debug.bind(console), STYLE_MSG_DEBUG, message, meta);\n    }\n  }\n}\n\n/**\n * Factory for creating Logger instances by scope. Injected via DI (transient).\n */\n@injectable()\nexport class LoggerFactory {\n  public create(scope: string): Logger {\n    return Logger.getInstance(scope);\n  }\n}\n","/**\n * Abstract base class for all application errors.\n * Provides common functionality like stack trace capture.\n */\nexport abstract class AppError extends Error {\n  public readonly cause: unknown;\n\n  constructor(message: string, cause?: unknown) {\n    super(message);\n    this.cause = cause;\n\n    // Maintains proper stack trace for where our error was thrown (only available on V8)\n    if (Error.captureStackTrace) {\n      Error.captureStackTrace(this, this.constructor);\n    }\n  }\n}\n","import { AppError } from './AppError';\n\n/**\n * Error thrown for general runtime errors (non-initialization).\n */\nexport class AppRuntimeError extends AppError {\n  constructor(message: string, cause?: unknown) {\n    super(message, cause);\n    this.name = 'AppRuntimeError';\n  }\n}\n","import { getParamMetadata, isInjectable } from './metadata';\nimport type { Constructor, ServiceId } from './types';\nimport { AppRuntimeError } from '../errors/AppRuntimeError';\n\ntype Scope = 'singleton' | 'transient';\n\ninterface Binding {\n  token: ServiceId;\n  implementation: Constructor;\n  scope: Scope;\n  instance?: unknown;\n}\n\nexport class Container {\n  private readonly bindings = new Map<ServiceId, Binding>();\n\n  /** Bind by Symbol/string token or by class constructor */\n  public bind<_T>(token: ServiceId): BindingTo {\n    return {\n      to: (implementation: Constructor) => {\n        this.bindings.set(token, {\n          token,\n          implementation,\n          scope: 'singleton',\n        });\n      },\n      toSelf: (): void => {\n        if (typeof token !== 'function') {\n          throw new AppRuntimeError('toSelf() requires a class constructor as token');\n        }\n        this.bindings.set(token, {\n          token,\n          implementation: token as Constructor,\n          scope: 'singleton',\n        });\n      },\n      toTransient: (implementation: Constructor) => {\n        this.bindings.set(token, {\n          token,\n          implementation,\n          scope: 'transient',\n        });\n      },\n      toInstance: (instance: unknown) => {\n        this.bindings.set(token, {\n          token,\n          /* v8 ignore next -- factory never called, instance returned directly */\n          implementation: (() => instance) as unknown as Constructor,\n          scope: 'singleton',\n          instance,\n        });\n      },\n    };\n  }\n\n  public get<T>(token: ServiceId, resolutionStack = new Set<ServiceId>()): T {\n    const name = this.tokenName(token);\n    const binding = this.bindings.get(token);\n\n    if (!binding) {\n      throw new AppRuntimeError(`No binding for ${name}`);\n    }\n\n    if (binding.instance !== undefined) {\n      return binding.instance as T;\n    }\n\n    if (resolutionStack.has(token)) {\n      const chain = [...resolutionStack, token].map((t) => this.tokenName(t)).join(' → ');\n      throw new AppRuntimeError(`Circular dependency detected: ${chain}`);\n    }\n\n    resolutionStack.add(token);\n    try {\n      const Ctor = binding.implementation;\n      if (!isInjectable(Ctor)) {\n        throw new AppRuntimeError(`${this.tokenName(token)} must be decorated with @injectable()`);\n      }\n      const paramTokens = getParamMetadata(Ctor) ?? [];\n      const paramCount = (Ctor as { length: number }).length;\n      for (let i = 0; i < paramCount; i++) {\n        if (paramTokens[i] === undefined) {\n          throw new AppRuntimeError(\n            `${this.tokenName(token)}: constructor parameter at index ${i} must be decorated with @inject(token)`\n          );\n        }\n      }\n      const args = paramTokens.map((t) => this.get(t as ServiceId, resolutionStack));\n\n      const instance = new (Ctor as new (...args: unknown[]) => T)(...args);\n\n      if (binding.scope === 'singleton') {\n        binding.instance = instance;\n      }\n\n      return instance;\n    } finally {\n      resolutionStack.delete(token);\n    }\n  }\n\n  private tokenName(token: ServiceId): string {\n    return typeof token === 'function' ? token.name || 'anonymous' : String(token);\n  }\n\n  public unbind(token: ServiceId): void {\n    this.bindings.delete(token);\n  }\n}\n\ninterface BindingTo {\n  to(implementation: Constructor): void;\n  toSelf(): void;\n  toTransient(implementation: Constructor): void;\n  toInstance(instance: unknown): void;\n}\n","import { Logger } from '../logger';\nimport { TIMEOUTS } from '../constants';\nimport { AppRuntimeError } from '../errors/AppRuntimeError';\nimport type { DocumentOrElement, Nullable } from '../types/app';\n\nconst logger = Logger.getInstance('DOMUtils');\n\n/**\n * Utility class for DOM operations\n */\nexport class DOMUtils {\n  /**\n   * Create a comment node to use as a placeholder (e.g. before moving an element elsewhere).\n   */\n  public static createPlaceholder(commentText: string): Comment {\n    return document.createComment(commentText);\n  }\n\n  /**\n   * Insert placeholder before element. Element stays in DOM; caller typically moves it.\n   * @returns true if inserted, false if element has no parent\n   */\n  public static insertPlaceholderBefore(element: Node, placeholder: Comment): boolean {\n    const parent = element.parentNode;\n    if (!parent) {\n      logger.warn('insertPlaceholderBefore: element has no parent');\n      return false;\n    }\n    parent.insertBefore(placeholder, element);\n    logger.debug('Placeholder inserted');\n    return true;\n  }\n\n  /**\n   * Restore element in place of placeholder, then remove placeholder.\n   * Use after moving element back from another container.\n   */\n  public static restoreElementFromPlaceholder(element: Node, placeholder: Comment): void {\n    const parent = placeholder.parentNode;\n    if (!parent) {\n      logger.warn('restoreElementFromPlaceholder: placeholder has no parent');\n      return;\n    }\n    parent.insertBefore(element, placeholder);\n    placeholder.remove();\n    logger.debug('Element restored from placeholder');\n  }\n\n  /**\n   * Copy all attributes from source element to target element\n   */\n  public static copyAttributes(source: Nullable<Element>, target: Nullable<Element>): void {\n    if (!source || !target) {\n      logger.warn('copyAttributes: source or target is null');\n      return;\n    }\n\n    try {\n      Array.from(source.attributes).forEach((attr) => {\n        try {\n          target.setAttribute(attr.nodeName, attr.nodeValue || '');\n        } catch (e) {\n          logger.warn(`Failed to copy attribute ${attr.nodeName}:`, e);\n        }\n      });\n      logger.debug('Attributes copied successfully');\n    } catch (e) {\n      logger.error('Error copying attributes:', e);\n    }\n  }\n\n  /**\n   * Copy text to clipboard via temporary textarea and execCommand('copy').\n   * Use when Clipboard API is unavailable or copy must run in a specific document (e.g. PiP).\n   * @returns true if execCommand('copy') succeeded\n   */\n  public static copyViaTextarea(doc: Document, text: string): boolean {\n    const el = doc.createElement('textarea');\n    el.value = text;\n    el.style.cssText = 'position:fixed;left:-9999px;top:0;opacity:0;pointer-events:none;';\n    doc.body.appendChild(el);\n    el.focus();\n    el.select();\n    let ok = false;\n    try {\n      ok = doc.execCommand('copy');\n    } catch {\n      /* ignore */\n    }\n    el.remove();\n    return ok;\n  }\n\n  /**\n   * Unwrap element by moving all children to parent and removing wrapper\n   */\n  public static unwrap(wrapper: Nullable<Element>): void {\n    if (!wrapper) {\n      logger.warn('unwrap: wrapper is null');\n      return;\n    }\n\n    const parent = wrapper.parentNode;\n    if (!parent) {\n      logger.warn('unwrap: Element has no parent node');\n      return;\n    }\n\n    try {\n      // Move all child nodes before the wrapper\n      while (wrapper.firstChild) {\n        parent.insertBefore(wrapper.firstChild, wrapper);\n      }\n\n      // Remove the now-empty wrapper\n      parent.removeChild(wrapper);\n      logger.debug('Element unwrapped successfully');\n    } catch (e) {\n      logger.error('Error unwrapping element:', e);\n    }\n  }\n\n  /**\n   * Wait for element to appear in DOM using selector\n   * @param selector CSS selector\n   * @param parent Parent element to search in (default: document)\n   * @param timeout Timeout in milliseconds (0 = infinite)\n   * @param targetWindow When timeout is 0, disconnect observer on this window's pagehide (e.g. PiP window)\n   * @returns Promise that resolves with the found element\n   */\n  public static waitForElementSelector<T extends Element>(\n    selector: string,\n    parent: DocumentOrElement = document,\n    timeout: number = TIMEOUTS.ELEMENT_WAIT,\n    targetWindow?: Window | null\n  ): Promise<T> {\n    return new Promise((resolve, reject) => {\n      // Check if element already exists\n      const existing = parent.querySelector<T>(selector);\n      if (existing) {\n        logger.debug(`Element found immediately: ${selector}`);\n        return resolve(existing);\n      }\n\n      logger.debug(`Waiting for element: ${selector}`);\n\n      // Create observer to watch for element\n      const observer = new MutationObserver((_, obs) => {\n        const element = parent.querySelector<T>(selector);\n        if (element) {\n          logger.debug(`Element appeared: ${selector}`);\n          obs.disconnect();\n          resolve(element);\n        }\n      });\n\n      // Observe the parent for changes\n      /* v8 ignore next -- @preserve */\n      const target = parent === document ? document.body : parent;\n      if (!target) {\n        reject(new AppRuntimeError(`Target element not found for selector: ${selector}`));\n        return;\n      }\n\n      observer.observe(target, {\n        childList: true,\n        subtree: true,\n      });\n\n      // Set timeout if specified\n      if (timeout > 0) {\n        setTimeout(() => {\n          observer.disconnect();\n          reject(new AppRuntimeError(`Timeout: ${selector} not found`));\n        }, timeout);\n      }\n\n      // For infinite timeout: disconnect and reject when targetWindow fires pagehide (e.g. PiP closed)\n      if (timeout === 0 && targetWindow && !targetWindow.closed) {\n        targetWindow.addEventListener(\n          'pagehide',\n          () => {\n            observer.disconnect();\n            reject(new AppRuntimeError('Wait aborted: target window closed'));\n          },\n          { once: true }\n        );\n      }\n    });\n  }\n}\n","/**\n * DOM selectors – single source of truth\n */\n\nexport const SELECTORS = {\n  MINIPLAYER: 'ytd-miniplayer',\n  MINIPLAYER_CONTAINER: 'ytd-miniplayer-player-container',\n  MINIPLAYER_HOST: '.ytdMiniplayerComponentHost.ytdMiniplayerComponentVisible',\n  MOVIE_PLAYER: '#movie_player',\n  /** <video> element inside the player (query relative to player root) */\n  PLAYER_VIDEO: 'video',\n  YTD_APP: 'ytd-app',\n  YT_DRAGGABLE: 'yt-draggable',\n  CONTEXT_MENU: '.ytp-popup.ytp-contextmenu',\n  CONTEXT_MENU_CONTAINER: '.ytp-contextmenu',\n  MENU_ITEM: '.ytp-menuitem',\n  PANEL_MENU: '.ytp-panel-menu',\n  PANEL_MENU_ITEMS: '.ytp-panel-menu > .ytp-menuitem',\n  MENU_BUTTON: '.yt-spec-button-shape-next[aria-expanded]',\n  PLAYLIST_PANEL: '.ytdMiniplayerComponentPlaylistPanel',\n  PROGRESS_BAR: '.ytp-progress-bar',\n  LIKE_BUTTON: 'ytd-slim-metadata-toggle-button-renderer',\n  BUTTON_SHAPE: '.yt-spec-button-shape-next',\n  SIMPLE_ENDPOINT: '.yt-simple-endpoint',\n  BUTTON: 'button',\n  STYLESHEETS: 'style, link[rel=\"stylesheet\"]',\n  NOTIFICATION_TOPBAR_BUTTON_RENDERER: 'ytd-notification-topbar-button-renderer',\n} as const;\n\nexport type Selector = (typeof SELECTORS)[keyof typeof SELECTORS];\n","import { AppError } from './AppError';\n\n/**\n * Error thrown during application initialization.\n */\nexport class AppInitializationError extends AppError {\n  constructor(message: string, cause?: unknown) {\n    super(message, cause);\n    this.name = 'AppInitializationError';\n  }\n}\n","import { YouTubePlayer, VideoData, PlayerState, PlayerSize } from '../types/youtube';\nimport { PLAYER_STATES } from '../constants';\nimport { DOMUtils } from '../utils/DOMUtils';\nimport { TIMEOUTS } from '../constants';\nimport { SELECTORS } from '../selectors';\nimport type { Nullable } from '../types/app';\nimport type { Logger } from '../logger';\nimport { AppInitializationError } from '../errors/AppInitializationError';\nimport { LoggerFactory } from '../logger';\nimport { inject, injectable } from '../di';\n\n/**\n * Manages player state and operations.\n * Holds a reference to the player DOM element, initialized at app startup.\n * The reference persists when the player moves between main window and PiP.\n */\n@injectable()\nexport class PlayerManager {\n  private readonly logger: Logger;\n  private player: Nullable<YouTubePlayer> = null;\n  private wasPlaying: boolean = false;\n  private wasMiniPlayerActiveBeforePiP: boolean = false;\n\n  constructor(@inject(LoggerFactory) loggerFactory: LoggerFactory) {\n    this.logger = loggerFactory.create('PlayerManager');\n  }\n\n  /**\n   * Initialize player reference from main document. Call at app startup.\n   * Waits for movie_player to appear (e.g. on watch page).\n   * @throws Error if movie_player element not found\n   */\n  public async initialize(): Promise<void> {\n    try {\n      const element = await DOMUtils.waitForElementSelector<YouTubePlayer>(\n        SELECTORS.MOVIE_PLAYER,\n        document,\n        TIMEOUTS.ELEMENT_WAIT\n      );\n      this.player = element;\n      this.logger.debug('Player initialized');\n    } catch (cause) {\n      throw new AppInitializationError(`${SELECTORS.MOVIE_PLAYER} element not found`, cause);\n    }\n  }\n\n  /**\n   * Get the player element. Always defined after initialize().\n   */\n  public getPlayer(): YouTubePlayer {\n    return this.player!;\n  }\n\n  /**\n   * Get player state (playing/paused)\n   */\n  public getPlayerState(player: YouTubePlayer): PlayerState {\n    if (typeof player.getPlayerState !== 'function') {\n      this.logger.error('getPlayerState method not found');\n      return PLAYER_STATES.UNSTARTED;\n    }\n    return player.getPlayerState();\n  }\n\n  /**\n   * Check if player is currently playing\n   */\n  public isPlaying(player: YouTubePlayer): boolean {\n    return this.getPlayerState(player) === PLAYER_STATES.PLAYING;\n  }\n\n  /**\n   * Save current playing state\n   */\n  public savePlayingState(player: YouTubePlayer): void {\n    this.wasPlaying = this.isPlaying(player);\n    this.logger.debug(`Player state saved: wasPlaying = ${this.wasPlaying}`);\n  }\n\n  /**\n   * Set whether mini player was active when PiP was opened. Used by TitleSyncHandler to skip initial title sync.\n   */\n  public setWasMiniPlayerActiveBeforePiP(value: boolean): void {\n    this.wasMiniPlayerActiveBeforePiP = value;\n  }\n\n  /**\n   * Whether mini player was visible when PiP was opened.\n   */\n  public getWasMiniPlayerActiveBeforePiP(): boolean {\n    return this.wasMiniPlayerActiveBeforePiP;\n  }\n\n  /**\n   * Restore playing state if it was playing before\n   */\n  public restorePlayingState(player: YouTubePlayer): void {\n    if (!this.wasPlaying) {\n      this.logger.debug('No need to restore playing state');\n      return;\n    }\n\n    try {\n      if (typeof player.playVideo === 'function') {\n        player.playVideo();\n        this.logger.log('Playback restored after return to main window');\n      } else {\n        this.logger.error('player.playVideo method not found, cannot restore playback');\n      }\n    } catch (e) {\n      this.logger.error('Error restoring playback:', e);\n    }\n  }\n\n  private getVideoDataFromPlayer(player: YouTubePlayer): Nullable<VideoData> {\n    if (typeof player.getVideoData !== 'function') {\n      return null;\n    }\n    return player.getVideoData() || null;\n  }\n\n  /**\n   * Get video ID from player\n   */\n  public getVideoId(): Nullable<string> {\n    const videoData = this.getVideoDataFromPlayer(this.getPlayer());\n    const videoId = videoData?.video_id;\n\n    if (!videoId) {\n      this.logger.error('Video ID not found, cannot navigate', { player: this.getPlayer() });\n      return null;\n    }\n\n    return videoId;\n  }\n\n  /**\n   * Get video data (video_id, title, list) from player.\n   */\n  public getVideoData(): Nullable<VideoData> {\n    return this.getVideoDataFromPlayer(this.getPlayer());\n  }\n\n  /**\n   * Get current playback time in seconds.\n   */\n  public getCurrentTime(): number {\n    const player = this.getPlayer();\n    if (typeof player.getCurrentTime !== 'function') {\n      return 0;\n    }\n    const t = player.getCurrentTime();\n    if (typeof t === 'number' && !Number.isNaN(t)) {\n      return Math.floor(t);\n    }\n    return 0;\n  }\n\n  /**\n   * Get player size (width, height).\n   */\n  public getPlayerSize(): Nullable<PlayerSize> {\n    const player = this.getPlayer();\n    if (typeof player.getPlayerSize !== 'function') {\n      return null;\n    }\n    return player.getPlayerSize();\n  }\n\n  /**\n   * Get debug information string from player.\n   */\n  public getDebugInfo(): Nullable<string> {\n    const player = this.getPlayer();\n    if (typeof player.getDebugText !== 'function') {\n      return null;\n    }\n    const text = player.getDebugText(true);\n    return typeof text === 'string' && text.length > 0 ? text : null;\n  }\n\n  /**\n   * Wait for main player to be ready\n   */\n  public async waitForMainPlayer(): Promise<Nullable<Element>> {\n    try {\n      const player = await DOMUtils.waitForElementSelector(\n        SELECTORS.MOVIE_PLAYER,\n        document,\n        TIMEOUTS.ELEMENT_WAIT\n      );\n      this.player = player as YouTubePlayer;\n      this.logger.debug('Main player is ready');\n      return player;\n    } catch (e) {\n      this.logger.error('Error waiting for main player:', e);\n      return null;\n    }\n  }\n\n  /**\n   * Wait for miniplayer to be ready\n   */\n  public async waitForMiniPlayer(): Promise<Nullable<Element>> {\n    try {\n      const miniplayer = await DOMUtils.waitForElementSelector(\n        SELECTORS.MINIPLAYER_HOST,\n        document,\n        TIMEOUTS.ELEMENT_WAIT\n      );\n      this.logger.debug('Miniplayer is ready');\n      return miniplayer;\n    } catch (e) {\n      this.logger.error('Error waiting for miniplayer:', e);\n      return null;\n    }\n  }\n\n  /**\n   * Reset saved state\n   */\n  public resetState(): void {\n    this.wasPlaying = false;\n    this.logger.debug('Player state reset');\n  }\n}\n","import { SELECTORS } from '../selectors';\nimport { TIMEOUTS } from '../constants';\nimport { DOMUtils } from '../utils/DOMUtils';\nimport type { Nullable } from '../types/app';\nimport type { NotificationTopbarButtonRenderer, YouTubeAppElement } from '../types/youtube';\nimport { AppInitializationError } from '../errors/AppInitializationError';\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { inject, injectable } from '../di';\n\n/**\n * Provides the main document's ytd-app element.\n * Initialized at app startup; reference persists when moving between main window and PiP.\n */\n@injectable()\nexport class YtdAppProvider {\n  private readonly logger: Logger;\n  private app: Nullable<YouTubeAppElement> = null;\n  private notifyRenderer: Nullable<NotificationTopbarButtonRenderer> = null;\n\n  constructor(@inject(LoggerFactory) loggerFactory: LoggerFactory) {\n    this.logger = loggerFactory.create('YtdAppProvider');\n  }\n\n  /**\n   * Initialize with main document. Call at app startup.\n   * Waits for ytd-app element with standard timeout.\n   * @throws AppInitializationError if ytd-app element not found\n   */\n  public async initialize(): Promise<void> {\n    try {\n      const element = await DOMUtils.waitForElementSelector<YouTubeAppElement>(\n        SELECTORS.YTD_APP,\n        document,\n        TIMEOUTS.ELEMENT_WAIT\n      );\n      this.app = element;\n      try {\n        this.notifyRenderer =\n          await DOMUtils.waitForElementSelector<NotificationTopbarButtonRenderer>(\n            SELECTORS.NOTIFICATION_TOPBAR_BUTTON_RENDERER,\n            document,\n            TIMEOUTS.ELEMENT_WAIT\n          );\n      } catch (err) {\n        this.logger.warn(\n          `${SELECTORS.NOTIFICATION_TOPBAR_BUTTON_RENDERER} not found, notification count will be omitted from title`,\n          err\n        );\n        this.notifyRenderer = null;\n      }\n      this.logger.debug('ytd-app initialized');\n    } catch (cause) {\n      throw new AppInitializationError(`${SELECTORS.YTD_APP} element not found`, cause);\n    }\n  }\n\n  /**\n   * Get the ytd-app element (main document). Always defined after initialize().\n   */\n  public getApp(): YouTubeAppElement {\n    return this.app!;\n  }\n\n  /**\n   * Get the notification topbar button renderer. May be null if not yet rendered.\n   */\n  public getNotifyRenderer(): Nullable<NotificationTopbarButtonRenderer> {\n    return this.notifyRenderer;\n  }\n}\n","import type { Nullable } from '../types/app';\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { inject, injectable } from '../di';\n\n/**\n * Provides the PiP window reference.\n * Set by PiPManager when PiP opens, cleared when PiP closes.\n */\n@injectable()\nexport class PipWindowProvider {\n  private readonly logger: Logger;\n  private pipWindow: Nullable<Window> = null;\n\n  constructor(@inject(LoggerFactory) loggerFactory: LoggerFactory) {\n    this.logger = loggerFactory.create('PipWindowProvider');\n  }\n\n  /**\n   * Set the PiP window (called by PiPManager when PiP opens/closes).\n   */\n  public setWindow(pipWindow: Nullable<Window>): void {\n    this.pipWindow = pipWindow;\n    this.logger.debug('PipWindowProvider.setWindow', { hasWindow: pipWindow !== null });\n  }\n\n  /**\n   * Get the PiP window.\n   */\n  public getWindow(): Nullable<Window> {\n    return this.pipWindow;\n  }\n}\n","import type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { SELECTORS } from '../selectors';\nimport { TIMEOUTS, YT_EVENTS, YT_ACTION_NAMES } from '../constants';\nimport { DOMUtils } from '../utils/DOMUtils';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { YtdAppProvider } from '../core/YtdAppProvider';\nimport type { Nullable } from '../types/app';\nimport type { MiniPlayerElement } from '../types/youtube';\nimport { AppInitializationError } from '../errors/AppInitializationError';\nimport { inject, injectable } from '../di';\n\n/**\n * Controls mini player visibility and state using YouTube native actions.\n * Holds a reference to the miniplayer DOM element, initialized at app startup.\n */\n@injectable()\nexport class MiniPlayerController {\n  private readonly logger: Logger;\n  private miniplayer: Nullable<MiniPlayerElement> = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager,\n    @inject(YtdAppProvider) private readonly ytdAppProvider: YtdAppProvider\n  ) {\n    this.logger = loggerFactory.create('MiniPlayerController');\n  }\n\n  /**\n   * Initialize with main document. Call at app startup.\n   * Waits for ytd-miniplayer element with standard timeout.\n   * @throws AppInitializationError if miniplayer element not found\n   */\n  public async initialize(): Promise<void> {\n    try {\n      const element = await DOMUtils.waitForElementSelector<MiniPlayerElement>(\n        SELECTORS.MINIPLAYER,\n        document,\n        TIMEOUTS.ELEMENT_WAIT\n      );\n      this.miniplayer = element;\n      this.logger.debug('Miniplayer initialized');\n    } catch (cause) {\n      throw new AppInitializationError(`${SELECTORS.MINIPLAYER} element not found`, cause);\n    }\n  }\n\n  /**\n   * Get the miniplayer element reference. Always defined after initialize().\n   */\n  public getMiniplayer(): MiniPlayerElement {\n    return this.miniplayer!;\n  }\n  /**\n   * Check if mini player is currently visible\n   */\n  public isVisible(): boolean {\n    const isVisible = !!document.querySelector(SELECTORS.MINIPLAYER_HOST);\n    this.logger.debug(`Mini player visible: ${isVisible}`);\n    return isVisible;\n  }\n\n  /**\n   * Activate mini player using YouTube native action\n   * Uses YT_EVENTS.ACTION with YT_ACTION_NAMES.ACTIVATE_MINIPLAYER\n   */\n  public activateMiniPlayer(): void {\n    this.logger.debug('Activating mini player via YouTube API');\n\n    const ytdApp = this.ytdAppProvider.getApp();\n    if (typeof ytdApp.fire !== 'function') {\n      this.logger.error('ytd-app fire method not found');\n      return;\n    }\n\n    try {\n      ytdApp.fire(YT_EVENTS.ACTION, {\n        actionName: YT_ACTION_NAMES.ACTIVATE_MINIPLAYER,\n        args: [false],\n        optionalAction: false,\n        returnValue: [undefined],\n      });\n      this.logger.debug('Mini player activation event dispatched');\n    } catch (e) {\n      this.logger.error('Error activating mini player:', e);\n    }\n  }\n\n  /**\n   * Toggle mini player mode using YouTube native API\n   *\n   * If mini player is active, navigates back to full player using YT_EVENTS.NAVIGATE.\n   * Otherwise, activates mini player using YT_EVENTS.ACTION with YT_ACTION_NAMES.ACTIVATE_MINIPLAYER_FROM_WATCH.\n   *\n   * @returns void\n   */\n  public toggleMiniPlayer(): void {\n    const ytdApp = this.ytdAppProvider.getApp();\n    if (typeof ytdApp.fire !== 'function') {\n      this.logger.error('ytd-app fire method not found');\n      return;\n    }\n\n    try {\n      if (ytdApp.miniplayerIsActive) {\n        // Return to full player: use YT_EVENTS.NAVIGATE with watchEndpoint\n        this.logger.debug('Returning to full player via YouTube API');\n\n        // Get video ID from player using PlayerManager\n        const videoId = this.playerManager.getVideoId();\n        if (!videoId) {\n          this.logger.error('Video ID not found, cannot navigate to full player');\n          return;\n        }\n\n        ytdApp.fire(YT_EVENTS.NAVIGATE, {\n          endpoint: {\n            watchEndpoint: { videoId },\n          },\n        });\n        this.logger.debug(`Navigation to full player dispatched for video ${videoId}`);\n      } else {\n        // Activate miniplayer: use YT_EVENTS.ACTION with YT_ACTION_NAMES.ACTIVATE_MINIPLAYER_FROM_WATCH\n        this.logger.debug('Activating miniplayer via YouTube API');\n        ytdApp.fire(YT_EVENTS.ACTION, {\n          actionName: YT_ACTION_NAMES.ACTIVATE_MINIPLAYER_FROM_WATCH,\n          args: null,\n          optionalAction: false,\n          returnValue: [undefined],\n        });\n        this.logger.debug('Miniplayer activation event dispatched');\n      }\n    } catch (e) {\n      this.logger.error('Error toggling mini player:', e);\n    }\n  }\n}\n","import { NavigationState } from '../types/youtube';\nimport { WEB_PAGE_TYPES, ROOT_VE, YT_EVENTS } from '../constants';\nimport { SELECTORS } from '../selectors';\nimport type { Nullable } from '../types/app';\n\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { PipWindowProvider } from './PipWindowProvider';\nimport { inject, injectable } from '../di';\n\n/**\n * Handles SPA navigation in PiP window\n */\n@injectable()\nexport class NavigationHandler {\n  private readonly logger: Logger;\n  private pipWindow: Nullable<Window> = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider\n  ) {\n    this.logger = loggerFactory.create('NavigationHandler');\n  }\n\n  /**\n   * Initialize navigation handler for PiP window\n   */\n  public initialize(): void {\n    this.pipWindow = this.pipWindowProvider.getWindow();\n    this.setupClickHandler();\n    this.logger.debug('Navigation handler initialized');\n  }\n\n  /**\n   * Setup click handler for navigation in PiP window\n   */\n  private setupClickHandler(): void {\n    if (!this.pipWindow) {\n      this.logger.error('PiP window not available for navigation handler');\n      return;\n    }\n\n    this.pipWindow.document.addEventListener(\n      'click',\n      (event: MouseEvent) => {\n        const endpoint = (event.target as Element)?.closest<HTMLAnchorElement>(\n          SELECTORS.SIMPLE_ENDPOINT\n        );\n        const button = (event.target as Element)?.closest<HTMLButtonElement>(SELECTORS.BUTTON);\n\n        // Skip button clicks (handled by LikeButtonHandler)\n        if (button || !endpoint) {\n          return;\n        }\n\n        const href = endpoint.href;\n        if (!href) {\n          this.logger.warn('Navigation endpoint has no href');\n          return;\n        }\n\n        this.logger.log('Navigation click detected');\n\n        try {\n          const url = new URL(href);\n          const params = Object.fromEntries(url.searchParams);\n\n          // Prevent default navigation\n          event.preventDefault();\n\n          // Build navigation state\n          const state: NavigationState = {\n            endpoint: {\n              commandMetadata: {\n                webCommandMetadata: {\n                  url: href,\n                  webPageType: WEB_PAGE_TYPES.WATCH,\n                  rootVe: ROOT_VE,\n                },\n              },\n              watchEndpoint: {\n                videoId: params.v,\n                playlistId: params.list || null,\n                index: params.index ? parseInt(params.index) - 1 : 0,\n                params: 'OAE%3D',\n                playerParams: params.pp,\n              },\n            },\n            entryTime: performance.now(),\n          };\n\n          this.logger.log(`SPA navigation via ${YT_EVENTS.NAVIGATE}: ${href}`);\n\n          // Trigger navigation (no pushState — YouTube does not update URL in mini player)\n          window.dispatchEvent(new PopStateEvent('popstate', { state }));\n        } catch (e) {\n          this.logger.error('Error handling navigation:', e);\n        }\n      },\n      true\n    ); // Use capture phase\n  }\n\n  /**\n   * Cleanup navigation handler\n   */\n  public cleanup(): void {\n    this.pipWindow = null;\n    this.logger.debug('Navigation handler cleaned up');\n  }\n}\n","import type { Nullable } from '../types/app';\n\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { inject, injectable } from '../di';\n\n/**\n * Tracks element resize events and updates player\n */\n@injectable()\nexport class ResizeTracker {\n  private readonly logger: Logger;\n  private observer: Nullable<ResizeObserver> = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager\n  ) {\n    this.logger = loggerFactory.create('ResizeTracker');\n  }\n\n  /**\n   * Start tracking resize events for element in PiP window\n   */\n  public start(targetElement: Element): void {\n    if (typeof ResizeObserver === 'undefined') {\n      this.logger.error('ResizeObserver not available');\n      return;\n    }\n\n    this.logger.debug('Starting resize tracking');\n\n    this.observer = new ResizeObserver((entries: ResizeObserverEntry[]) => {\n      for (const entry of entries) {\n        const width = entry.contentRect.width;\n        this.logger.debug(`New size: ${width}px`);\n\n        const player = this.playerManager.getPlayer();\n        // Check if resize methods are available\n        const hasResizeMethods =\n          typeof player.setInternalSize === 'function' || typeof player.setSize === 'function';\n\n        if (!hasResizeMethods) {\n          this.logger.warn('Player resize methods (setInternalSize, setSize) not found');\n        }\n\n        // Call resize methods if available\n        player.setInternalSize?.();\n        player.setSize?.();\n\n        player.dispatchEvent(new Event('resize', { bubbles: true }));\n        this.logger.debug('Player size updated');\n      }\n    });\n\n    this.observer.observe(targetElement);\n  }\n\n  /**\n   * Stop tracking resize events\n   */\n  public stop(): void {\n    if (this.observer) {\n      this.observer.disconnect();\n      this.observer = null;\n      this.logger.debug('Resize tracking stopped');\n    }\n  }\n}\n","import { DEFAULT_DIMENSIONS, TIMEOUTS } from '../constants';\nimport { SELECTORS } from '../selectors';\nimport { DOMUtils } from '../utils/DOMUtils';\nimport type { Nullable } from '../types/app';\n\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport { inject, injectable } from '../di';\n\n/**\n * Observes menu button state and adjusts PiP window.\n * Re-waits for the button when it is removed from DOM (e.g. navigate to video without playlist).\n */\n@injectable()\nexport class MenuObserver {\n  private readonly logger: Logger;\n  private observer: Nullable<MutationObserver> = null;\n  private removalObserver: Nullable<MutationObserver> = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider\n  ) {\n    this.logger = loggerFactory.create('MenuObserver');\n  }\n\n  /**\n   * Start observing menu button in PiP window\n   */\n  public async start(): Promise<void> {\n    const pipWindow = this.pipWindowProvider.getWindow();\n    if (pipWindow) {\n      await this.runObservation(pipWindow);\n    }\n  }\n\n  /**\n   * Wait for button, observe it; when removed from DOM, re-wait and re-observe.\n   */\n  private async runObservation(pipWindow: Window): Promise<void> {\n    if (pipWindow.closed) {\n      this.logger.debug('PiP window already closed, skipping menu observation');\n      return;\n    }\n\n    let button: Element;\n    try {\n      button = await DOMUtils.waitForElementSelector(\n        SELECTORS.MENU_BUTTON,\n        pipWindow.document,\n        TIMEOUTS.ELEMENT_WAIT_INFINITE,\n        pipWindow\n      );\n    } catch (e) {\n      this.logger.warn('Wait for menu button aborted', e);\n      return;\n    }\n\n    this.logger.debug('Starting menu observation');\n\n    this.observer = new MutationObserver((mutations) => {\n      for (const mutation of mutations) {\n        if (mutation.attributeName === 'aria-expanded') {\n          const isExpanded = button.getAttribute('aria-expanded') === 'true';\n          const playListContainer = pipWindow.document.querySelector<HTMLElement>(\n            SELECTORS.PLAYLIST_PANEL\n          );\n          const currentHeight = pipWindow.outerHeight;\n\n          this.logger.debug(`Menu state changed: expanded = ${isExpanded}`);\n\n          if (isExpanded) {\n            // Expand window if needed\n            if (currentHeight < DEFAULT_DIMENSIONS.PIP_EXPANDED_HEIGHT) {\n              pipWindow.resizeTo(pipWindow.outerWidth, DEFAULT_DIMENSIONS.PIP_EXPANDED_HEIGHT);\n              this.logger.debug('PiP window expanded');\n            }\n\n            // Show playlist panel\n            if (playListContainer) {\n              playListContainer.style.display = 'block';\n            }\n          } else {\n            // Hide playlist panel\n            if (playListContainer) {\n              playListContainer.style.display = 'none';\n            }\n          }\n        }\n      }\n    });\n\n    // Observe only aria-expanded attribute\n    this.observer.observe(button, {\n      attributes: true,\n      attributeFilter: ['aria-expanded'],\n    });\n\n    this.removalObserver = new MutationObserver(() => {\n      if (!button.isConnected) {\n        this.observer?.disconnect();\n        this.removalObserver?.disconnect();\n        this.observer = null;\n        this.removalObserver = null;\n        this.logger.debug('Menu button removed from DOM, re-waiting');\n        void this.runObservation(pipWindow);\n      }\n    });\n\n    this.removalObserver.observe(pipWindow.document.body, {\n      childList: true,\n      subtree: true,\n    });\n  }\n\n  /**\n   * Stop observing menu button\n   */\n  public stop(): void {\n    if (this.observer) {\n      this.observer.disconnect();\n      this.observer = null;\n    }\n    if (this.removalObserver) {\n      this.removalObserver.disconnect();\n      this.removalObserver = null;\n    }\n    this.logger.debug('Menu observation stopped');\n  }\n}\n","/**\n * Application-wide types (non-YouTube)\n */\n\n/** T | null */\nexport type Nullable<T> = T | null;\n\n/** Copy menu action: video URL, URL at current time, embed code, or debug info */\nexport enum CopyType {\n  VIDEO_URL = 'video_url',\n  URL_AT_TIME = 'url_at_time',\n  EMBED = 'embed',\n  DEBUG_INFO = 'debug_info',\n}\n\n/** T | Promise<T> - for values that can be sync or async */\nexport type MaybePromise<T> = T | Promise<T>;\n\n/** Document or Element – e.g. for querySelector root */\nexport type DocumentOrElement = Document | Element;\n\n/** Cleanup fn invoked before returning player from PiP to main */\nexport type PiPCleanupCallback = () => MaybePromise<void>;\n\n/** Callback when PiP window is ready: init handlers, optionally return cleanup fn */\nexport type PiPWindowReadyCallback = (\n  pipWindow: Window,\n  miniplayer: Element\n) => MaybePromise<void | PiPCleanupCallback>;\n\n/** Getter/setter descriptor (e.g. for MediaSession metadata override) */\nexport interface MetadataPropertyDescriptor {\n  get?(): unknown;\n  set?(value: unknown): void;\n}\n","/**\n * Shared URL/iframe builders for context menu \"Copy\" actions.\n * Single source of truth: code and tests use these so template changes don't break tests.\n */\nimport type { Nullable } from '../types/app';\nimport { CopyType } from '../types/app';\nimport {\n  EMBED_IFRAME_DEFAULTS,\n  YOUTUBE_SHORT_BASE,\n  YOUTUBE_EMBED_BASE,\n  COPY_PAYLOAD_QUERY,\n  IFRAME_TITLE_QUOT,\n  IFRAME_ALLOW,\n  IFRAME_REFERRER_POLICY,\n} from '../constants';\n\nexport interface CopyPayloadParams {\n  videoId: string;\n  playlistId: Nullable<string>;\n  currentTime: number;\n  title: string;\n  copyType: CopyType;\n  embedSize?: Nullable<{ width: number; height: number }>;\n}\n\n/**\n * Build payload for \"Copy video URL\" (with optional playlist).\n */\nexport function buildVideoUrlPayload(videoId: string, playlistId: Nullable<string>): string {\n  const base = `${YOUTUBE_SHORT_BASE}/${videoId}`;\n  const listPart = playlistId ? `?${COPY_PAYLOAD_QUERY.LIST}=${playlistId}` : '';\n  return listPart ? `${base}${listPart}` : base;\n}\n\n/**\n * Build payload for \"Copy URL at current time\".\n */\nexport function buildUrlAtTimePayload(\n  videoId: string,\n  playlistId: Nullable<string>,\n  currentTime: number\n): string {\n  const base = `${YOUTUBE_SHORT_BASE}/${videoId}`;\n  const listPart = playlistId ? `?${COPY_PAYLOAD_QUERY.LIST}=${playlistId}` : '';\n  const tPart =\n    currentTime > 0\n      ? listPart\n        ? `&${COPY_PAYLOAD_QUERY.TIME}=${currentTime}s`\n        : `?${COPY_PAYLOAD_QUERY.TIME}=${currentTime}s`\n      : '';\n  return `${base}${listPart}${tPart}`;\n}\n\n/**\n * Build embed iframe HTML (single source of truth for template).\n */\nexport function buildEmbedPayload(\n  videoId: string,\n  playlistId: Nullable<string>,\n  title: string,\n  embedSize?: Nullable<{ width: number; height: number }>\n): string {\n  const w = embedSize?.width ?? EMBED_IFRAME_DEFAULTS.WIDTH;\n  const h = embedSize?.height ?? EMBED_IFRAME_DEFAULTS.HEIGHT;\n  const listQ = playlistId ? `?${COPY_PAYLOAD_QUERY.LIST}=${playlistId}` : '';\n  const src = `${YOUTUBE_EMBED_BASE}/${videoId}${listQ}`;\n  const safeTitle = title.replace(/\"/g, IFRAME_TITLE_QUOT);\n  return `<iframe width=\"${w}\" height=\"${h}\" src=\"${src}\" title=\"${safeTitle}\" frameborder=\"0\" allow=\"${IFRAME_ALLOW}\" referrerpolicy=\"${IFRAME_REFERRER_POLICY}\" allowfullscreen></iframe>`;\n}\n\n/**\n * Build copy payload by copy type (used by ContextMenuHandler and tests).\n */\nexport function buildCopyPayload(params: CopyPayloadParams): string {\n  const { videoId, playlistId, currentTime, title, copyType, embedSize } = params;\n  switch (copyType) {\n    case CopyType.VIDEO_URL:\n      return buildVideoUrlPayload(videoId, playlistId);\n    case CopyType.URL_AT_TIME:\n      return buildUrlAtTimePayload(videoId, playlistId, currentTime);\n    case CopyType.EMBED:\n      return buildEmbedPayload(videoId, playlistId, title, embedSize);\n    default:\n      return '';\n  }\n}\n","import type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { TIMEOUTS, MOUSE_BUTTONS, COPY_MENU_INDICES } from '../constants';\nimport { DOMUtils } from '../utils/DOMUtils';\nimport { SELECTORS } from '../selectors';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { YtdAppProvider } from '../core/YtdAppProvider';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport { CopyType, type Nullable } from '../types/app';\nimport { buildCopyPayload } from '../utils/copyPayload';\nimport { inject, injectable } from '../di';\n\n/**\n * Handles context menu movement between windows and copy commands in PiP.\n * YouTube uses a hidden textarea in the main window for copy; when the menu\n * is moved to the PiP popup that link breaks. We intercept copy menu clicks\n * and copy via a temporary textarea in the PiP document.\n */\nexport type ContextMenuVisibilityCallback = (visible: boolean) => void;\n\n@injectable()\nexport class ContextMenuHandler {\n  private readonly logger: Logger;\n  private visibilityObserver: Nullable<MutationObserver> = null;\n  private pipWindow: Nullable<Window> = null;\n  private contextMenu: Nullable<HTMLElement> = null;\n  private contextMenuPlaceholder: Nullable<Comment> = null;\n  private readonly visibilitySubscribers = new Set<ContextMenuVisibilityCallback>();\n\n  /**\n   * Subscribe to context menu visibility changes.\n   * @returns Unsubscribe function\n   */\n  public subscribeContextMenu(callback: ContextMenuVisibilityCallback): () => void {\n    this.visibilitySubscribers.add(callback);\n    return () => {\n      this.visibilitySubscribers.delete(callback);\n    };\n  }\n\n  private notifyVisibility(visible: boolean): void {\n    this.visibilitySubscribers.forEach((cb) => cb(visible));\n  }\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager,\n    @inject(YtdAppProvider) private readonly ytdAppProvider: YtdAppProvider,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider\n  ) {\n    this.logger = loggerFactory.create('ContextMenuHandler');\n  }\n\n  /**\n   * Capture-phase click handler for copy menu items.\n   * Defined as arrow function to preserve `this` binding.\n   */\n  private handleCopyClick = (e: MouseEvent): void => {\n    if (!this.pipWindow) {\n      return;\n    }\n\n    const doc = this.pipWindow.document;\n    const item = (e.target as Element)?.closest(SELECTORS.MENU_ITEM);\n    if (!item?.parentElement) {\n      this.logger.debug('Copy click: not a menu item or no parent', { item });\n      return;\n    }\n\n    const items = doc.querySelectorAll(SELECTORS.PANEL_MENU_ITEMS);\n    const index = Array.prototype.indexOf.call(items, item);\n    if (index === -1) {\n      this.logger.warn('Copy click: menu item index not found');\n      return;\n    }\n\n    const copyType = this.getCopyTypeForIndex(index);\n    if (!copyType) {\n      this.logger.debug('Copy click: not a copy action', { index });\n      return;\n    }\n\n    let text: string;\n    switch (copyType) {\n      case CopyType.DEBUG_INFO: {\n        text = this.playerManager.getDebugInfo() ?? '';\n        if (!text) {\n          this.logger.warn('Debug info not available, cannot copy');\n          return;\n        }\n        break;\n      }\n      default: {\n        const videoData = this.playerManager.getVideoData();\n        const videoId = videoData?.video_id;\n        if (!videoId) {\n          this.logger.warn('Video ID not found, cannot copy');\n          return;\n        }\n        const playlistId = videoData?.list ?? null;\n        const currentTime = this.playerManager.getCurrentTime();\n        const title = videoData?.title ?? '';\n        const embedSize = copyType === CopyType.EMBED ? this.playerManager.getPlayerSize() : null;\n        text = this.getCopyPayload({\n          videoId,\n          playlistId,\n          currentTime,\n          title,\n          copyType,\n          embedSize,\n        });\n        if (!text) {\n          this.logger.warn('Copy click: empty payload', { copyType });\n          return;\n        }\n        break;\n      }\n    }\n\n    const ok = DOMUtils.copyViaTextarea(doc, text);\n    if (ok) {\n      this.logger.debug(`Copied ${copyType} to clipboard`);\n    }\n  };\n\n  /**\n   * Initialize context menu handler\n   */\n  public async initialize(): Promise<void> {\n    this.pipWindow = this.pipWindowProvider.getWindow();\n    this.contextMenuPlaceholder = DOMUtils.createPlaceholder('context_menu_placeholder');\n\n    // Wait for context menu to appear\n    try {\n      this.contextMenu = await DOMUtils.waitForElementSelector<HTMLElement>(\n        SELECTORS.CONTEXT_MENU,\n        document,\n        TIMEOUTS.ELEMENT_WAIT_INFINITE,\n        this.pipWindow\n      );\n\n      this.logger.log('Context menu element found, starting visibility monitoring');\n\n      this.startMonitoring();\n      this.setupDismissalHandler();\n      this.setupCopyHandler();\n    } catch (e) {\n      this.logger.warn('Error initializing context menu handler:', e);\n    }\n  }\n\n  /**\n   * Start monitoring context menu visibility\n   */\n  private startMonitoring(): void {\n    if (!this.contextMenu || !this.pipWindow) {\n      return;\n    }\n\n    this.visibilityObserver = new MutationObserver(() => {\n      if (!this.contextMenu || !this.pipWindow) {\n        return;\n      }\n\n      const isVisible = this.contextMenu.style.display !== 'none';\n      const isInMainWindow = this.contextMenu.parentNode !== this.pipWindow.document.body;\n\n      if (isVisible && isInMainWindow) {\n        this.logger.log('Context menu opened in main window. Intercepting...');\n\n        if (this.contextMenuPlaceholder) {\n          DOMUtils.insertPlaceholderBefore(this.contextMenu, this.contextMenuPlaceholder);\n        }\n        this.pipWindow.document.body.appendChild(this.contextMenu);\n        this.notifyVisibility(true);\n      } else if (!isVisible && this.contextMenu.parentNode === this.pipWindow.document.body) {\n        this.notifyVisibility(false);\n        if (this.contextMenuPlaceholder?.parentNode) {\n          this.logger.log('Context menu closed in PiP window. Returning to main...');\n          DOMUtils.restoreElementFromPlaceholder(this.contextMenu, this.contextMenuPlaceholder);\n          this.simulateMainContextMenu();\n        }\n      }\n    });\n\n    // Observe style attribute changes\n    this.visibilityObserver.observe(this.contextMenu, {\n      attributes: true,\n      attributeFilter: ['style'],\n    });\n\n    // Move menu immediately if already visible\n    if (this.contextMenu.style.display !== 'none') {\n      if (this.contextMenuPlaceholder) {\n        DOMUtils.insertPlaceholderBefore(this.contextMenu, this.contextMenuPlaceholder);\n      }\n      this.pipWindow.document.body.appendChild(this.contextMenu);\n      this.notifyVisibility(true);\n    }\n\n    // Cleanup is done via PiPManager.onBeforeReturn\n  }\n\n  /**\n   * Setup handler to dismiss context menu on click outside\n   */\n  private setupDismissalHandler(): void {\n    if (!this.pipWindow) {\n      return;\n    }\n\n    const handleEvent = (e: MouseEvent) => {\n      const menuInPip = this.pipWindow!.document.querySelector<HTMLElement>(SELECTORS.CONTEXT_MENU);\n\n      if (\n        menuInPip &&\n        menuInPip.style.display !== 'none' &&\n        !(e.target as Element)?.closest(SELECTORS.CONTEXT_MENU_CONTAINER)\n      ) {\n        e.stopPropagation();\n        menuInPip.style.display = 'none';\n        this.notifyVisibility(false);\n        this.logger.debug('Context menu dismissed');\n      }\n    };\n\n    this.pipWindow.document.addEventListener('click', handleEvent, true);\n    this.pipWindow.document.addEventListener('contextmenu', handleEvent, true);\n  }\n\n  /**\n   * Setup capture-phase click handler for copy menu items in PiP.\n   * Copies via temporary textarea since main-window textarea is disconnected.\n   */\n  private setupCopyHandler(): void {\n    if (!this.pipWindow) {\n      return;\n    }\n\n    this.pipWindow.document.addEventListener('click', this.handleCopyClick, true);\n  }\n\n  private getCopyTypeForIndex(index: number): CopyType | null {\n    if (index === COPY_MENU_INDICES.VIDEO_URL) return CopyType.VIDEO_URL;\n    if (index === COPY_MENU_INDICES.URL_AT_TIME) return CopyType.URL_AT_TIME;\n    if (index === COPY_MENU_INDICES.EMBED) return CopyType.EMBED;\n    if (index === COPY_MENU_INDICES.DEBUG_INFO) return CopyType.DEBUG_INFO;\n    return null;\n  }\n\n  private getCopyPayload(params: {\n    videoId: string;\n    playlistId: Nullable<string>;\n    currentTime: number;\n    title: string;\n    copyType: CopyType;\n    embedSize?: Nullable<{ width: number; height: number }>;\n  }): string {\n    return buildCopyPayload(params);\n  }\n\n  /**\n   * Simulate context menu event in main window\n   */\n  private simulateMainContextMenu(): void {\n    const mainApp = this.ytdAppProvider.getApp();\n\n    const event = new MouseEvent('contextmenu', {\n      bubbles: true,\n      cancelable: true,\n      clientX: 0,\n      clientY: 0,\n      button: MOUSE_BUTTONS.SECONDARY,\n    });\n\n    mainApp.dispatchEvent(event);\n    this.logger.debug('Synthetic contextmenu event sent to main window');\n  }\n\n  /**\n   * Stop monitoring and cleanup\n   */\n  public stop(): void {\n    if (this.visibilityObserver) {\n      this.visibilityObserver.disconnect();\n      this.visibilityObserver = null;\n    }\n\n    if (this.pipWindow) {\n      this.pipWindow.document.removeEventListener('click', this.handleCopyClick, true);\n    }\n\n    // Return menu to main window if still in PiP\n    if (\n      this.contextMenu &&\n      this.pipWindow &&\n      this.contextMenu.parentNode === this.pipWindow.document.body &&\n      this.contextMenuPlaceholder?.parentNode\n    ) {\n      this.logger.log('Returning context menu to main window');\n      DOMUtils.restoreElementFromPlaceholder(this.contextMenu, this.contextMenuPlaceholder);\n      this.simulateMainContextMenu();\n    }\n\n    this.pipWindow = null;\n    this.logger.debug('Context menu handler stopped');\n  }\n}\n","import { SELECTORS } from '../selectors';\nimport type { Nullable } from '../types/app';\n\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport { inject, injectable } from '../di';\n\n/**\n * Handles manual seeking on progress bar\n */\n@injectable()\nexport class SeekHandler {\n  private readonly logger: Logger;\n  private pipWindow: Nullable<Window> = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider\n  ) {\n    this.logger = loggerFactory.create('SeekHandler');\n  }\n\n  /**\n   * Initialize seek handler for PiP window\n   */\n  public initialize(): void {\n    this.pipWindow = this.pipWindowProvider.getWindow();\n    this.setupSeekHandler();\n    this.logger.debug('Seek handler initialized');\n  }\n\n  /**\n   * Setup mouse handlers for progress bar seeking\n   */\n  private setupSeekHandler(): void {\n    if (!this.pipWindow) {\n      return;\n    }\n\n    const pipDoc = this.pipWindow.document;\n    pipDoc.addEventListener(\n      'mousedown',\n      (e: MouseEvent) => {\n        const progressBar = (e.target as Element)?.closest(SELECTORS.PROGRESS_BAR);\n        if (!progressBar) return;\n\n        const player = this.playerManager.getPlayer();\n        if (typeof player.getDuration !== 'function') {\n          this.logger.error('player.getDuration method not found');\n          return;\n        }\n        if (typeof player.seekTo !== 'function') {\n          this.logger.error('player.seekTo method not found');\n          return;\n        }\n\n        this.logger.debug('Progress bar clicked, initiating seek');\n\n        e.preventDefault();\n        e.stopPropagation();\n\n        const performSeek = (event: MouseEvent) => {\n          const rect = progressBar.getBoundingClientRect();\n          const x = event.clientX - rect.left;\n          const percent = Math.max(0, Math.min(1, x / rect.width));\n          const duration = player.getDuration?.();\n          if (duration) {\n            const seekTime = percent * duration;\n            player.seekTo?.(seekTime, true);\n            this.logger.debug(\n              `Seeking to ${seekTime.toFixed(2)}s (${(percent * 100).toFixed(1)}%)`\n            );\n          }\n        };\n\n        performSeek(e);\n\n        const onMouseMove = (moveEvent: MouseEvent) => performSeek(moveEvent);\n        const onMouseUp = () => {\n          pipDoc.removeEventListener('mousemove', onMouseMove);\n          pipDoc.removeEventListener('mouseup', onMouseUp);\n          this.logger.debug('Seek drag ended');\n        };\n\n        pipDoc.addEventListener('mousemove', onMouseMove);\n        pipDoc.addEventListener('mouseup', onMouseUp);\n      },\n      true\n    );\n  }\n\n  /**\n   * Cleanup seek handler\n   */\n  public cleanup(): void {\n    this.pipWindow = null;\n    this.logger.debug('Seek handler cleaned up');\n  }\n}\n","import { PlayerManager } from './PlayerManager';\nimport { YtdAppProvider } from './YtdAppProvider';\nimport { PipWindowProvider } from './PipWindowProvider';\nimport { YouTubeCommand, LikeActionType } from '../types/youtube';\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { inject, injectable } from '../di';\n\n/**\n * Sends YouTube like/dislike/remove commands to main window ytd-app.\n * Used by LikeButtonHandler when user clicks like/dislike in PiP.\n */\n@injectable()\nexport class YtActionSender {\n  private readonly logger: Logger;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager,\n    @inject(YtdAppProvider) private readonly ytdAppProvider: YtdAppProvider,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider\n  ) {\n    this.logger = loggerFactory.create('YtActionSender');\n  }\n\n  /**\n   * Send like/dislike action (LIKE / DISLIKE / REMOVE) for the current PiP video.\n   * Resolves video ID from PiP player and invokes main window resolveCommand.\n   */\n  public sendLikeAction(actionType: LikeActionType): void {\n    const pipWindow = this.pipWindowProvider.getWindow();\n    if (!pipWindow) {\n      return;\n    }\n\n    const videoId = this.playerManager.getVideoId();\n\n    if (!videoId) {\n      this.logger.error('Video ID not found, cannot send like action');\n      return;\n    }\n\n    const mainApp = this.ytdAppProvider.getApp();\n    if (typeof mainApp.resolveCommand !== 'function') {\n      this.logger.error('Failed to find resolveCommand in main window');\n      return;\n    }\n\n    const command: YouTubeCommand = {\n      likeEndpoint: {\n        status: actionType,\n        target: { videoId },\n      },\n    };\n\n    try {\n      mainApp.resolveCommand(command);\n      this.logger.log(`Sent ${actionType} for video ${videoId}`);\n    } catch (e) {\n      this.logger.error('Error sending YouTube action:', e);\n    }\n  }\n}\n","import { YT_LIKE_ACTIONS } from '../constants';\nimport { SELECTORS } from '../selectors';\nimport { YtActionSender } from '../core/YtActionSender';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport type { Nullable } from '../types/app';\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { inject, injectable } from '../di';\n\n/**\n * Handles like/dislike button clicks in PiP window\n */\n@injectable()\nexport class LikeButtonHandler {\n  private readonly logger: Logger;\n  private pipWindow: Nullable<Window> = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider,\n    @inject(YtActionSender) private readonly ytActionSender: YtActionSender\n  ) {\n    this.logger = loggerFactory.create('LikeButtonHandler');\n  }\n\n  /**\n   * Initialize like button handler for PiP window\n   */\n  public initialize(): void {\n    this.pipWindow = this.pipWindowProvider.getWindow();\n    this.setupClickHandler();\n    this.logger.debug('Like button handler initialized');\n  }\n\n  /**\n   * Setup click handler for like/dislike buttons\n   */\n  private setupClickHandler(): void {\n    if (!this.pipWindow) {\n      return;\n    }\n\n    this.pipWindow.document.addEventListener(\n      'click',\n      (event: MouseEvent) => {\n        const toggleButton = (event.target as Element)?.closest<HTMLButtonElement>(\n          SELECTORS.LIKE_BUTTON\n        );\n        if (!toggleButton) {\n          return;\n        }\n\n        const buttonContainer = toggleButton.parentElement;\n        if (!buttonContainer) {\n          return;\n        }\n\n        const isLikeButton = buttonContainer.childNodes[0] === toggleButton;\n        const isDislikeButton = buttonContainer.childNodes[1] === toggleButton;\n\n        if (!isLikeButton && !isDislikeButton) {\n          return;\n        }\n\n        const button = (event.target as Element)?.closest<HTMLButtonElement>(\n          SELECTORS.BUTTON_SHAPE\n        );\n        if (!button) {\n          return;\n        }\n\n        const isPressed = button.getAttribute('aria-pressed') === 'true';\n        const actionType = isPressed\n          ? YT_LIKE_ACTIONS.REMOVE\n          : isLikeButton\n            ? YT_LIKE_ACTIONS.LIKE\n            : YT_LIKE_ACTIONS.DISLIKE;\n\n        this.logger.log(`${actionType} button clicked (currently pressed: ${isPressed})`);\n\n        this.ytActionSender.sendLikeAction(actionType);\n      },\n      true\n    ); // Capture phase\n  }\n\n  /**\n   * Cleanup like button handler\n   */\n  public cleanup(): void {\n    this.pipWindow = null;\n    this.logger.debug('Like button handler cleaned up');\n  }\n}\n","import type { Nullable } from '../types/app';\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport { ContextMenuHandler } from '../ui/ContextMenuHandler';\nimport { inject, injectable } from '../di';\n\n/**\n * Listens to capture click on body and keyup on document (any key except Tab).\n * Returns focus to player when it moves outside the player, but only when context\n * menu is closed. Uses setTimeout(0) so focus runs after other handlers.\n */\n@injectable()\nexport class DocumentFocusHandler {\n  private readonly logger: Logger;\n  private pipWindow: Nullable<Window> = null;\n  private isContextMenuOpen = false;\n  private unsubscribeContextMenu: (() => void) | null = null;\n\n  private returnFocusToPlayerIfNeeded(): void {\n    if (!this.pipWindow || this.isContextMenuOpen) return;\n\n    const active = this.pipWindow.document.activeElement;\n    const player = this.playerManager.getPlayer();\n    if (!active || active === player) return;\n\n    if (typeof player.focus === 'function') {\n      this.logger.debug('Returning focus to player');\n      setTimeout(() => player.focus(), 0);\n    }\n  }\n\n  private readonly onBodyClick = (): void => this.returnFocusToPlayerIfNeeded();\n\n  private readonly onKeyUp = (e: KeyboardEvent): void => {\n    if (e.key === 'Tab') return;\n    this.returnFocusToPlayerIfNeeded();\n  };\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider,\n    @inject(ContextMenuHandler) private readonly contextMenuHandler: ContextMenuHandler\n  ) {\n    this.logger = loggerFactory.create('DocumentFocusHandler');\n  }\n\n  /**\n   * Initialize focus observer for PiP window\n   */\n  public initialize(): void {\n    this.pipWindow = this.pipWindowProvider.getWindow();\n    if (!this.pipWindow) {\n      this.logger.error('PiP window not available for document focus handler');\n      return;\n    }\n\n    this.unsubscribeContextMenu = this.contextMenuHandler.subscribeContextMenu((visible) => {\n      this.isContextMenuOpen = visible;\n      if (!visible) {\n        this.returnFocusToPlayerIfNeeded();\n      }\n    });\n\n    this.pipWindow.document.body.addEventListener('click', this.onBodyClick, true);\n    this.pipWindow.document.addEventListener('keyup', this.onKeyUp, true);\n    this.logger.debug('Document focus handler initialized');\n  }\n\n  /**\n   * Cleanup\n   */\n  public cleanup(): void {\n    this.unsubscribeContextMenu?.();\n    this.unsubscribeContextMenu = null;\n    if (this.pipWindow?.document?.body) {\n      this.pipWindow.document.body.removeEventListener('click', this.onBodyClick, true);\n      this.pipWindow.document.removeEventListener('keyup', this.onKeyUp, true);\n    }\n    this.pipWindow = null;\n    this.isContextMenuOpen = false;\n    this.logger.debug('Document focus handler cleaned up');\n  }\n}\n","import type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport { YtdAppProvider } from '../core/YtdAppProvider';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { SELECTORS } from '../selectors';\nimport type { YouTubePlayer } from '../types/youtube';\nimport { inject, injectable } from '../di';\n\n/**\n * Syncs PiP window title when the player's video element src or the notify renderer DOM changes.\n * Observes the video element (src) and the notify renderer element (childList) and updates title from getVideoData().\n * Does not depend on PiPManager to avoid circular dependencies.\n */\n@injectable()\nexport class TitleSyncHandler {\n  private readonly logger: Logger;\n  private mutationObserver: MutationObserver | null = null;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider,\n    @inject(YtdAppProvider) private readonly ytdAppProvider: YtdAppProvider,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager\n  ) {\n    this.logger = loggerFactory.create('TitleSyncHandler');\n  }\n\n  /**\n   * Start observing the player's video element (src) and the notify renderer element (DOM changes) and sync title.\n   * Call when PiP window is open. Skips sync when PiP was opened from mini player (from PlayerManager).\n   */\n  public initialize(): void {\n    if (this.playerManager.getWasMiniPlayerActiveBeforePiP()) {\n      this.logger.debug('Title sync skipped (e.g. PiP opened from mini player)');\n      return;\n    }\n\n    const pipWindow = this.pipWindowProvider.getWindow();\n    if (!pipWindow) {\n      this.logger.debug('PiP not open, skipping title sync init');\n      return;\n    }\n\n    const player = this.playerManager.getPlayer() as YouTubePlayer;\n    const video = player.querySelector(SELECTORS.PLAYER_VIDEO);\n    if (!video) {\n      this.logger.warn('Video element not found inside player');\n      return;\n    }\n\n    const syncTitle = (): void => {\n      const data = player.getVideoData?.();\n      if (data?.title) {\n        this.setWindowsTitle(data.title);\n      }\n    };\n\n    syncTitle();\n\n    this.mutationObserver = new MutationObserver(() => {\n      syncTitle();\n    });\n    this.mutationObserver.observe(video, {\n      attributes: true,\n      attributeFilter: ['src'],\n    });\n    const notifyRenderer = this.ytdAppProvider.getNotifyRenderer();\n    if (notifyRenderer) {\n      this.mutationObserver.observe(notifyRenderer, {\n        childList: true,\n        subtree: true,\n      });\n    }\n    this.logger.debug('Title sync observing video src and notify renderer subtree');\n  }\n\n  private setWindowsTitle(title: string): void {\n    const pipWindow = this.pipWindowProvider.getWindow();\n    if (!pipWindow) return;\n    const notify = this.ytdAppProvider.getNotifyRenderer();\n    const countNotif = notify?.showNotificationCount ? `(${notify.showNotificationCount}) ` : '';\n    const newTitle = `${countNotif}${title} - YouTube`;\n    document.title = newTitle;\n    pipWindow.document.title = newTitle;\n    this.logger.debug(`Title synced: ${title}`);\n  }\n\n  /**\n   * Stop observing. Call when PiP window is closing.\n   */\n  public cleanup(): void {\n    if (this.mutationObserver) {\n      this.mutationObserver.disconnect();\n      this.mutationObserver = null;\n      this.logger.debug('Title sync observer disconnected');\n    }\n  }\n}\n","import { ResizeTracker } from '../ui/ResizeTracker';\nimport { MenuObserver } from '../ui/MenuObserver';\nimport { ContextMenuHandler } from '../ui/ContextMenuHandler';\nimport { SeekHandler } from '../handlers/SeekHandler';\nimport { LikeButtonHandler } from '../handlers/LikeButtonHandler';\nimport { NavigationHandler } from '../core/NavigationHandler';\nimport { DocumentFocusHandler } from '../handlers/DocumentFocusHandler';\nimport { TitleSyncHandler } from '../handlers/TitleSyncHandler';\nimport type { PiPCleanupCallback } from '../types/app';\nimport { inject, injectable } from '../di';\n\n/**\n * Initializes PiP window handlers and returns cleanup callback\n */\n@injectable()\nexport class PiPWindowHandlers {\n  constructor(\n    @inject(ResizeTracker) private readonly resizeTracker: ResizeTracker,\n    @inject(MenuObserver) private readonly menuObserver: MenuObserver,\n    @inject(ContextMenuHandler) private readonly contextMenuHandler: ContextMenuHandler,\n    @inject(SeekHandler) private readonly seekHandler: SeekHandler,\n    @inject(LikeButtonHandler) private readonly likeButtonHandler: LikeButtonHandler,\n    @inject(NavigationHandler) private readonly navigationHandler: NavigationHandler,\n    @inject(DocumentFocusHandler) private readonly documentFocusHandler: DocumentFocusHandler,\n    @inject(TitleSyncHandler) private readonly titleSyncHandler: TitleSyncHandler\n  ) {}\n\n  public async initialize(miniplayer: Element): Promise<PiPCleanupCallback> {\n    this.navigationHandler.initialize();\n    this.resizeTracker.start(miniplayer);\n    void this.menuObserver.start();\n    void this.contextMenuHandler.initialize();\n    this.seekHandler.initialize();\n    this.likeButtonHandler.initialize();\n    this.documentFocusHandler.initialize();\n    this.titleSyncHandler.initialize();\n\n    return () => {\n      this.titleSyncHandler.cleanup();\n      this.documentFocusHandler.cleanup();\n      this.seekHandler.cleanup();\n      this.likeButtonHandler.cleanup();\n      this.contextMenuHandler.stop();\n      this.menuObserver.stop();\n      this.resizeTracker.stop();\n      this.navigationHandler.cleanup();\n    };\n  }\n}\n","export default \"/**\\n * CSS styles for PiP window fixes and customizations\\n */\\n\\nbody {\\n  overflow: auto !important;\\n}\\n\\nytd-miniplayer {\\n  left: 0 !important;\\n  right: 0 !important;\\n  top: 0 !important;\\n  bottom: 0 !important;\\n  width: 100% !important;\\n  height: 100% !important;\\n  max-height: 100% !important;\\n  display: flex !important;\\n}\\n\\nytd-miniplayer-player-container {\\n  width: 100% !important;\\n  height: 100% !important;\\n}\\n\\n.html5-video-container {\\n  position: absolute;\\n  top: 0;\\n  left: 0;\\n  bottom: 0;\\n  right: 0;\\n}\\n\\n.html5-video-container .html5-main-video {\\n  width: 100% !important;\\n  height: 100% !important;\\n  object-fit: contain;\\n}\\n\\nvideo {\\n  left: 0 !important;\\n  top: 0 !important;\\n  right: 0 !important;\\n  bottom: 0 !important;\\n}\\n\\n.ytDraggableComponentHost.ytdMiniplayerComponentDraggable {\\n  pointer-events: auto !important;\\n}\\n\\n.ytp-miniplayer-expand-watch-page-button,\\n.ytp-miniplayer-close-button,\\n#header-contents,\\n.ytdMiniplayerComponentResizers,\\n/* TODO: remove when context menu interception is implemented */\\n.dropdown-trigger {\\n  display: none !important;\\n}\\n\\n/* Stretch progress bar to full width */\\n.ytp-progress-bar-container,\\n.ytp-progress-bar {\\n  width: 100% !important;\\n}\\n\\n.ytp-prev-button,\\n.ytp-next-button {\\n  display: inline-flex !important;\\n}\\n\"","import { CSS_FIXES } from '../styles';\nimport { Logger } from '../logger';\nimport { SELECTORS } from '../selectors';\n\nconst logger = Logger.getInstance('StyleUtils');\n\n/**\n * Utility class for style and CSS operations\n */\nexport class StyleUtils {\n  /**\n   * Copy all stylesheets from source document to target document\n   */\n  public static copyStyles(sourceDoc: Document, targetDoc: Document): void {\n    try {\n      const styles = sourceDoc.querySelectorAll(SELECTORS.STYLESHEETS);\n      logger.debug(`Copying ${styles.length} style elements`);\n\n      styles.forEach((node) => {\n        try {\n          targetDoc.head.appendChild(node.cloneNode(true));\n        } catch (e) {\n          logger.warn('Failed to copy style node:', e);\n        }\n      });\n\n      logger.debug('Styles copied successfully');\n    } catch (e) {\n      logger.error('Error copying styles:', e);\n    }\n  }\n\n  /**\n   * Inject CSS fixes into target document\n   */\n  public static injectCSSFixes(targetDoc: Document): void {\n    try {\n      const styleFix = targetDoc.createElement('style');\n      styleFix.textContent = CSS_FIXES;\n      targetDoc.head.appendChild(styleFix);\n      logger.debug('CSS fixes injected successfully');\n    } catch (e) {\n      logger.error('Error injecting CSS fixes:', e);\n    }\n  }\n}\n","import { Logger } from '../logger';\n\nconst logger = Logger.getInstance('AsyncLock');\n\n/**\n * Async mutex for critical sections.\n * Ensures only one async block runs at a time; concurrent callers wait in queue.\n */\nexport class AsyncLock {\n  private locked = false;\n  private readonly queue: Array<() => void> = [];\n\n  /**\n   * Execute fn while holding the lock. Waits if another execution is in progress.\n   * Lock is always released in finally, even if fn throws.\n   */\n  public async withLock<T>(fn: () => Promise<T>): Promise<T> {\n    logger.debug('withLock: waiting for lock');\n    await this.acquire();\n    logger.debug('withLock: lock acquired, running fn');\n    try {\n      return await fn();\n    } finally {\n      this.release();\n      logger.debug('withLock: lock released');\n    }\n  }\n\n  private acquire(): Promise<void> {\n    if (!this.locked) {\n      this.locked = true;\n      logger.debug('acquire: lock taken immediately');\n      return Promise.resolve();\n    }\n    logger.debug('acquire: lock busy, joining queue', { queueSize: this.queue.length });\n    return new Promise<void>((resolve) => {\n      this.queue.push(() => {\n        this.locked = true;\n        logger.debug('acquire: woke from queue');\n        resolve();\n      });\n    });\n  }\n\n  private release(): void {\n    if (this.queue.length > 0) {\n      const next = this.queue.shift()!;\n      logger.debug('release: passing lock to next in queue', { remaining: this.queue.length });\n      next();\n    } else {\n      this.locked = false;\n      logger.debug('release: lock freed');\n    }\n  }\n}\n","import { AppError } from './AppError';\n\n/**\n * Error thrown during PiP operations.\n */\nexport class PiPError extends AppError {\n  constructor(message: string, cause?: unknown) {\n    super(message, cause);\n    this.name = 'PiPError';\n  }\n}\n","import { AppError } from './AppError';\n\n/**\n * Thrown when a PiP operation has left the YouTube page in a broken state.\n * e.g. Mini player was moved to PiP window but a required DOM element is missing,\n * so the main page cannot be restored correctly.\n */\nexport class PiPCriticalError extends AppError {\n  constructor(message: string, cause?: unknown) {\n    super(message, cause);\n    this.name = 'PiPCriticalError';\n  }\n}\n","import type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { DOMUtils } from '../utils/DOMUtils';\nimport { StyleUtils } from '../utils/StyleUtils';\nimport { AsyncLock } from '../utils/AsyncLock';\nimport { MiniPlayerController } from '../ui/MiniPlayerController';\nimport { PlayerManager } from './PlayerManager';\nimport { YtdAppProvider } from './YtdAppProvider';\nimport { PipWindowProvider } from './PipWindowProvider';\nimport { DEFAULT_DIMENSIONS, TIMEOUTS } from '../constants';\nimport { SELECTORS } from '../selectors';\nimport type { Nullable, PiPCleanupCallback } from '../types/app';\nimport { PiPError } from '../errors/PiPError';\nimport { PiPCriticalError } from '../errors/PiPCriticalError';\nimport { PiPWindowHandlers } from './PiPWindowHandlers';\nimport { inject, injectable } from '../di';\n\n/**\n * Manages Picture-in-Picture window lifecycle\n */\n@injectable()\nexport class PiPManager {\n  private readonly logger: Logger;\n  private miniPlayerContainer: Nullable<Element> = null;\n  private placeholder: Nullable<Comment> = null;\n\n  private onBeforeReturn: Nullable<PiPCleanupCallback> = null;\n\n  private readonly asyncLock = new AsyncLock();\n\n  private close = async (): Promise<void> => {\n    return this.asyncLock\n      .withLock(() => this.returnPlayerToMain())\n      .catch((e) => {\n        this.logger.error('Unhandled error in returnPlayerToMain:', e);\n      });\n  };\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(MiniPlayerController) private readonly miniPlayerController: MiniPlayerController,\n    @inject(PlayerManager) private readonly playerManager: PlayerManager,\n    @inject(YtdAppProvider) private readonly ytdAppProvider: YtdAppProvider,\n    @inject(PipWindowProvider) private readonly pipWindowProvider: PipWindowProvider,\n    @inject(PiPWindowHandlers) private readonly pipWindowHandlers: PiPWindowHandlers\n  ) {\n    this.logger = loggerFactory.create('PiPManager');\n  }\n\n  /**\n   * Check if PiP window is open\n   */\n  public isOpen(): boolean {\n    return this.pipWindowProvider.getWindow() !== null;\n  }\n\n  /**\n   * Get PiP window instance\n   */\n  public getWindow(): Nullable<Window> {\n    return this.pipWindowProvider.getWindow();\n  }\n\n  /**\n   * Open Picture-in-Picture window\n   * Critical section: concurrent calls are serialized to prevent race conditions\n   */\n  public open(): Promise<void> {\n    return this.asyncLock.withLock(async () => {\n      if (this.isOpen()) {\n        this.logger.warn('PiP window already open');\n        return;\n      }\n\n      this.logger.log('Opening PiP window');\n\n      try {\n        await this.movePlayerToPIP();\n\n        const pipWindow = this.pipWindowProvider.getWindow();\n        if (pipWindow) {\n          const result = await this.pipWindowHandlers.initialize(\n            this.miniPlayerController.getMiniplayer()\n          );\n          if (typeof result === 'function') {\n            this.onBeforeReturn = result;\n          }\n        }\n      } catch (error) {\n        if (error instanceof PiPCriticalError) {\n          throw error;\n        }\n        throw new PiPError('Error opening PiP', error);\n      }\n    });\n  }\n\n  /**\n   * Move player to PiP window\n   */\n  private async movePlayerToPIP(): Promise<void> {\n    const miniplayer = this.miniPlayerController.getMiniplayer();\n\n    // Save mini player state\n    this.playerManager.setWasMiniPlayerActiveBeforePiP(this.miniPlayerController.isVisible());\n\n    if (!this.playerManager.getWasMiniPlayerActiveBeforePiP()) {\n      this.miniPlayerController.toggleMiniPlayer();\n\n      // Wait for mini player container\n      await DOMUtils.waitForElementSelector(SELECTORS.MINIPLAYER_CONTAINER);\n      this.logger.debug('Mini player container ready');\n    }\n\n    // Get dimensions\n    const width = miniplayer.offsetWidth || DEFAULT_DIMENSIONS.PIP_WIDTH;\n    const height = miniplayer.offsetHeight || DEFAULT_DIMENSIONS.PIP_HEIGHT;\n\n    this.logger.debug(`Requesting PiP window: ${width}x${height}`);\n\n    // Request PiP window\n    const dpp = window.documentPictureInPicture;\n    if (!dpp) {\n      throw new PiPError('Document Picture-in-Picture API not available');\n    }\n\n    const ytdApp = this.ytdAppProvider.getApp();\n\n    this.miniPlayerContainer = document.querySelector(SELECTORS.MINIPLAYER_CONTAINER);\n    if (!this.miniPlayerContainer) {\n      throw new PiPError('miniplayer-container element not found');\n    }\n\n    const pipWindow = await dpp.requestWindow({ width, height });\n    this.pipWindowProvider.setWindow(pipWindow);\n\n    setTimeout(() => {\n      void this.asyncLock.withLock(async () => {\n        const win = this.pipWindowProvider.getWindow();\n        if (win?.closed) {\n          this.logger.warn('phantom window detected, closing');\n          this.pipWindowProvider.setWindow(null);\n          void this.close();\n          return;\n        }\n      });\n    }, TIMEOUTS.PHANTOM_WINDOW_CHECK);\n    pipWindow.addEventListener('pagehide', this.close);\n    this.logger.log('PiP window opened');\n\n    const pipDoc = pipWindow.document;\n\n    // Copy attributes and styles\n    DOMUtils.copyAttributes(document.documentElement, pipDoc.documentElement);\n    DOMUtils.copyAttributes(document.body, pipDoc.body);\n    StyleUtils.copyStyles(document, pipDoc);\n    StyleUtils.injectCSSFixes(pipDoc);\n\n    // Create ytd-app in PiP window\n    const pipApp = pipDoc.createElement(SELECTORS.YTD_APP);\n    DOMUtils.copyAttributes(ytdApp, pipApp);\n    pipDoc.body.appendChild(pipApp);\n\n    // Move mini player to PiP\n    this.placeholder = DOMUtils.createPlaceholder('mini_player_placeholder');\n    DOMUtils.insertPlaceholderBefore(miniplayer, this.placeholder);\n\n    pipApp.appendChild(miniplayer);\n\n    // Move container to draggable\n    const ytDraggable = pipDoc.querySelector(SELECTORS.YT_DRAGGABLE);\n    if (!ytDraggable) {\n      throw new PiPCriticalError('yt-draggable element not found');\n    }\n    ytDraggable.prepend(this.miniPlayerContainer);\n\n    DOMUtils.unwrap(ytDraggable);\n  }\n\n  /**\n   * Return player to main window\n   */\n  private async returnPlayerToMain(): Promise<void> {\n    this.logger.log('Returning player to main window');\n\n    if (!this.placeholder || !this.miniPlayerContainer) {\n      this.logger.warn('Placeholder or miniPlayerContainer not found', {\n        placeholder: this.placeholder,\n        miniPlayerContainer: this.miniPlayerContainer,\n      });\n      return;\n    }\n\n    // Run before-return cleanup (e.g. move context menu back)\n    if (this.onBeforeReturn) {\n      try {\n        await this.onBeforeReturn();\n      } catch (e) {\n        this.logger.error('Error in onBeforeReturn:', e);\n      }\n      this.onBeforeReturn = null;\n    }\n\n    try {\n      await this.movePlayerToMain();\n    } catch (error) {\n      this.logger.error('Error returning player to main window:', error);\n    }\n\n    this.pipWindowProvider.setWindow(null);\n  }\n\n  /**\n   * Move player back to main window\n   */\n  private async movePlayerToMain(): Promise<void> {\n    const miniplayer = this.miniPlayerController.getMiniplayer();\n    if (!this.placeholder || !this.miniPlayerContainer) {\n      return;\n    }\n\n    // Get player and save state\n    const player = this.playerManager.getPlayer();\n    this.playerManager.savePlayingState(player);\n\n    // Move mini player back\n    DOMUtils.restoreElementFromPlaceholder(miniplayer, this.placeholder);\n    this.placeholder = null;\n\n    // Move container back\n    const ytDraggable = document.querySelector(SELECTORS.YT_DRAGGABLE);\n    if (!ytDraggable) {\n      throw new PiPCriticalError('yt-draggable element not found');\n    }\n    ytDraggable.prepend(this.miniPlayerContainer);\n\n    if (!this.playerManager.getWasMiniPlayerActiveBeforePiP()) {\n      this.miniPlayerController.toggleMiniPlayer();\n\n      // Wait for main player\n      await this.playerManager.waitForMainPlayer();\n    }\n\n    // Remove mini player container\n    const miniplayerContainer = document.querySelector(SELECTORS.MINIPLAYER_CONTAINER);\n    if (miniplayerContainer) {\n      miniplayerContainer.remove();\n    }\n\n    // Restore playback state\n    await new Promise<void>((resolve) => {\n      setTimeout(async () => {\n        this.playerManager.restorePlayingState(player);\n        if (this.playerManager.getWasMiniPlayerActiveBeforePiP()) {\n          this.miniPlayerController.activateMiniPlayer();\n          await this.playerManager.waitForMiniPlayer().catch(() => {});\n        }\n        resolve();\n      });\n    });\n  }\n}\n","import { PiPManager } from '../core/PiPManager';\nimport { AppInitializationError } from '../errors/AppInitializationError';\nimport type { Logger } from '../logger';\nimport { LoggerFactory } from '../logger';\nimport { inject, injectable } from '../di';\n\n/**\n * Handles Media Session API integration\n */\n@injectable()\nexport class MediaSessionHandler {\n  private readonly logger: Logger;\n\n  constructor(\n    @inject(LoggerFactory) loggerFactory: LoggerFactory,\n    @inject(PiPManager) private readonly pipManager: PiPManager\n  ) {\n    this.logger = loggerFactory.create('MediaSessionHandler');\n  }\n\n  /**\n   * Initialize media session handler\n   */\n  public initialize(): void {\n    if (!('mediaSession' in navigator)) {\n      this.logger.warn('Media Session API not available');\n      return;\n    }\n\n    this.registerActionHandler();\n    this.logger.debug('Media session handler initialized');\n  }\n\n  /**\n   * Register enterpictureinpicture action handler\n   */\n  private registerActionHandler(): void {\n    if (!('mediaSession' in navigator)) {\n      return;\n    }\n\n    try {\n      // 'enterpictureinpicture' is a Chrome-specific Media Session action\n      navigator.mediaSession.setActionHandler('enterpictureinpicture', () => {\n        this.logger.log('Media session enterpictureinpicture action triggered');\n        this.pipManager.open().catch((e) => {\n          this.logger.error('Error opening PiP from media session:', e);\n        });\n      });\n      this.logger.debug('Media session action handler registered');\n    } catch (e) {\n      throw new AppInitializationError('Error registering media session action handler', e);\n    }\n  }\n}\n","/**\n * Utility for detecting YouTube client version and script version\n */\n\n/**\n * Get script version from Vite define (injected at build time)\n */\nfunction getScriptVersion(): string {\n  /* v8 ignore next -- @preserve */\n  return typeof SCRIPT_VERSION !== 'undefined' ? SCRIPT_VERSION : 'unknown';\n}\n\n/**\n * Get YouTube client version from window.ytcfg\n */\nfunction getYouTubeVersion(): string {\n  try {\n    const ytcfg = window.ytcfg;\n    if (ytcfg?.data_?.INNERTUBE_CONTEXT?.client?.clientVersion) {\n      return ytcfg.data_.INNERTUBE_CONTEXT.client.clientVersion;\n    }\n  } catch {\n    // Silently fail if version detection fails\n  }\n\n  return 'unknown';\n}\n\n/**\n * Get YouTube feature flags/experiments configuration from window.ytcfg\n */\nfunction getYouTubeFeatureFlags(): Record<string, unknown> {\n  try {\n    const ytcfg = window.ytcfg;\n\n    // EXPERIMENT_FLAGS - feature flags location\n    if (ytcfg?.data_?.EXPERIMENT_FLAGS) {\n      return ytcfg.data_.EXPERIMENT_FLAGS;\n    }\n\n    return {};\n  } catch {\n    // Silently fail if feature flags detection fails\n  }\n\n  return {};\n}\n\n/**\n * Extract browser name and version from navigator userAgent\n */\nfunction getBrowserVersion(): string {\n  try {\n    const userAgent = navigator.userAgent;\n    if (!userAgent) {\n      return 'unknown';\n    }\n\n    // Chrome (check before Edge as Edge UA contains \"Chrome\")\n    const chromeMatch = userAgent.match(/Chrome\\/([\\d.]+)/);\n    if (chromeMatch && !userAgent.includes('Edg/')) {\n      return `Chrome/${chromeMatch[1]}`;\n    }\n\n    // Edge\n    const edgeMatch = userAgent.match(/Edg\\/([\\d.]+)/);\n    if (edgeMatch) {\n      return `Edge/${edgeMatch[1]}`;\n    }\n\n    // Firefox\n    const firefoxMatch = userAgent.match(/Firefox\\/([\\d.]+)/);\n    if (firefoxMatch) {\n      return `Firefox/${firefoxMatch[1]}`;\n    }\n\n    // Safari (check Version first, then Safari)\n    const safariVersionMatch = userAgent.match(/Version\\/([\\d.]+).*Safari/);\n    if (safariVersionMatch && !userAgent.includes('Chrome')) {\n      return `Safari/${safariVersionMatch[1]}`;\n    }\n\n    // Fallback: return full user agent if no match\n    return userAgent;\n  } catch {\n    // Silently fail if browser version detection fails\n  }\n\n  return 'unknown';\n}\n\n/**\n * Get global metadata object with YouTube, script, browser versions, and feature flags\n */\nexport function getGlobalMetadata(): Record<string, unknown> {\n  const metadata: Record<string, unknown> = {\n    youtubeVersion: getYouTubeVersion(),\n    scriptVersion: getScriptVersion(),\n    browserVersion: getBrowserVersion(),\n  };\n\n  const featureFlags = getYouTubeFeatureFlags();\n  if (Object.keys(featureFlags).length > 0) {\n    metadata.youtubeFeatureFlags = featureFlags;\n  }\n\n  return metadata;\n}\n","import { Logger } from './logger';\nimport { createContainer } from './di/container-config';\nimport { PlayerManager } from './core/PlayerManager';\nimport { YtdAppProvider } from './core/YtdAppProvider';\nimport { MiniPlayerController } from './ui/MiniPlayerController';\nimport { MediaSessionHandler } from './handlers/MediaSessionHandler';\nimport { getGlobalMetadata } from './utils/VersionDetector';\n\nconst logger = Logger.getInstance('Main');\n\n/**\n * Initialize the application\n */\nasync function initializeApp(): Promise<void> {\n  Logger.setGlobalMetadata(getGlobalMetadata());\n\n  const container = createContainer();\n\n  logger.log('Initializing YouTube PiP application');\n\n  try {\n    const ytdAppProvider = container.get<YtdAppProvider>(YtdAppProvider);\n    await ytdAppProvider.initialize();\n\n    const playerManager = container.get<PlayerManager>(PlayerManager);\n    await playerManager.initialize();\n\n    const miniPlayerController = container.get<MiniPlayerController>(MiniPlayerController);\n    await miniPlayerController.initialize();\n\n    const mediaSessionHandler = container.get<MediaSessionHandler>(MediaSessionHandler);\n    mediaSessionHandler.initialize();\n\n    logger.log('YouTube PiP application initialized');\n  } catch (error) {\n    logger.error('YouTube PiP initialization failed', error);\n  }\n}\n\n// Initialize application when DOM is ready\nif (document.readyState === 'loading') {\n  document.addEventListener('DOMContentLoaded', () => void initializeApp());\n} else {\n  void initializeApp();\n}\n","import { Container } from './container';\nimport { LoggerFactory } from '../logger';\nimport { PlayerManager } from '../core/PlayerManager';\nimport { YtdAppProvider } from '../core/YtdAppProvider';\nimport { PipWindowProvider } from '../core/PipWindowProvider';\nimport { MiniPlayerController } from '../ui/MiniPlayerController';\nimport { NavigationHandler } from '../core/NavigationHandler';\nimport { ResizeTracker } from '../ui/ResizeTracker';\nimport { MenuObserver } from '../ui/MenuObserver';\nimport { ContextMenuHandler } from '../ui/ContextMenuHandler';\nimport { SeekHandler } from '../handlers/SeekHandler';\nimport { LikeButtonHandler } from '../handlers/LikeButtonHandler';\nimport { DocumentFocusHandler } from '../handlers/DocumentFocusHandler';\nimport { TitleSyncHandler } from '../handlers/TitleSyncHandler';\nimport { PiPWindowHandlers } from '../core/PiPWindowHandlers';\nimport { PiPManager } from '../core/PiPManager';\nimport { YtActionSender } from '../core/YtActionSender';\nimport { MediaSessionHandler } from '../handlers/MediaSessionHandler';\n\nexport function createContainer(): Container {\n  const container = new Container();\n\n  container.bind(LoggerFactory).toSelf();\n  container.bind(PlayerManager).toSelf();\n  container.bind(YtdAppProvider).toSelf();\n  container.bind(PipWindowProvider).toSelf();\n  container.bind(MiniPlayerController).toSelf();\n  container.bind(NavigationHandler).toSelf();\n  container.bind(ResizeTracker).toSelf();\n  container.bind(MenuObserver).toSelf();\n  container.bind(ContextMenuHandler).toSelf();\n  container.bind(SeekHandler).toSelf();\n  container.bind(YtActionSender).toSelf();\n  container.bind(LikeButtonHandler).toSelf();\n  container.bind(DocumentFocusHandler).toSelf();\n  container.bind(TitleSyncHandler).toSelf();\n  container.bind(PiPWindowHandlers).toSelf();\n  container.bind(PiPManager).toSelf();\n  container.bind(MediaSessionHandler).toSelf();\n\n  return container;\n}\n"],"names":["TIMEOUTS","DEFAULT_DIMENSIONS","EMBED_IFRAME_DEFAULTS","YOUTUBE_SHORT_BASE","COPY_PAYLOAD_QUERY","PLAYER_STATES","YT_EVENTS","YT_ACTION_NAMES","YT_LIKE_ACTIONS","WEB_PAGE_TYPES","COPY_MENU_INDICES","MOUSE_BUTTONS","paramMetadata","WeakMap","injectableClasses","WeakSet","inject","token","target","_key","index","existing","get","set","setParamMetadata","injectable","add","setInjectable","timestampFormatter","Intl","DateTimeFormat","year","month","day","hour","minute","second","hour12","_Logger","constructor","scope","this","getInstance","storageListenerAdded","enabled","checkDebugFlag","window","addEventListener","instance","instances","localStorage","getItem","setGlobalMetadata","metadata","globalMetadata","styled","fn","msgStyle","message","meta","ts","date","ms","String","getMilliseconds","padStart","format","replace","formatTimestamp","Date","escaped","args","push","Object","keys","length","log","console","bind","warn","error","debug","Map","Logger","LoggerFactory","create","__decorateClass","AppError","Error","cause","super","captureStackTrace","AppRuntimeError","name","Container","bindings","to","implementation","toSelf","toTransient","toInstance","resolutionStack","Set","tokenName","binding","has","chain","map","t","join","Ctor","paramTokens","getParamMetadata","paramCount","i","delete","unbind","logger","DOMUtils","createPlaceholder","commentText","document","createComment","insertPlaceholderBefore","element","placeholder","parent","parentNode","insertBefore","restoreElementFromPlaceholder","remove","copyAttributes","source","Array","from","attributes","forEach","attr","setAttribute","nodeName","nodeValue","e","copyViaTextarea","doc","text","el","createElement","value","style","cssText","body","appendChild","focus","select","ok","execCommand","unwrap","wrapper","firstChild","removeChild","waitForElementSelector","selector","timeout","targetWindow","Promise","resolve","reject","querySelector","observer","MutationObserver","_","obs","disconnect","observe","childList","subtree","setTimeout","closed","once","SELECTORS","AppInitializationError","PlayerManager","loggerFactory","player","wasPlaying","wasMiniPlayerActiveBeforePiP","initialize","getPlayer","getPlayerState","isPlaying","savePlayingState","setWasMiniPlayerActiveBeforePiP","getWasMiniPlayerActiveBeforePiP","restorePlayingState","playVideo","getVideoDataFromPlayer","getVideoData","getVideoId","videoData","videoId","video_id","getCurrentTime","Number","isNaN","Math","floor","getPlayerSize","getDebugInfo","getDebugText","waitForMainPlayer","waitForMiniPlayer","miniplayer","resetState","YtdAppProvider","app","notifyRenderer","err","getApp","getNotifyRenderer","__decorateParam","PipWindowProvider","pipWindow","setWindow","hasWindow","getWindow","MiniPlayerController","playerManager","ytdAppProvider","getMiniplayer","isVisible","activateMiniPlayer","ytdApp","fire","actionName","optionalAction","returnValue","toggleMiniPlayer","miniplayerIsActive","endpoint","watchEndpoint","NavigationHandler","pipWindowProvider","setupClickHandler","event","_a","closest","_b","href","url","URL","params","fromEntries","searchParams","preventDefault","state","commandMetadata","webCommandMetadata","webPageType","rootVe","v","playlistId","list","parseInt","playerParams","pp","entryTime","performance","now","dispatchEvent","PopStateEvent","cleanup","ResizeTracker","start","targetElement","ResizeObserver","entries","entry","width","contentRect","setInternalSize","setSize","call","Event","bubbles","stop","MenuObserver","removalObserver","runObservation","button","mutations","mutation","attributeName","isExpanded","getAttribute","playListContainer","currentHeight","outerHeight","resizeTo","outerWidth","display","attributeFilter","isConnected","CopyType","buildCopyPayload","currentTime","title","copyType","embedSize","VIDEO_URL","base","listPart","buildVideoUrlPayload","URL_AT_TIME","buildUrlAtTimePayload","EMBED","height","buildEmbedPayload","ContextMenuHandler","visibilityObserver","contextMenu","contextMenuPlaceholder","visibilitySubscribers","handleCopyClick","item","parentElement","items","querySelectorAll","prototype","indexOf","getCopyTypeForIndex","DEBUG_INFO","getCopyPayload","subscribeContextMenu","callback","notifyVisibility","visible","cb","startMonitoring","setupDismissalHandler","setupCopyHandler","isInMainWindow","simulateMainContextMenu","handleEvent","menuInPip","stopPropagation","mainApp","MouseEvent","cancelable","clientX","clientY","removeEventListener","SeekHandler","setupSeekHandler","pipDoc","progressBar","getDuration","seekTo","performSeek","rect","getBoundingClientRect","x","left","percent","max","min","duration","seekTime","toFixed","onMouseMove","moveEvent","onMouseUp","YtActionSender","sendLikeAction","actionType","resolveCommand","command","likeEndpoint","status","LikeButtonHandler","ytActionSender","toggleButton","buttonContainer","isLikeButton","childNodes","isDislikeButton","isPressed","DocumentFocusHandler","contextMenuHandler","isContextMenuOpen","unsubscribeContextMenu","onBodyClick","returnFocusToPlayerIfNeeded","onKeyUp","key","active","activeElement","_c","TitleSyncHandler","mutationObserver","video","syncTitle","data","setWindowsTitle","notify","newTitle","showNotificationCount","PiPWindowHandlers","resizeTracker","menuObserver","seekHandler","likeButtonHandler","navigationHandler","documentFocusHandler","titleSyncHandler","StyleUtils","copyStyles","sourceDoc","targetDoc","styles","node","head","cloneNode","injectCSSFixes","styleFix","textContent","AsyncLock","locked","queue","withLock","acquire","release","queueSize","next","shift","remaining","PiPError","PiPCriticalError","PiPManager","miniPlayerController","pipWindowHandlers","miniPlayerContainer","onBeforeReturn","asyncLock","close","async","returnPlayerToMain","catch","isOpen","open","movePlayerToPIP","result","offsetWidth","offsetHeight","dpp","documentPictureInPicture","requestWindow","win","documentElement","pipApp","ytDraggable","prepend","movePlayerToMain","miniplayerContainer","MediaSessionHandler","pipManager","navigator","registerActionHandler","mediaSession","setActionHandler","getYouTubeVersion","ytcfg","data_","INNERTUBE_CONTEXT","client","clientVersion","getBrowserVersion","userAgent","chromeMatch","match","includes","edgeMatch","firefoxMatch","safariVersionMatch","getGlobalMetadata","youtubeVersion","scriptVersion","browserVersion","featureFlags","EXPERIMENT_FLAGS","getYouTubeFeatureFlags","youtubeFeatureFlags","initializeApp","container","createContainer","readyState"],"mappings":"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"}