MissAV去广告、影院模式

测试一下聊天室

  1. // ==UserScript==
  2. // @run-at document-start
  3. // @name MissAV去广告、影院模式
  4. // @description 测试一下聊天室
  5. // @icon https://missav.ws/img/favicon.ico
  6. // @namespace loadingi.local
  7. // @version 5.0.3.1
  8. // @author chris
  9. // @match *://*.missav.ws/*
  10. // @match *://*.missav.ai/*
  11. // @match *://*.missav123.com/*
  12. // @match *://*/view_video.php?viewkey=*
  13. // @match *://fuliba2025.net/*
  14. // @grant GM_setValue
  15. // @grant GM_getValue
  16. // @grant unsafeWindow
  17. // @grant GM_xmlhttpRequest
  18. // @compatible chrome
  19. // @compatible firefox
  20. // @compatible edge
  21. // @compatible safari
  22. // @license GPL-3.0-only
  23. // @require https://code.jquery.com/jquery-3.6.0.min.js
  24. // ==/UserScript==
  25.  
  26. (function() {
  27. 'use strict';
  28. /**
  29. * 主要配置对象
  30. * - selectors: DOM元素选择器配置
  31. * - styles: 样式配置
  32. * - player: 播放器功能配置
  33. */
  34. const Config = {
  35. // DOM 选择器
  36. selectors: {
  37. player: {
  38. container: '.relative.-mx-4.sm\\:m-0.-mt-6',
  39. wrapper: '.aspect-w-16.aspect-h-9',
  40. video: 'video#player',
  41. progress: 'div.sm\\:hidden.flex.justify-between.-mx-4.px-4.pt-3.pb-1.bg-black',
  42. abLoop: 'div.flex.items-center.flex-nowrap.leading-5',
  43. abLoopControls: '.theater-controls-abloop',
  44. genres: '.absolute.bottom-1.left-1.rounded-lg.px-2.py-1.text-xs.text-nord5.bg-blue-800.bg-opacity-75',
  45. uncensoredLink: "a[id^='option-menu-item'][href*='uncensored']",
  46. qualityOptions: '.plyr__menu__container [data-plyr="quality"]'
  47. },
  48. ads: {
  49. scripts: [
  50. "script[src*='app.1aad5686.js']",
  51. "script[src*='inpage.push.js']",
  52. "script[src*='hartattenuate.com']",
  53. "script[src*='ads']",
  54. "script[src*='pop']",
  55. "script[src*='banner']",
  56. "script[src*='htmlAds']",
  57. "script[src*='popAds']",
  58. "script[src*='bannerAds']",
  59. "script[src*='adsConfig']"
  60. ],
  61. elements: [
  62. // 'div.sm\\:container.mx-auto.mb-5.px-4',
  63. 'ul.mb-4.list-none.text-nord14.grid.grid-cols-2.gap-2',
  64. 'div.relative.ml-4',
  65. 'div.root--ujvuu',
  66. 'div.under_player',
  67. 'div.space-y-5.mb-5',
  68. 'div[class^="rootContent--"]',
  69. 'div[class^="fixed right-2 bottom-2"]',
  70. 'div[class^="space-y-6 mb-6"]',
  71. 'div.space-y-2.mb-4.ml-4.list-disc.text-nord14',
  72. 'div[id*="ads"]',
  73. 'div[id*="banner"]',
  74. 'div[class*="ads"]',
  75. 'div[class*="banner"]',
  76. '.ad-container',
  77. '#ad-container'
  78. ],
  79. scriptPatterns: ['htmlAds', 'popAds', 'bannerAds', 'adsConfig']
  80. }
  81. },
  82. styles: {
  83. button: {
  84. base: {
  85. backgroundColor: '#222',
  86. borderRadius: '15px',
  87. borderColor: 'black',
  88. borderWidth: '1px',
  89. color: 'burlywood',
  90. cursor: 'pointer',
  91. transition: 'all 0.3s ease',
  92. outline: 'none',
  93. minWidth: '80px',
  94. padding: '2px 4px',
  95. marginBottom: '10px'
  96. },
  97. hover: {
  98. backgroundColor: '#333'
  99. }
  100. },
  101. theaterMode: `
  102. .theater-overlay {
  103. position: fixed;
  104. top: 0;
  105. left: 0;
  106. width: 100vw;
  107. height: 100vh;
  108. background: rgba(0, 0, 0, 0.95);
  109. z-index: 9998;
  110. display: none;
  111. }
  112. .theater-mode-container {
  113. position: fixed !important;
  114. top: 0 !important;
  115. left: 0 !important;
  116. width: 100vw !important;
  117. height: 100vh !important;
  118. transform: none !important;
  119. z-index: 9999 !important;
  120. margin: 0 !important;
  121. padding: 0 !important;
  122. display: flex !important;
  123. align-items: center !important;
  124. justify-content: center !important;
  125. background: transparent !important;
  126. pointer-events: auto !important;
  127. }
  128. .theater-mode-container .aspect-w-16.aspect-h-9 {
  129. position: relative !important;
  130. width: 100vw !important;
  131. max-width: none !important;
  132. height: 100vh !important;
  133. margin: 0 auto !important;
  134. pointer-events: auto !important;
  135. }
  136. .theater-mode-container video {
  137. position: absolute !important;
  138. top: 0 !important;
  139. left: 0 !important;
  140. width: 100% !important;
  141. height: 100% !important;
  142. object-fit: contain !important;
  143. pointer-events: auto !important;
  144. }
  145. .theater-mode-container * {
  146. max-width: none !important;
  147. max-height: none !important;
  148. pointer-events: auto !important;
  149. }
  150. .theater-mode-container .plyr__controls {
  151. position: fixed !important;
  152. bottom: 0 !important;
  153. left: 0 !important;
  154. width: 100% !important;
  155. z-index: 10000 !important;
  156. background: transparent !important;
  157. padding: 10px !important;
  158. opacity: 1 !important;
  159. visibility: visible !important;
  160. display: flex !important;
  161. }
  162. .fixed.z-max.w-full.bg-gradient-to-b.from-darkest {
  163. z-index: 1 !important;
  164. }
  165. .theater-mode-container .fixed.z-max.w-full.bg-gradient-to-b.from-darkest {
  166. display: none !important;
  167. }
  168. .theater-mode-container .plyr__time {
  169. display: inline-block !important;
  170. color: white !important;
  171. opacity: 1 !important;
  172. visibility: visible !important;
  173. }
  174. .theater-controls-progress {
  175. position: fixed !important;
  176. bottom: 104px !important;
  177. z-index: 10000 !important;
  178. background: transparent !important;
  179. padding: 10px !important;
  180. width: 100% !important;
  181. max-width: none !important;
  182. pointer-events: auto !important;
  183. }
  184. .theater-controls-abloop {
  185. position: fixed !important;
  186. bottom: 52px !important;
  187. left: 0px !important;
  188. width: 100% !important;
  189. z-index: 10000 !important;
  190. padding: 10px !important;
  191. pointer-events: auto !important;
  192. }
  193. .theater-mode-container .plyr__controls__item.plyr__volume {
  194. width: 40px !important;
  195. }
  196. .theater-mode-container .plyr__controls__item[data-plyr="rewind"],
  197. .theater-mode-container .plyr__controls__item[data-plyr="fast-forward"],
  198. .theater-mode-container .plyr__control[data-plyr="settings"],
  199. .theater-mode-container .plyr__controls__item[data-plyr="pip"],
  200. .theater-mode-container .plyr__controls__item[data-plyr="fullscreen"] {
  201. display: none !important;
  202. }
  203. .theater-mode-container ~ div .theater-mode-button,
  204. .theater-mode-container ~ div .ab-loop-button {
  205. background-color: rgba(34, 34, 34, 0.5) !important;
  206. border-color: rgba(0, 0, 0, 0.5) !important;
  207. }
  208. .theater-mode-container ~ div .theater-mode-button:hover,
  209. .theater-mode-container ~ div .ab-loop-button:hover {
  210. background-color: rgba(51, 51, 51, 0.7) !important;
  211. }
  212. `
  213. },
  214. player: {
  215. autoHighestQuality: true,
  216. preventFocusPause: true,
  217. autoSwitchUncensored: true,
  218. hideVideoGenres: true
  219. }
  220. };
  221. /**
  222. * 工具类
  223. * - debounce: 函数防抖
  224. * - createButton: 创建自定义按钮
  225. * - addStyle: 添加自定义样式
  226. */
  227. const Utils = {
  228. debounce(func, wait) {
  229. let timeout;
  230. return function(...args) {
  231. clearTimeout(timeout);
  232. timeout = setTimeout(() => func.apply(this, args), wait);
  233. };
  234. },
  235. createButton(text, onClick) {
  236. const button = document.createElement('button');
  237. Object.assign(button.style, Config.styles.button.base);
  238. button.innerText = text;
  239. button.addEventListener('mouseover', () => Object.assign(button.style, Config.styles.button.hover));
  240. button.addEventListener('mouseout', () => Object.assign(button.style, { backgroundColor: Config.styles.button.base.backgroundColor }));
  241. button.addEventListener('click', onClick);
  242. return button;
  243. },
  244. addStyle(css) {
  245. const style = document.createElement('style');
  246. style.textContent = css;
  247. document.head.appendChild(style);
  248. }
  249. };
  250. /**
  251. * 广告拦截器模块
  252. * - 移除广告脚本和元素
  253. * - 拦截动态加载的广告
  254. * - 使用MutationObserver监听DOM变化
  255. */
  256. const AdBlocker = {
  257. init() {
  258. this.blockAds = Utils.debounce(this.blockAds.bind(this), 100);
  259. this.blockAds();
  260. this.setupMutationObserver();
  261. this.interceptDynamicScripts();
  262. },
  263. blockAds() {
  264. const { scripts, elements, scriptPatterns } = Config.selectors.ads;
  265. // 移除广告脚本和元素
  266. [...scripts, ...elements].forEach(selector => {
  267. document.querySelectorAll(selector).forEach(el => el?.remove());
  268. });
  269. // 移除 iframes
  270. document.querySelectorAll('iframe').forEach(iframe => iframe.remove());
  271. // 移除匹配模式的脚本
  272. const scriptPattern = new RegExp(scriptPatterns.join('|'));
  273. document.querySelectorAll('script').forEach(script => {
  274. if (scriptPattern.test(script.innerText)) {
  275. script.remove();
  276. }
  277. });
  278. },
  279. setupMutationObserver() {
  280. const observer = new MutationObserver(
  281. Utils.debounce(() => this.blockAds(), 100)
  282. );
  283. observer.observe(document.body, {
  284. childList: true,
  285. subtree: true
  286. });
  287. },
  288. interceptDynamicScripts() {
  289. const scriptPattern = new RegExp(Config.selectors.ads.scriptPatterns.join('|'));
  290. const originalCreateElement = document.createElement.bind(document);
  291. document.createElement = function(tagName) {
  292. const element = originalCreateElement(tagName);
  293. if (tagName.toLowerCase() === 'script') {
  294. const originalSetAttribute = element.setAttribute.bind(element);
  295. element.setAttribute = function(name, value) {
  296. if (name === 'src' && scriptPattern.test(value)) {
  297. return; // 阻止加载广告脚本
  298. }
  299. return originalSetAttribute(name, value);
  300. };
  301. }
  302. return element;
  303. };
  304. }
  305. };
  306. /**
  307. * 播放器增强模块
  308. * 主要功能:
  309. * - 影院模式
  310. * - 进度控制
  311. * - AB循环
  312. * - 自动最高画质
  313. * - 防止失焦暂停
  314. * - 隐藏视频类型标签
  315. */
  316. const PlayerEnhancer = {
  317. init() {
  318. this.setupPlayer();
  319. this.createControls();
  320. Utils.addStyle(Config.styles.theaterMode);
  321. // 新增功能初始化
  322. if(Config.player.hideVideoGenres) {
  323. this.hideVideoGenres();
  324. }
  325. if(Config.player.autoSwitchUncensored) {
  326. this.setupAutoUncensored();
  327. }
  328. if(Config.player.autoHighestQuality) {
  329. this.setupAutoHighestQuality();
  330. }
  331. if(Config.player.preventFocusPause) {
  332. this.preventFocusPause();
  333. }
  334. },
  335. setupPlayer() {
  336. // 移除点击事件
  337. document.querySelectorAll('[\\@click="pop()"]').forEach(el => {
  338. el.removeAttribute('@click');
  339. });
  340. // 移除窗口失焦暂停
  341. const aspectElements = document.getElementsByClassName('aspect-w-16 aspect-h-9');
  342. if (aspectElements[11]) {
  343. aspectElements[11].removeAttribute('@click');
  344. aspectElements[11].removeAttribute('@keyup.space.window');
  345. }
  346. },
  347. createControls() {
  348. const container = document.createElement('div');
  349. Object.assign(container.style, {
  350. position: 'fixed',
  351. top: '14px',
  352. right: '98px',
  353. zIndex: '10001',
  354. display: 'flex',
  355. flexDirection: 'row',
  356. gap: '10px'
  357. });
  358. // 创建影院模式按钮
  359. const theaterButton = Utils.createButton('影院模式', () => this.toggleTheaterMode());
  360. theaterButton.className = 'theater-mode-button';
  361. container.appendChild(theaterButton);
  362. // 创建进度控制按钮
  363. const progressButton = Utils.createButton('进度控制', () => this.toggleProgressControls());
  364. progressButton.className = 'progress-control-button';
  365. progressButton.style.display = 'none'; // 初始隐藏
  366. container.appendChild(progressButton);
  367. // 创建 AB 循环按钮
  368. const abLoopButton = Utils.createButton('A/B', () => this.toggleABLoopControls());
  369. abLoopButton.className = 'ab-loop-button';
  370. abLoopButton.style.display = 'none';
  371. container.appendChild(abLoopButton);
  372. document.body.appendChild(container);
  373. },
  374. toggleTheaterMode() {
  375. const { player } = Config.selectors;
  376. const playerContainer = document.querySelector(player.container);
  377. const progress = document.querySelector(player.progress);
  378. const abLoop = document.querySelector(player.abLoop);
  379. let overlay = document.querySelector('.theater-overlay');
  380. if (!overlay) {
  381. overlay = document.createElement('div');
  382. overlay.className = 'theater-overlay';
  383. document.body.appendChild(overlay);
  384. }
  385. const isTheaterMode = overlay.style.display === 'none' || overlay.style.display === '';
  386. const theaterButton = document.querySelector('.theater-mode-button');
  387. const abLoopButton = document.querySelector('.ab-loop-button');
  388. const progressButton = document.querySelector('.progress-control-button');
  389. if (theaterButton) {
  390. theaterButton.innerText = isTheaterMode ? '关闭影院' : '影院模式';
  391. }
  392. if (abLoopButton) {
  393. abLoopButton.style.display = isTheaterMode ? 'block' : 'none';
  394. }
  395. if (progressButton) {
  396. progressButton.style.display = isTheaterMode ? 'block' : 'none';
  397. }
  398. if (isTheaterMode) {
  399. this.enterTheaterMode(overlay, playerContainer, progress, abLoop);
  400. } else {
  401. this.exitTheaterMode(overlay, playerContainer, progress, abLoop);
  402. }
  403. },
  404. toggleProgressControls() {
  405. const progress = document.querySelector('.theater-controls-progress');
  406. const progressButton = document.querySelector('.progress-control-button');
  407. if (progress) {
  408. const isVisible = progress.style.display !== 'none';
  409. progress.style.display = isVisible ? 'none' : 'flex';
  410. if (progressButton) {
  411. progressButton.innerText = isVisible ? '显示进度' : '隐藏进度';
  412. }
  413. }
  414. },
  415. enterTheaterMode(overlay, playerContainer, progress, abLoop) {
  416. overlay.style.display = 'block';
  417. if (playerContainer) {
  418. playerContainer.classList.add('theater-mode-container');
  419. this.adjustParentContainers(playerContainer);
  420. }
  421. if (progress) {
  422. progress.classList.add('theater-controls-progress');
  423. // 默认显示进度条
  424. progress.style.display = 'flex';
  425. const progressButton = document.querySelector('.progress-control-button');
  426. if (progressButton) {
  427. progressButton.innerText = '隐藏进度';
  428. }
  429. }
  430. if (abLoop) {
  431. abLoop.classList.add('theater-controls-abloop');
  432. abLoop.style.display = 'none';
  433. }
  434. document.addEventListener('keydown', this.handleEscKey);
  435. },
  436. exitTheaterMode(overlay, playerContainer, progress, abLoop) {
  437. overlay.style.display = 'none';
  438. if (playerContainer) {
  439. playerContainer.classList.remove('theater-mode-container');
  440. this.resetParentContainers(playerContainer);
  441. }
  442. if (progress) {
  443. progress.classList.remove('theater-controls-progress');
  444. }
  445. if (abLoop) {
  446. abLoop.classList.remove('theater-controls-abloop');
  447. }
  448. document.removeEventListener('keydown', this.handleEscKey);
  449. },
  450. adjustParentContainers(element) {
  451. let parent = element.parentElement;
  452. while (parent && parent !== document.body) {
  453. Object.assign(parent.style, {
  454. maxWidth: 'none',
  455. maxHeight: 'none',
  456. overflow: 'visible',
  457. zIndex: 'auto'
  458. });
  459. parent = parent.parentElement;
  460. }
  461. },
  462. resetParentContainers(element) {
  463. let parent = element.parentElement;
  464. while (parent && parent !== document.body) {
  465. ['maxWidth', 'maxHeight', 'overflow', 'zIndex'].forEach(prop => {
  466. parent.style.removeProperty(prop);
  467. });
  468. parent = parent.parentElement;
  469. }
  470. },
  471. handleEscKey(e) {
  472. if (e.key === 'Escape') {
  473. PlayerEnhancer.toggleTheaterMode();
  474. }
  475. },
  476. toggleABLoopControls() {
  477. const abLoopControls = document.querySelector(Config.selectors.player.abLoopControls);
  478. const abLoopButton = document.querySelector('.ab-loop-button');
  479. if (abLoopControls) {
  480. const isVisible = abLoopControls.style.display !== 'none';
  481. abLoopControls.style.display = isVisible ? 'none' : 'flex';
  482. if (abLoopButton) {
  483. abLoopButton.innerText = isVisible ? 'A/B' : 'no A/B';
  484. }
  485. }
  486. },
  487. // 隐藏视频类型
  488. hideVideoGenres() {
  489. const genresElements = document.querySelectorAll(Config.selectors.player.genres);
  490. genresElements.forEach(el => el.style.display = 'none');
  491. },
  492. // 自动切换无码版本beta
  493. // setupAutoUncensored() {
  494. // const uncensoredLink = document.querySelector(Config.selectors.player.uncensoredLink);
  495. // if(uncensoredLink) {
  496. // uncensoredLink.click();
  497. // }
  498. // },
  499. // 自动设置最高画质
  500. setupAutoHighestQuality() {
  501. const setHighestQuality = () => {
  502. const player = unsafeWindow.player;
  503. if(!player?.config?.quality?.options) return;
  504. const maxQuality = Math.max(...player.config.quality.options);
  505. player.quality = maxQuality;
  506. player.config.quality.default = maxQuality;
  507. player.config.quality.selected = maxQuality;
  508. };
  509. // 等待播放器加载完成
  510. const checkPlayer = setInterval(() => {
  511. if(unsafeWindow.player) {
  512. setHighestQuality();
  513. clearInterval(checkPlayer);
  514. }
  515. }, 100);
  516. },
  517. // 防止失焦暂停
  518. preventFocusPause() {
  519. document.addEventListener('visibilitychange', () => {
  520. const player = unsafeWindow.player;
  521. if(document.hidden && player?.playing) {
  522. player.play();
  523. }
  524. });
  525. // 移除原有的失焦暂停事件
  526. const playerContainer = document.querySelector(Config.selectors.player.container);
  527. if(playerContainer) {
  528. playerContainer.removeAttribute('@keyup.space.window');
  529. }
  530. }
  531. };
  532.  
  533. // ↑----------------------------------------------------------------------
  534.  
  535.  
  536.  
  537.  
  538.  
  539.  
  540. // 主函数------------------------------------------------------------------
  541.  
  542.  
  543. //
  544.  
  545. /**
  546. * 主程序入口
  547. * - 检测 missav 网站并开启额外功能
  548. */
  549. const App = {
  550. init() {
  551. // 使用 MutationObserver 监听 DOM 变化
  552. const observer = new MutationObserver((mutations, obs) => {
  553. // 当 body 元素存在时初始化
  554. if (document.body) {
  555. obs.disconnect(); // 停止观察
  556. this.initWithDomainCheck();
  557. }
  558. });
  559.  
  560. // 如果 body 已存在则直接初始化
  561. if (document.body) {
  562. this.initWithDomainCheck();
  563. } else {
  564. // 否则开始观察 DOM 变化
  565. observer.observe(document.documentElement, {
  566. childList: true,
  567. subtree: true
  568. });
  569. }
  570. },
  571.  
  572.  
  573. };
  574.  
  575. // 启动程序
  576. if (document.readyState === 'loading') {
  577. document.addEventListener('DOMContentLoaded', () => App.init());
  578. } else {
  579. App.init();
  580. }
  581. })();

QingJ © 2025

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