Greasy Fork

Greasy Fork is available in English.

抖音全屏优化&自动清屏-抖音网页版优化

抖音网页版全屏优化 16:9 显示,自动清屏/自动选择最高分辨率/隐藏控制栏/侧边栏/搜索栏-抖音网页版优化

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         抖音全屏优化&自动清屏-抖音网页版优化
// @version      1.2.2
// @author       akace
// @description  抖音网页版全屏优化 16:9 显示,自动清屏/自动选择最高分辨率/隐藏控制栏/侧边栏/搜索栏-抖音网页版优化
// @icon         
// @match        *://*.douyin.com/*
// @match        *://*.iesdouyin.com/*
// @exclude      *://lf-zt.douyin.com*
// @license      GPL-3.0 License
// @run-at       document-start
// @grant    GM_registerMenuCommand
// @grant    GM_unregisterMenuCommand
// @namespace    http://greasyfork.icu/scripts/508672
// @supportURL   http://greasyfork.icu/users/1368522
// @homepageURL  https://github.com/akFace/userScript
// ==/UserScript==

(() => {
  "use strict";
  const CONFIG = {
    side: "J_side_bar_show",
    clearTime: "J_clsear_time",
  };
  const ConstClearTime = 2000;
  let clearTs = parseInt(localStorage.getItem(CONFIG.clearTime));
  if (clearTs !== 0 && !clearTs) {
    clearTs = ConstClearTime;
  }
  const state = {
    showSide: localStorage.getItem(CONFIG.side) === "true",
    clearTime: clearTs,
  };

  GM_registerMenuCommand(
    state.showSide ? "💡隐藏侧边栏" : "💡显示侧边栏",
    () => {
      state.showSide = !state.showSide;
      localStorage.setItem(CONFIG.side, state.showSide);
      updateStyle();
      window.location.reload();
    }
  );
  GM_registerMenuCommand(`⏳设置自动清屏倒计时,0=关闭自动清屏`, () => {
    const input = prompt(
      `自动清屏倒计时长, 当前值: ${state.clearTime}毫秒=${
        state.clearTime / 1000
      }秒,设置0=关闭自动清屏,(默认${ConstClearTime / 1000}秒)`,
      state.clearTime
    );
    const value = parseInt(input);
    localStorage.setItem(CONFIG.clearTime, value);
    state.clearTime = value;
    alert(`当前设置:自动清屏倒计时长 ${value}毫秒=${value / 1000}秒`);
    window.location.reload();
  });
  GM_registerMenuCommand("📣问题反馈", () => {
    const r = confirm("确定要创建问题反馈吗?");
    if (r == true) {
      window.open("https://github.com/akFace/userScript/issues/new");
    }
  });

  // 更新样式
  updateStyle();
  function updateStyle() {
    let style_Add = document.createElement("style");
    let hideSide = `.isCssFullScreen .positionBox{opacity: 0 !important;}
        .isCssFullScreen .xgplayer-inactive .positionBox{opacity: 0 !important;}`;
    let showSide = `.slider-video .positionBox{visibility: visible !important; opacity: 1 !important; display: flex !important;}
    .slider-video .positionBox .immersive-player-switch-on-hide-interaction-area{visibility: visible !important; opacity: 1 !important; display: flex !important;}`;
    style_Add.innerHTML = `.isCssFullScreen .xg-video-container{bottom: 0 !important;}
    .isCssFullScreen .xgplayer-inactive .xg-inner-controls{opacity: 0 !important;}
    .isCssFullScreen .xg-inner-controls{opacity: 1;}
    .isCssFullScreen .positionBox{opacity: 1;}
    .isCssFullScreen .immersive-player-switch-on-hide-video-info.xgplayer-inactive .xgplayer-controls{background: none !important;}
    .isCssFullScreen .xgplayer-inactive.xgplayer.xgplayer-pc.xgplayer-pc .xgplayer-controls{background: none !important;}
    #video-info-wrap .video-info-mask {background: none !important;}
    ${state.showSide ? showSide : hideSide}
    `;
    if (document.head) {
      document.head.appendChild(style_Add);
    } else {
      let timer = setInterval(function () {
        if (document.head) {
          document.head.appendChild(style_Add);
          clearInterval(timer);
        }
      });
    }
  }

  document.addEventListener("DOMContentLoaded", () => {
    initClear();
    switchMaxResolution();
  });

  // 设置高分辨率
  const switchMaxResolution = () => {
    // 循环设置
    const loopSetIt = function () {
      // 找不到分辨率就先循环查询,过滤掉直播页面
      if (window.location.href.indexOf("live") === -1) {
        let setTimeoutFlg = setTimeout(() => {
          switchMaxResolution();
          clearTimeout(setTimeoutFlg);
        }, 2000);
      }
    };
    // 获取当前正在播放的节点
    const activeVideo = document.querySelector(
      "[data-e2e='feed-active-video']"
    );
    if (!activeVideo) {
      loopSetIt();
      return;
    }
    const resolutionOptions = Array.from(
      activeVideo.querySelectorAll(".xgplayer-playing div.virtual > div.item")
    ).filter(
      (el) =>
        el.textContent &&
        /[4K|2K|1080P|720P|540P|智能]/i.test(el.textContent.trim())
    );

    if (resolutionOptions.length > 0) {
      const isHight = resolutionOptions.length < 4;
      // 按照优先级排序,选择最高分辨率(假设顺序为:4k > 2k > 1080P > 720P > 540P > 智能),不区分大小写
      const priorityOrder = ["4K", "2K", "1080P", "720P", "540P", "智能"];
      resolutionOptions.sort((a, b) => {
        const aPriority = priorityOrder.indexOf(
          a.textContent.trim().toLowerCase()
        );
        const bPriority = priorityOrder.indexOf(
          b.textContent.trim().toLowerCase()
        );
        return aPriority - bPriority;
      });

      // 预设小于1080p的重新设置(假设最高分辨率只有1080p)
      if (isHight) {
        loopSetIt();
      }
      // 获取优先级最高的选项
      const highestResolutionOption = resolutionOptions[0];

      // 如果该选项未被选中,则点击它
      if (!highestResolutionOption.classList.contains("selected")) {
        highestResolutionOption.click();
      } else {
        console.log("当前已是最高清晰度");
      }
    } else {
      loopSetIt();
      console.error("未找到任何分辨率选项");
    }
  };

  // 清屏操作
  let initClear = function () {
    let isCleared = false;
    let lastClearTime = 0;
    let minClearInterval = state.clearTime; // 最小清屏间隔时间(毫秒)
    if (minClearInterval === 0) {
      return;
    }
    // 添加监听器以检测页面变化
    let observer = new MutationObserver(function (mutations) {
      mutations.forEach(function (mutation) {
        let currentTime = Date.now();
        if (currentTime - lastClearTime >= minClearInterval) {
          clearScreen();
          lastClearTime = currentTime;
        }
      });
    });

    // 开始监听 DOM 变化
    observer.observe(document.body, { childList: true, subtree: true });

    function clearScreen() {
      try {
        let controlsDom = document.querySelectorAll(
          '[data-e2e="feed-active-video"]'
        );
        let switchNoe = "xg-switch"; // 未清屏状态
        let switchChecked = "xg-switch xg-switch-checked"; // 清屏状态
        if (controlsDom.length) {
          let qingping = controlsDom[0].getElementsByClassName("xg-switch");
          if (qingping.length && qingping[0].className == switchNoe) {
            qingping[0].click();
            isCleared = true;
          } else if (
            qingping.length &&
            qingping[0].className == switchChecked
          ) {
            if (isCleared) {
              isCleared = false;
            }
          }
        }
      } catch (error) {
        console.error("清屏操作失败:", error);
      }
    }
  };
})();