画中画

在任意网站开启画中画模式,使视频漂浮在所有窗口最上层

  1. // ==UserScript==
  2. // @name 画中画
  3. // @name:en Pic-in-pic
  4. // @namespace https://github.com/zaaack/tools/
  5. // @description 在任意网站开启画中画模式,使视频漂浮在所有窗口最上层
  6. // @description:en Enable picture-in-picture mode in any video site, make video float on top window
  7. // @version 1.0
  8. // @author zaaack
  9. // @include *
  10. // @grant unsafeWindow
  11. // @run-at context-menu
  12. // @homepageURL https://github.com/zaaack/tools/blob/master/tampermonkey/pic-in-pic/README.md
  13. // @supportURL https://github.com/zaaack/tools/issues
  14. // ==/UserScript==
  15.  
  16. ;(() => {
  17. // fork from https://github.com/GoogleChromeLabs/picture-in-picture-chrome-extension/blob/20288da7c56b525f2c21129df58d45b027d4f328/src/script.js#L2
  18. //
  19. //
  20. // Copyright 2018 Google LLC
  21. //
  22. // Licensed under the Apache License, Version 2.0 (the "License");
  23. // you may not use this file except in compliance with the License.
  24. // You may obtain a copy of the License at
  25. //
  26. // http://www.apache.org/licenses/LICENSE-2.0
  27. //
  28. // Unless required by applicable law or agreed to in writing, software
  29. // distributed under the License is distributed on an "AS IS" BASIS,
  30. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  31. // See the License for the specific language governing permissions and
  32. // limitations under the License.
  33.  
  34. function findLargestPlayingVideo() {
  35. const videos = Array.from(document.querySelectorAll('video'))
  36. .filter((video) => video.readyState != 0)
  37. .filter((video) => video.disablePictureInPicture == false)
  38. .sort((v1, v2) => {
  39. const v1Rect = v1.getClientRects()[0] || { width: 0, height: 0 }
  40. const v2Rect = v2.getClientRects()[0] || { width: 0, height: 0 }
  41. return v2Rect.width * v2Rect.height - v1Rect.width * v1Rect.height
  42. })
  43.  
  44. if (videos.length === 0) {
  45. return
  46. }
  47.  
  48. return videos[0]
  49. }
  50.  
  51. async function requestPictureInPicture(video) {
  52. await video.requestPictureInPicture()
  53. video.setAttribute('__pip__', true)
  54. video.addEventListener(
  55. 'leavepictureinpicture',
  56. (event) => {
  57. video.removeAttribute('__pip__')
  58. },
  59. { once: true }
  60. )
  61. new ResizeObserver(maybeUpdatePictureInPictureVideo).observe(video)
  62. }
  63.  
  64. function maybeUpdatePictureInPictureVideo(entries, observer) {
  65. const observedVideo = entries[0].target
  66. if (!document.querySelector('[__pip__]')) {
  67. observer.unobserve(observedVideo)
  68. return
  69. }
  70. const video = findLargestPlayingVideo()
  71. if (video && !video.hasAttribute('__pip__')) {
  72. observer.unobserve(observedVideo)
  73. requestPictureInPicture(video)
  74. }
  75. }
  76.  
  77. ;(async () => {
  78. const video = findLargestPlayingVideo()
  79. if (!video) {
  80. return
  81. }
  82. if (video.hasAttribute('__pip__')) {
  83. document.exitPictureInPicture()
  84. return
  85. }
  86. await requestPictureInPicture(video)
  87. // chrome.runtime.sendMessage({ message: 'enter' })
  88. })()
  89. })()

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址