麻豆传媒

🚀新功能上线,不浪费一秒,会员随时停启💰。付费内容无限次观看、便捷下载和即时分享链接。内容自动展开和完全无广告的环境。🚀

Від 06.08.2025. Дивіться остання версія.

// ==UserScript==
// @name         麻豆传媒
// @name:zh-TW   麻豆傳媒
// @namespace    http://tampermonkey.net/
// @version      v1.7
// @description  🚀新功能上线,不浪费一秒,会员随时停启💰。付费内容无限次观看、便捷下载和即时分享链接。内容自动展开和完全无广告的环境。🚀
// @description:zh-TW   🚀新功能上線,不浪費一秒,會員隨時停啟💰。付費內容無限次觀看、便捷下載和即時分享連結受益於,內容自動展開和完全無廣告的環境。🚀
// @author            lx
// @match        *://h5dfsg.anwangjd1.com/*
// @match        *://d1ojl4419qh989.cloudfront.net/*
// @exclude			  *://www.haijiaom.cc/*
// @grant             unsafeWindow
// @grant             GM_addStyle
// @grant             GM_getValue
// @grant             GM_setValue
// @grant             GM_xmlhttpRequest
// @connect 		  haijiao.live
// @charset		      UTF-8
// @run-at 			  document-start
// @antifeature       payment
// @require           https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.0/jquery.min.js
// @require 		  https://cdnjs.cloudflare.com/ajax/libs/hls.js/1.5.8/hls.min.js
// @require 		  https://cdnjs.cloudflare.com/ajax/libs/dplayer/1.27.1/DPlayer.min.js
// @license           MIT
// @icon         data:image/gif;base64,R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==
// ==/UserScript==

(function () {
  let baseUrl = "https://www.haijiaom.cc/api/app-api";
  //let baseUrl = "http://127.0.0.1:8080/api/app-api";

  let vipUser = {};
  let isPlaying = false;
  let siteCode = "MADOU";
  let slefVideo = {
    videoPlayUrl: "",
    videoDuration: 0,
    id: 0,
    articleId: 0,
    videoBaseUrl: "",
    videoArticleId: "",
    downloadUrl: "",
  };
  let baseDownadloadUrl = baseUrl + "/business/download/down.m3u8?token=";
  let m3u8Url = "";
  let hasPrev = false;
  waitForPageLoad()
    .then((body) => {
      // createLoginForm();
      checkPermission();
      fetchUserInfo();
      removeAdds();
      createNavbar(body);
      initHlsPlayer();
      jljdRequest();
    })
    .catch((error) => {
      console.error("Failed to load page:", error);
    });

  //创建导航栏
  function createNavbar(body) {
    const fontAwesome = document.createElement("link");
    fontAwesome.rel = "stylesheet";
    fontAwesome.href =
      "https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css";
    document.head.appendChild(fontAwesome);

    const navbar = document.createElement("div");
    navbar.id = "custom-navbar";
    navbar.style.cssText = `
      position: fixed;
      top: 20%;
      right:-20px;
      transform: translate(-50%, -50%);
      background: rgba(255, 255, 255, 0);
      padding: 15px;
      border-radius: 12px;
      box-shadow: 0 4px 15px rgba(0,0,0,0.15);
      transition: all 0.3s ease;
      z-index: 99999;
    `;
    const buttonsContainer = document.createElement("div");
    buttonsContainer.style.cssText = `
      display: flex;
      flex-direction: column;
      gap: 20px;
    `;
    // Add event.preventDefault() to button actions
    const buttons = [
      {
        id: "personalInfoBtn",
        icon: "fas fa-user",
        action: (e) => {
          e.preventDefault();
          createPersonalInfoDialog();
        },
      },
      {
        id: "downloadBtn",
        icon: "fas fa-download",
        action: (e) => {
          e.preventDefault();
          createDownloadDialog();
        },
      },
      {
        id: "playBtn",
        icon: "fas fa-play",
        action: (e) => {
          e.preventDefault();
          togglePlay();
        },
      },
      {
        id: "toggleBtn",
        icon: "fas fa-eye-slash",
        action: (e) => {
          e.preventDefault();
          toggleVisibility();
        },
      },
    ];

    const buttonStyle = `
      width: 20px;
      height: 20px;
      border: none;
      border-radius: 50%;
      background: rgb(0, 0, 0,0.2);
      color: white;
      cursor: pointer;
      transition: all 0.2s ease;
      display: flex;
      align-items: center;
      justify-content: center;
      font-size: 16px;
    `;
    buttons.forEach((btn) => {
      const button = document.createElement("button");
      button.id = btn.id;
      button.innerHTML = `<i class="${btn.icon}"></i>`;
      button.style.cssText = buttonStyle;
      button.addEventListener("click", btn.action);
      buttonsContainer.appendChild(button);
    });
    navbar.appendChild(buttonsContainer);
    body.appendChild(navbar);

    // Floating toggle button
    const floatingBtn = document.createElement("button");
    floatingBtn.id = "floating-toggle-btn";
    floatingBtn.style.cssText = `
          position: fixed;
          bottom: 20px;
          right: 20px;
          width: 20px;
          height: 20px;
          border: none;
          border-radius: 50%;
          background: #4a90e2;
          color: white;
          cursor: pointer;
          transition: all 0.2s ease;
          display: none;
          align-items: center;
          justify-content: center;
          font-size: 12px;
          z-index: 9998;
        `;
    floatingBtn.innerHTML = '<i class="fas fa-eye"></i>';
    floatingBtn.addEventListener("click", toggleVisibility);
    document.body.appendChild(floatingBtn);
  }
  // 等待页面加载完成
  function createLoginFormView() {
    const loginDialog = document.createElement("div");
    let existingDialog = document.querySelector('[id^="login-dialog"]');

    if (existingDialog) return;
    loginDialog.innerHTML = `
      <style>
        .login-container {
          position: fixed;
          top: 50%;
          left: 50%;
          transform: translate(-50%, -50%);
          background: white;
          padding: 30px;
          border-radius: 12px;
          box-shadow: 0 2px 10px rgba(0,0,0,0.1);
          width: 90%;
          max-width: 400px;
          box-sizing: border-box;
        }

        .login-input {
          width: 100%;
          padding: 12px;
          border: 1px solid #ddd;
          border-radius: 6px;
          font-size: 14px;
          box-sizing: border-box;
        }

        .login-btn {
          width: 100%;
          padding: 12px;
          background: #4a90e2;
          color: white;
          border: none;
          border-radius: 6px;
          font-size: 15px;
          cursor: pointer;
        }

        .login-error {
          color: #ff4d4f;
          font-size: 13px;
          margin-top: 8px;
        }

        @media (max-width: 480px) {
          .login-container {
            padding: 20px;
            width: 70%;
          }

          h2 {
            font-size: 20px !important;
            margin-bottom: 8px !important;
          }

          p {
            font-size: 13px !important;
          }

          .login-input {
            padding: 10px;
            font-size: 13px;
            color:gray
          }

          .login-btn {
            padding: 10px;
            font-size: 14px;
          }
          .home-link {
            text-align: center;
            margin-top: 15px;
          }
          .home-link a {
            color: #4a90e2;
            text-decoration: none;
            font-size: 14px;
          }
          .home-link a:hover {
            text-decoration: underline;
          }
        }
      </style>

      <div class="login-container" id="login-dialog">
        <div style="text-align: right;">
          <span class="dialog-close" style="cursor: pointer; font-size: 18px;">×</span>
        </div>

        <div style="margin-bottom: 20px;">
          <h2 style="margin: 0 0 8px; color: #333; font-size: 22px; font-weight: 600;">欢迎登录</h2>
          <p style="margin: 0; color: #666; font-size: 14px;">请输入您的账号密码</p>
        </div>

        <div style="margin-bottom: 15px;">
          <input type="text" id="username" placeholder="用户名" class="login-input">
        </div>

        <div style="margin-bottom: 20px;">
          <input type="password" id="password" placeholder="密码" class="login-input">
        </div>

        <button id="loginSubmit" class="login-btn">
          <span id="loginText">登 录</span>
          <span id="loginLoading" style="display:none;">
            <i class="fas fa-spinner fa-spin"></i>
          </span>
        </button>
        <div class="home-link">
          <a href="https://www.haijiaom.cc" id="goHome">
          <i class="fas fa-home"></i> 会员注册
           </a>
        </div>

        <div id="loginError" class="login-error"></div>
      </div>
    `;

    document.body.appendChild(loginDialog);
    // Make login dialog responsive

    // Close dialog when clicking X
    document.querySelector(".dialog-close").addEventListener("click", () => {
      loginDialog.remove();
    });

    // 添加输入框焦点效果
    const inputs = loginDialog.querySelectorAll("input");
    inputs.forEach((input) => {
      input.addEventListener("focus", () => {
        input.style.borderColor = "#4a90e2";
      });
      input.addEventListener("blur", () => {
        input.style.borderColor = "#ddd";
      });
    });

    document
      .getElementById("loginSubmit")
      .addEventListener("click", async () => {
        const username = document.getElementById("username").value;
        const password = document.getElementById("password").value;

        const loginBtn = document.getElementById("loginSubmit");
        const loginText = document.getElementById("loginText");
        const loginLoading = document.getElementById("loginLoading");

        loginText.style.display = "none";
        loginLoading.style.display = "inline-block";
        loginBtn.disabled = true;
        if (!username || !password) {
          const errorDiv = document.getElementById("loginError");
          errorDiv.textContent = "请输入用户名和密码";
          errorDiv.style.display = "block";
          return;
        }

        try {
          const response = await fetch(baseUrl + "/business/member/login", {
            method: "POST",
            headers: {
              "Content-Type": "application/json",
            },
            body: JSON.stringify({ username, password }),
          });

          const data = await response.json();

          if (data.code == 200 && data.data.access_token) {
            GM_setValue("myToken", data.data.access_token);
            // localStorage.setItem("token", data.accessToken);
            loginDialog.remove();
          } else {
            throw new Error(data.msg || "登录失败");
          }
        } catch (error) {
          const errorDiv = document.getElementById("loginError");
          errorDiv.textContent = error.message;
          errorDiv.style.display = "block";
          loginText.style.display = "inline-block";
          loginLoading.style.display = "none";
          loginBtn.disabled = false;
        }
      });
  }
  function createLoginForm() {
    // 防止重复创建
    if (document.getElementById("login-container-unique-v8")) {
      return;
    }

    // 创建主容器
    const container = document.createElement("div");
    container.id = "login-container-unique-v8";

    container.innerHTML = `
          <style>
              @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@300;400;600&display=swap');

              #login-container-unique-v8 {
                  all: initial !important;

                  position: fixed !important;
                  top: 0 !important;
                  left: 0 !important;
                  width: 100% !important;
                  height: 100% !important;
                  background: transparent !important;
                  display: flex !important;
                  justify-content: center !important;
                  align-items: center !important;
                  z-index: 2147483647 !important;
                  font-family: 'Poppins', sans-serif !important;
                  font-size: 12px !important;

                  opacity: 0;
                  transition: opacity 0.5s ease !important;
              }

              #login-container-unique-v8 * {
                  font-family: inherit !important;
                  box-sizing: border-box !important;
              }

              .login-box-v8 {
                  background: rgba(75, 75, 214, 0.85) !important;
                  backdrop-filter: blur(10px) !important;
                  -webkit-backdrop-filter: blur(10px) !important;
                  border-radius: 20px !important;
                  border: 1px solid rgba(255, 255, 255, 0.2) !important;
                  box-shadow: 0 8px 32px 0 rgba(0, 0, 0, 0.37) !important;
                  padding: 16px 40px !important;
                  width: 60% !important;
                  max-width: 400px !important;
                  text-align: center !important;
                  color: white !important;
                  position: relative !important;

                  transform: scale(0.9);
                  transition: transform 0.5s ease !important;
              }

              .login-title-v8 {
                  font-size: 16px !important;
                  font-weight: 600 !important;
                  margin-bottom: 32px !important;
              }

              .input-group-v8 {
                  position: relative !important;
                  margin-bottom: 24px !important;
              }

              .login-input-v8 {
                  width: 100% !important;
                  padding: 6px 6px 6px 48px !important;
                  border: 1px solid rgba(255, 255, 255, 0.3) !important;
                  border-radius: 10px !important;
                  font-size: 12px !important;
                  background: rgba(255, 255, 255, 0.2) !important;
                  color: white !important;
                  transition: background-color 0.3s, box-shadow 0.3s !important;
              }

              .login-input-v8::placeholder {
                  color: rgba(255, 255, 255, 0.7) !important;
              }

              .login-input-v8:focus {
                  outline: none !important;
                  background: rgba(255, 255, 255, 0.3) !important;
                  box-shadow: 0 0 0 2px rgba(255, 255, 255, 0.5) !important;
              }

              .input-icon-v8 {
                  position: absolute !important;
                  left: 16px !important;
                  top: 50% !important;
                  transform: translateY(-50%) !important;
                  color: rgba(255, 255, 255, 0.8) !important;
                  width: 20px !important;
                  height: 20px !important;
              }

              .login-button-v8 {
                  width: 100% !important;
                  padding: 6px !important;
                  border: none !important;
                  border-radius: 10px !important;
                  background: #ff6b6b !important;
                  color: white !important;
                  font-size: 18px !important;
                  font-weight: 600 !important;
                  cursor: pointer !important;
                  transition: background-color 0.3s, transform 0.2s !important;
                  margin-bottom: 16px !important; /* 为下方的链接留出空间 */
              }

              .login-button-v8:hover {
                  background: #ff4757 !important;
                  transform: translateY(-2px) !important;
              }

              .close-button-v8 {
                  position: absolute !important;
                  top: 15px !important;
                  right: 15px !important;
                  background: none !important;
                  border: none !important;
                  font-size: 16px !important;
                  color: rgba(255, 255, 255, 0.8) !important;
                  cursor: pointer !important;
                  transition: color 0.3s, transform 0.3s !important;
                  line-height: 1 !important;
                  padding: 0 !important;
              }

              .close-button-v8:hover {
                  color: white !important;
                  transform: rotate(90deg) !important;
              }

              /* 新增: 注册链接区域样式 */
              .extra-links-v8 {
                  font-size: 12px !important;
                  color: rgba(255, 255, 255, 0.8) !important;
              }

              .register-link-v8 {
                  color: #ff8c8c !important;
                  text-decoration: none !important;
                  font-weight: 600 !important;
                  cursor: pointer !important;
                  transition: color 0.3s !important;
              }

              .register-link-v8:hover {
                  text-decoration: underline !important;
                  color: #ffffff !important;
              }

              .error-message-v8 {
                  color: #ffcdd2 !important;
                  font-size: 14px !important;
                  text-align: left !important;
                  margin-top: -16px !important; /* 放在输入框和按钮之间 */
                  margin-bottom: 16px !important;
                  min-height: 21px !important; /* 预留空间防止布局抖动 */
                  visibility: hidden;
                  opacity: 0;
                  transition: opacity 0.3s ease-in-out !important;
              }

              .error-message-v8.visible {
                  visibility: visible;
                  opacity: 1;
              }

          </style>
          <div class="login-box-v8">
              <button class="close-button-v8">&times;</button>
              <h2 class="login-title-v8">欢迎回来</h2>
              <div class="input-group-v8">
                  <svg class="input-icon-v8" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path d="M10 2a5 5 0 00-5 5v2a2 2 0 00-2 2v5a2 2 0 002 2h10a2 2 0 002-2v-5a2 2 0 00-2-2V7a5 5 0 00-5-5zm0 2.5a2.5 2.5 0 012.5 2.5V7h-5V7a2.5 2.5 0 012.5-2.5z"></path></svg>
                  <input type="text" class="login-input-v8" placeholder="用户名">
              </div>
              <div class="input-group-v8">
                  <svg class="input-icon-v8" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M5 9V7a5 5 0 0110 0v2a2 2 0 012 2v5a2 2 0 01-2 2H5a2 2 0 01-2-2v-5a2 2 0 012-2zm8-2v2H7V7a3 3 0 016 0z" clip-rule="evenodd"></path></svg>
                  <input type="password" class="login-input-v8" placeholder="密码">
              </div>
              <p class="error-message-v8"></p>
              <button class="login-button-v8">登录</button>
              <!-- 新增: 注册链接 -->
              <div class="extra-links-v8">
                  <span>没有账户? <a class="register-link-v8">立即注册</a></span>
              </div>
          </div>
      `;

    document.body.appendChild(container);

    const loginBox = container.querySelector(".login-box-v8");
    const errorElement = container.querySelector(".error-message-v8");
    requestAnimationFrame(() => {
      container.style.opacity = "1";
      if (loginBox) {
        loginBox.style.transform = "scale(1)";
      }
    });
    const showError = (message) => {
      errorElement.textContent = message;
      errorElement.classList.add("visible");
    };

    const hideError = () => {
      errorElement.classList.remove("visible");
    };
    const close = () => {
      container.style.opacity = "0";
      if (loginBox) {
        loginBox.style.transform = "scale(0.9)";
      }
      setTimeout(() => {
        if (document.body.contains(container)) {
          document.body.removeChild(container);
        }
      }, 500);
    };
    container.querySelector(".close-button-v8").onclick = close;
    container.querySelector(".login-button-v8").onclick = async () => {
      const username = container.querySelector('input[type="text"]').value;
      const password = container.querySelector('input[type="password"]').value;
      if (!username || !password) {
        showError("请输入用户名和密码!");
        return;
      }
      try {
        const response = await fetch(baseUrl + "/business/member/login", {
          method: "POST",
          headers: {
            "Content-Type": "application/json",
          },
          body: JSON.stringify({ username, password }),
        });

        const data = await response.json();

        if (data.code == 200 && data.data.access_token) {
          GM_setValue("myToken", data.data.access_token);
          // localStorage.setItem("token", data.accessToken);
          container.remove();
        } else {
          throw new Error(data.msg || "登录失败");
        }
      } catch (error) {
        showError("用户名或密码错误。");
      }
    };

    // 新增: 注册链接点击事件
    container.querySelector(".register-link-v8").onclick = (e) => {
      e.preventDefault();
      window.location.href = "https://www.haijiaom.cc";
      // 在这里可以添加跳转到注册页或显示注册模态框的逻辑
    };
  }
  function createPersonalInfoDialog() {
    if (!checkToken()) {
      return;
    }
    let featureEnabled = GM_getValue("feature_enabled", true);

    let memberLvel = vipUser.sites.filter((item) => {
      return item.siteCode === siteCode;
    });

    if (memberLvel[0]) {
      featureEnabled = memberLvel[0].scriptOpen;
    }

    const dialog = document.createElement("div");
    dialog.innerHTML = `
        <style>
            .info-dialog {
                position: fixed;
                top: 50%;
                left: 50%;
                transform: translate(-50%, -50%);
                background: white;
                padding: 20px;
                border-radius: 12px;
                box-shadow: 0 2px 10px rgba(0,0,0,0.1);
                width: 90%;
                max-width: 320px;
                box-sizing: border-box;
                z-index: 10000;
            }

            .info-header {
                display: flex;
                justify-content: space-between;
                align-items: center;
                margin-bottom: 15px;
            }

            .info-close {
                width: 24px;
                height: 24px;
                border: none;
                border-radius: 50%;
                background: #f5f5f5;
                color: #666;
                cursor: pointer;
                display: flex;
                align-items: center;
                justify-content: center;
                transition: all 0.3s;
            }

            .info-avatar {
                width: 60px;
                height: 60px;
                border-radius: 50%;
                background: #4a90e2;
                color: white;
                display: flex;
                align-items: center;
                justify-content: center;
                margin: 0 auto 15px;
            }

            .info-stats {
                display: grid;
                grid-template-columns: repeat(2, 1fr);
                gap: 10px;
                margin: 15px 0;
            }

            .info-stat-item {
                background: #f8f9fa;
                padding: 10px;
                border-radius: 8px;
                text-align: center;
            }

            .info-actions {
                display: grid;
                gap: 10px;
                margin-top: 15px;
            }

            .vip-btn {
                background: #4a90e2;
                color: white;
                padding: 10px;
                border-radius: 8px;
                text-decoration: none;
                text-align: center;
                font-size: 14px;
            }

            .logout-btn {
                background: #f44336;
                color: white;
                border: none;
                padding: 10px;
                border-radius: 8px;
                cursor: pointer;
                font-size: 14px;
            }

            .expired-notice {
                background: #fff3cd;
                color: #856404;
                padding: 10px;
                border-radius: 8px;
                margin: 10px 0;
                font-size: 13px;
            }
            .script-settings {
                border-top: 1px solid #f0f0f0;
                margin-top: 15px;
                padding-top: 10px;
            }
            .setting-item {
                display: flex;
                justify-content: space-between;
                align-items: center;
                padding: 8px 0;
            }
            .setting-item span {
                font-size: 14px;
                color: #333;
            }
            .switch {
                position: relative;
                display: inline-block;
                width: 44px;
                height: 24px;
            }
            .switch input {
                opacity: 0;
                width: 0;
                height: 0;
            }
            .slider {
                position: absolute;
                cursor: pointer;
                top: 0;
                left: 0;
                right: 0;
                bottom: 0;
                background-color: #ccc;
                transition: .4s;
                border-radius: 24px;
            }
            .slider:before {
                position: absolute;
                content: "";
                height: 20px;
                width: 20px;
                left: 2px;
                bottom: 2px;
                background-color: white;
                transition: .4s;
                border-radius: 50%;
            }
            input:checked + .slider {
                background-color: #4a90e2;
            }
            input:checked + .slider:before {
                transform: translateX(20px);
            }
        </style>

        <div class="info-dialog">
            <div class="info-header">
                <h3 style="margin:0">个人信息</h3>
                <button class="info-close">×</button>
            </div>

            <div class="info-avatar">
                <i class="fas fa-user fa-2x"></i>
            </div>

            <div style="text-align:center">
                <h2 style="margin:0;font-size:18px">${
                  vipUser.username || "未登录"
                }</h2>
                <p style="margin:5px 0;color:#666;font-size:13px">
                    ${vipUser.vipLevel || "普通用户"}
                </p>
            </div>

            <div class="info-stats">
                <div class="info-stat-item">
                    <div style="font-size:18px;color:#4a90e2;font-weight:bold">
                        ${vipUser.downloadNum || 0}
                    </div>
                    <div style="font-size:12px;color:#666">下载次数</div>
                </div>
                <div class="info-stat-item">
                    <div style="font-size:18px;color:#4a90e2;font-weight:bold">
                        ${vipUser.watchNum || 0}
                    </div>
                    <div style="font-size:12px;color:#666">观看次数</div>
                </div>
            </div>

            ${
              vipUser.expireTime
                ? `
                <div class="expired-notice">
                    ${
                      new Date(vipUser.expireTime) > new Date()
                        ? `会员有效期至:${new Date(
                            vipUser.expireTime
                          ).toLocaleDateString()}`
                        : "您的会员已过期,请重新开通"
                    }
                </div>
            `
                : ""
            }

            <div class="script-settings">
                <div class="setting-item">
                    <span>脚本开关</span>
                    <label class="switch">
                        <input type="checkbox" id="featureToggle" ${
                          featureEnabled ? "checked" : ""
                        }>
                        <span class="slider"></span>
                    </label>
                </div>
            </div>

            <div class="info-actions">
                <a href="https://www.haijiaom.cc/" class="vip-btn">
                    <i class="fas fa-crown"></i>
                    ${
                      vipUser.expireTime &&
                      new Date(vipUser.expireTime) > new Date()
                        ? "续费会员"
                        : "开通会员"
                    }
                </a>
                <button class="logout-btn" id="logoutBtn">
                    <i class="fas fa-sign-out-alt"></i> 退出登录
                </button>
            </div>
        </div>
    `;
    document.body.appendChild(dialog);
    const featureToggle = dialog.querySelector("#featureToggle");
    if (featureToggle) {
      featureToggle.addEventListener("change", async () => {
        const isEnabled = featureToggle.checked;
        featureToggle.disabled = true;
        try {
          const result = await putAsync(
            baseUrl + "/business/member/script-operation",
            {
              scriptOpen: isEnabled,
              siteCode: siteCode,
            }
          );
          if (result.code == 200) {
            GM_setValue("feature_enabled", isEnabled);

            showApiMessage(
              `功能已${isEnabled ? "开启" : "关闭"},页面即将刷新...`
            );
            setTimeout(() => {
              window.location.reload();
            }, 1500);
          } else {
            showApiMessage(result.msg || "操作失败", "error");
            featureToggle.checked = !isEnabled;
          }
        } catch (error) {
          showApiMessage("请求失败,请检查网络连接", "error");
          featureToggle.checked = !isEnabled;
        } finally {
          featureToggle.disabled = false;
        }
      });
    }

    // Add logout handler
    dialog.querySelector("#logoutBtn").addEventListener("click", () => {
      GM_setValue("myToken", "");
      dialog.remove();

      setTimeout(() => window.location.reload(), 1000);
    });

    // Add close handler
    dialog.querySelector(".info-close").addEventListener("click", () => {
      dialog.remove();
    });
  }
  const baseTools = [
    {
      name: "在线下载1",
      icon: "fas fa-bolt",
      url: "https://tools.thatwind.com/tool/m3u8downloader",
      clients: ["windows", "mac"],
    },

    {
      name: "在线下载2",
      icon: "fas fa-cloud-download-alt",
      url: "http://tools.bugscaner.com/m3u8.html",
      clients: ["windows", "mac", "linux"],
    },
  ];

  // 根据客户端类型获取工具列表
  function getDownloadTools() {
    const client = detectClient();
    const mobileTools = [
      {
        name: "安卓视频下载软件",
        icon: "fas fa-mobile-alt",
        url: "market://details?id=com.xunlei.downloadprovider",
        clients: ["android"],
      },
      {
        name: "苹果视频下载软件",
        icon: "fas fa-file-download",
        url: "https://apps.apple.com/cn/app/m3u8-mpjex/id6449724938",
        clients: ["ios"],
      },
    ];

    const tools = [...baseTools, ...mobileTools].filter((tool) =>
      tool.clients.includes(client)
    );

    return tools;
  }

  // 检测客户端类型
  function detectClient() {
    const ua = navigator.userAgent.toLowerCase();
    if (/android/i.test(ua)) return "android";
    if (/iphone|ipad|ipod/i.test(ua)) return "ios";
    if (/macintosh/i.test(ua)) return "mac";
    if (/linux/i.test(ua)) return "linux";
    return "windows";
  }

  function createDownloadDialog(downloadUrl) {
    if (!checkToken()) {
      return;
    }
    const firstDialog = document.createElement("div");
    firstDialog.style.cssText = `
        position: fixed;
        top: 50%;
        left: 50%;
        transform: translate(-50%, -50%);
        background: white;
        padding: 25px;
        padding-top: 40px;
        border-radius: 12px;
        box-shadow: 0 10px 25px rgba(0,0,0,0.15);
        z-index: 10000;
        min-width: 320px;
        text-align: center;
        font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
        animation: fadeIn 0.3s ease;
    `;

    const styleSheet = document.createElement("style");
    styleSheet.textContent = `
        @keyframes fadeIn {
            from { opacity: 0; transform: translate(-50%, -48%); }
            to { opacity: 1; transform: translate(-50%, -50%); }
        }
        .dialog-btn {
            padding: 10px 24px;
            margin: 0 8px;
            border: none;
            border-radius: 6px;
            font-size: 14px;
            font-weight: 500;
            cursor: pointer;
            transition: all 0.2s ease;
        }
        .dialog-btn:hover {
            transform: translateY(-2px);
            box-shadow: 0 4px 12px rgba(0,0,0,0.1);
        }
        .primary-btn {
            background: linear-gradient(135deg, #4a90e2, #357abd);
            color: white;
        }
        .secondary-btn {
            background: #f5f5f5;
            color: #666;
        }
        .dialog-close {
            position: absolute;
            top: 10px;
            right: 10px;
            width: 24px;
            height: 24px;
            border-radius: 50%;
            background: #f5f5f5;
            border: none;
            cursor: pointer;
            display: flex;
            align-items: center;
            justify-content: center;
            transition: all 0.2s ease;
            color: #666;
            font-size: 14px;
        }
        .dialog-close:hover {
            background: #e0e0e0;
            transform: rotate(90deg);
        }
    `;
    document.head.appendChild(styleSheet);

    firstDialog.innerHTML = `
        <button class="dialog-close">✕</button>
        <h3 style="margin: 0 0 20px; font-size: 20px; color: #333; font-weight: 500;">下载确认</h3>
        <p style="margin: 0 0 8px; color: #666; font-size: 15px;">本次下载将消耗1次下载机会</p>
        <p style="margin: 0 0 25px; font-size: 15px;">
            剩余下载次数:<span style="color: #ff4444; font-weight: 600;">${vipUser.downloadNum}</span>
        </p>
        <div style="margin-top: 25px;">
            <button id="confirmDownload" class="dialog-btn primary-btn">确认下载</button>
            <button id="cancelDownload" class="dialog-btn secondary-btn">取消</button>
        </div>
    `;

    document.body.appendChild(firstDialog);

    function showDownloadOptions() {
      const secondDialog = document.createElement("div");
      secondDialog.style.cssText = firstDialog.style.cssText;

      const downloadTools = getDownloadTools();
      const toolButtons = downloadTools
        .map(
          (tool) => `
        <button id="${tool.name}" class="dialog-btn primary-btn" style="
            width: 100%;
            margin: 8px 0;
            background: linear-gradient(135deg, #4CAF50, #45a049);
            display: flex;
            align-items: center;
            justify-content: center;
            gap: 8px;
        ">
            <i class="${tool.icon}"></i>${tool.name}
        </button>
    `
        )
        .join("");

      secondDialog.innerHTML = `
        <button class="dialog-close">✕</button>
        <h3 style="margin: 0 0 20px; font-size: 20px; color: #333; font-weight: 500;">下载选项</h3>
        <div style="margin: 20px 0;">
            <button id="copyLink" class="dialog-btn primary-btn" style="
                width: 100%;
                margin: 8px 0;
                display: flex;
                align-items: center;
                justify-content: center;
                gap: 8px;
            ">
                <i class="fas fa-copy"></i>复制下载链接
            </button>
            ${toolButtons}
        </div>
    `;

      document.body.appendChild(secondDialog);

      // 为每个工具添加点击事件
      downloadTools.forEach((tool) => {
        document.getElementById(tool.name)?.addEventListener("click", () => {
          window.open(tool.url, "_blank");
        });
      });

      // 复制链接事件
      document.getElementById("copyLink").addEventListener("click", () => {
        navigator.clipboard.writeText(slefVideo.downloadUrl);
        showApiMessage("下载链接已复制到剪贴板");
      });

      // 关闭按钮事件
      secondDialog
        .querySelector(".dialog-close")
        .addEventListener("click", () => {
          secondDialog.remove();
        });
    }

    // 确认下载事件
    document
      .getElementById("confirmDownload")
      .addEventListener("click", async () => {
        firstDialog.remove();
        let result = await gcreateDownloadUrl();
        if (result.code == 200) {
          slefVideo.downloadUrl = m3u8Url;
          if (vipUser.downloadNum == 0) {
            showApiMessage("下载次数已用完", "error");
            return;
          }
          showDownloadOptions();
        }
      });

    // 取消下载事件
    document.getElementById("cancelDownload").addEventListener("click", () => {
      firstDialog.remove();
    });

    // 关闭按钮事件
    firstDialog.querySelector(".dialog-close").addEventListener("click", () => {
      firstDialog.remove();
    });
  }

  function initHlsPlayer() {
    const hlsScript = document.createElement("script");
    hlsScript.src = "https://cdn.jsdelivr.net/npm/hls.js@latest";
    document.head.appendChild(hlsScript);

    const playerContainer = document.createElement("div");
    playerContainer.id = "hls-player-container";
    playerContainer.style.cssText = `
        position: fixed;
        top: 0;
        bottom: 0;
        left:0;
        right:0;
        background: rgba(0, 0, 0, 1);
        padding: 10px;
        border-radius: 8px;
        box-shadow: 0 2px 10px rgba(0,0,0,0.2);
        z-index: 9996;
        display: none;
    `;
    // Add close button for player
    const closeButton = document.createElement("button");
    closeButton.innerHTML = '<i class="fas fa-times"></i>';
    closeButton.style.cssText = `
        position: absolute;
        top: 10px;
        right: 10px;
        background: rgba(0,0,0,0.5);
        color: white;
        border: none;
        border-radius: 50%;
        width: 36px;
        height: 36px;
        cursor: pointer;
        z-index: 9999;
        display: flex;
        align-items: center;
        justify-content: center;
      `;
    closeButton.onclick = () => {
      window.hlsPlayer.hide();
    };
    playerContainer.appendChild(closeButton);
    let currentUrl = "";

    // Create video element
    const video = document.createElement("video");
    video.id = "hls-video";
    video.style.cssText = `
        width: 100%;
        height: 100%;
        object-fit: contain;
      `;
    video.controls = true;

    playerContainer.appendChild(video);
    document.body.appendChild(playerContainer);
    hlsScript.onload = () => {
      window.hlsPlayer = {
        setContent: (input) => {
          if (isM3U8Content(input)) {
            // Handle M3U8 content string
            currentUrl = createBlobUrl(input);
          } else {
            // Handle URL
            currentUrl = input;
          }
        },
        play: () => {
          if (!currentUrl) {
            console.error("Please set URL first");
            return;
          }
          playerContainer.style.display = "block";
          if (Hls.isSupported()) {
            const hls = new Hls();
            hls.loadSource(currentUrl);
            hls.attachMedia(video);
            hls.on(Hls.Events.MANIFEST_PARSED, () => {
              video.play();
              isPlaying = true;
              updatePlayButton();
            });
          }
        },
        pause: () => {
          video.pause();
          isPlaying = false;
          updatePlayButton();
        },
        toggle: () => {
          if (isPlaying) {
            window.hlsPlayer.pause();
          } else {
            window.hlsPlayer.play();
          }
        },
        hide: () => {
          playerContainer.style.display = "none";
          video.pause();
          isPlaying = false;
          updatePlayButton();
          toggleVisibility();
        },
      };
    };
  }
  function isM3U8Content(str) {
    return str.trim().startsWith("#EXTM3U");
  }
  function updatePlayButton() {
    const playBtn = document.getElementById("playBtn");
    if (playBtn) {
      playBtn.innerHTML = isPlaying
        ? '<i class="fas fa-pause"></i>'
        : '<i class="fas fa-play"></i>';
      playBtn.style.background = isPlaying ? "#4CAF50" : "#4a90e2";
    }
  }
  async function togglePlay() {
    if (!(await checkToken())) {
      return;
    }
    console.log(!checkPermission());
    if (!(await checkPermission())) {
      return;
    }
    if (m3u8Url == "") {
      showApiMessage("请先获取视频链接", "error");
      return;
    }
    let m3u8Content = "";
    await fetch(m3u8Url, {
      method: "GET",
    })
      .then((res) => res.text())
      .then((data) => {
        m3u8Content = data;
      });
    m3u8Content = m3u8Content.replace(
      "/api/app/media/enkey",
      "https://d1kek4wgeaw03m.cloudfront.net/api/app/media/enkey"
    );
    window.hlsPlayer.setContent(m3u8Url);
    window.hlsPlayer.play();
    toggleVisibility();
  }

  function serializeM3u8() {
    let m3u8Content = "";
    fetch(m3u8Url, {
      method: "GET",
    })
      .then((res) => res.text())
      .then((data) => {
        m3u8Content = data;
        let lines = m3u8Content.split("\n");
        slefVideo.m3u8Url = lines[lines.length - 2];
      });
  }

  function toggleVisibility() {
    const navbar = document.getElementById("custom-navbar");
    const floatingBtn = document.getElementById("floating-toggle-btn");
    const isVisible = navbar.style.display !== "none";
    navbar.style.display = isVisible ? "none" : "block";
    floatingBtn.style.display = isVisible ? "flex" : "none";
  }
  const OriginalXHR = unsafeWindow.XMLHttpRequest || window.XMLHttpRequest;

  function CustomXHR() {
    const xhr = new OriginalXHR();
    const originalOpen = xhr.open;
    const originalSend = xhr.send;
    let requestURL = "";

    let modifiedRequestURL = "";

    const urlMappings = {
      "videos/v2/getUrl": (url) => {
        // Example: Change URL for video requests
        return url.replace(
          url,
          "https://api.qianyuewenhua.xyz/videos/getPreUrl"
        );
      },
    };

    function modifyURL(originalURL) {
      for (const [pattern, modifier] of Object.entries(urlMappings)) {
        if (originalURL.includes(pattern)) {
          modifiedRequestURL = modifier(originalURL);
          return modifiedRequestURL;
        }
      }
      return originalURL;
    }
    // Add URL modifications before open() is called
    xhr.open = function () {
      requestURL = arguments[1];
      let modifiedURL = modifyURL(requestURL);
      arguments[1] = modifiedURL;
      return originalOpen.apply(this, arguments);
    };

    // xhr.open = function () {
    //   requestURL = arguments[1];

    //   return originalOpen.apply(this, arguments);
    // };

    xhr.send = function () {
      const originalOnReadyStateChange = xhr.onreadystatechange;
      xhr.onreadystatechange = function () {
        if (xhr.readyState === 4) {
          if (xhr.status === 200) {
            try {
              // URL pattern matching
              Object.keys(urlPatterns).forEach((pattern) => {
                if (new RegExp(pattern).test(requestURL)) {
                  let response = JSON.parse(xhr.responseText);
                  const modifiedResponse = urlPatterns[pattern](response);
                  if (modifiedResponse) {
                    Object.defineProperty(xhr, "response", {
                      value: JSON.stringify(modifiedResponse),
                    });
                    Object.defineProperty(xhr, "responseText", {
                      value: JSON.stringify(modifiedResponse),
                    });
                  }
                  console.log(
                    "[Modified Response]:",
                    requestURL,
                    modifiedResponse
                  );
                }
              });
            } catch (error) {
              console.error("[XHR Error]:", error, requestURL);
            }
          }
        }

        if (originalOnReadyStateChange) {
          originalOnReadyStateChange.apply(this, arguments);
        }
      };

      return originalSend.apply(this, arguments);
    };

    return xhr;
  }

  // Replace XMLHttpRequest
  if (typeof unsafeWindow !== "undefined") {
    unsafeWindow.XMLHttpRequest = CustomXHR;
  } else {
    window.XMLHttpRequest = CustomXHR;
  }
  const urlPatterns = {
    "api\\/video\\/checkVideoCanPlay(\\?[^/]*)?$": (response) => {
      let decodeData = response.data;
      const freeVideo = { type: 1, amount: 0, money_type: 0, vip: 0 };
      response.data = Object.assign(
        {},
        JSON.stringify(atob(atob(atob(decodeData))), `utf-8`),
        freeVideo
      );
      return response;
    },
    "api\\/banner\\/banner_list(\\?[^/]*)?$": (response) => {
      return null;
    },
    "videos\\/getInfo(\\?[^/]*)?$": (response) => {
      if (checkToken() && hasPrev) {
        response.data.canPlay = true;
        slefVideo.id = response.data.info.id;
      }
      return response;
    },
    "videos\\/v2\\/getUrl(\\?[^/]*)?$": (response) => {
      if (checkToken() && hasPrev) {
        response.data.url = response.data.url.replace(
          /start=\d+\&end=\d+\&/,
          ""
        );
        m3u8Url = response.data.url;
        serializeM3u8();
      }
      return response;
    },
  };
  function showApiMessage(message = "操作成功", type = "success") {
    // 防止重复弹窗
    if (document.getElementById("api-message-dialog")) return;
    const colorMap = {
      success: "#43a047",
      error: "#d32f2f",
      info: "#1976d2",
      warning: "#ffa000",
    };
    const iconMap = {
      success: "fa-check-circle",
      error: "fa-exclamation-triangle",
      info: "fa-info-circle",
      warning: "fa-exclamation-circle",
    };
    const dialog = document.createElement("div");
    dialog.id = "api-message-dialog";
    dialog.innerHTML = `
        <style>
          .api-message-dialog {
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: #fff;
            color: ${colorMap[type] || "#333"};
            border-radius: 10px;
            box-shadow: 0 2px 10px rgba(0,0,0,0.15);
            padding: 24px 20px 16px 20px;
            min-width: 220px;
            z-index: 10001;
            text-align: center;
            font-size: 15px;
          }
          .api-message-close {
            margin-top: 18px;
            background: ${colorMap[type] || "#333"};
            color: #fff;
            border: none;
            border-radius: 6px;
            padding: 8px 24px;
            font-size: 14px;
            cursor: pointer;
          }
        </style>
        <div class="api-message-dialog">
          <div style="font-size:18px;margin-bottom:10px;">
            <i class="fas ${iconMap[type] || "fa-info-circle"}"></i>
          </div>
          <div>${message}</div>
          <button class="api-message-close">关闭</button>
        </div>
      `;
    document.body.appendChild(dialog);
    dialog.querySelector(".api-message-close").onclick = () => dialog.remove();
    setTimeout(() => dialog.remove(), 4500);
  }
  // 初始化播放器
  function initPlayer(container, videoUrl) {
    waitForElement(".mine-ad").then((e) => {
      console.log("e", e);
      e.remove();
    });
    const dp = new DPlayer({
      container: container,
      autoplay: true,
      video: {
        url: videoUrl,
        type: "hls", // 直接使用 'hls' 类型
        customType: {
          hls: (video, player) => {
            if (Hls.isSupported()) {
              const hls = new Hls();
              hls.loadSource(videoUrl);
              hls.attachMedia(video);
            } else if (video.canPlayType("application/vnd.apple.mpegurl")) {
              video.src = video.src;
              video.addEventListener("loadedmetadata", () => {
                video.play();
              });
            }
          },
        },
      },
    });
    dp.on("play", () => {
      waitForElement(".dplayer-mobile-play").then((res) => {
        res.style.display = "none";
      });
    });
    dp.on("pause", () => {
      waitForElement(".dplayer-mobile-play").then((res) => {
        res.style.display = "inline";
      });
    });
    waitForElement(".dplayer-play-icon").then((res) => {
      res.style.display = "inline";
    });

    window.dp = dp; // 挂载到 window 方便调试
  }
  async function jljdRequest() {
    const originalFetch = unsafeWindow.fetch;
    let videoContainer = "";
    if (!(await checkPermission())) {
      return;
    }
    unsafeWindow.fetch = async function (url, options) {
      const response = await originalFetch(url, options);

      if (url.endsWith("/media/play") || url.endsWith("/onepost/detail")) {
        slefVideo.id = window.location.href.split("/").pop();
        const clone = response.clone();
        let responseJson = await clone.json();
        slefVideo.signData = responseJson.data;
        await serializeVideo();
        if (slefVideo.id.includes("short-video")) {
          waitForElement("#short-play")
            .then((container) => {
              const noticeContainer = document.createElement("div");
              noticeContainer.style.cssText = `
        position: relative;
        width: 100%;
        height: 100%;
        `;
              const notice = document.createElement("div");
              notice.style.cssText = `
            position: absolute;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%) rotate(-15deg);
            color: red;
            text-align: center;
            padding: 10px;
            font-size: 20px;
            font-weight: bold;
            text-shadow: 2px 2px 4px rgba(0,0,0,0.5);
            z-index: 1000;
            pointer-events: none;
            opacity: 0.8;
          `;
              notice.textContent =
                "温馨提示:请用右边悬浮导航的播放按钮进行观看!";
              container.parentNode.insertBefore(noticeContainer, container);
              noticeContainer.appendChild(container);
              noticeContainer.appendChild(notice);
            })
            .catch(console.error);
        } else {
          waitForElement(".md-video-play").then((res) => {
            // const videoUrl = 'https://d1ra935cee4n7e.cloudfront.net/api/app/media/m3u8/sp/hl/5m/5x/sq/0303eae2efd84d1385c292dc8b4a8d22.m3u8';
            initPlayer(res, m3u8Url);
            return response;
          });
        }

        // const noticeContainer = document.createElement("div");
        ///noticeContainer.style.cssText = `
        //position: relative;
        //width: 100%;
        //height: 100%;
        // `;
        // const notice = document.createElement("div");
        // notice.style.cssText = `
        //     position: absolute;
        //     top: 50%;
        //     left: 50%;
        //     transform: translate(-50%, -50%) rotate(-15deg);
        //     color: red;
        //     text-align: center;
        //     padding: 10px;
        //     font-size: 20px;
        //     font-weight: bold;
        //     text-shadow: 2px 2px 4px rgba(0,0,0,0.5);
        //     z-index: 1000;
        //     pointer-events: none;
        //     opacity: 0.8;
        //   `;
        //notice.textContent = "温馨提示:请用右边悬浮导航的播放按钮进行观看!";
        //res.parentNode.insertBefore(noticeContainer, res);
        // noticeContainer.appendChild(res);
        //noticeContainer.appendChild(notice);
        return response;
      }
      return response;
    };
  }
  function waitForPageLoad() {
    return new Promise((resolve, reject) => {
      // Set timeout for safety
      const timeout = setTimeout(() => {
        reject("Page load timeout");
      }, 10000);

      // Check if page is already loaded
      if (document.readyState === "complete") {
        clearTimeout(timeout);
        resolve(document.body);
        return;
      }

      // Create observer to monitor DOM changes
      const observer = new MutationObserver((mutations, obs) => {
        if (document.readyState === "complete" && document.body) {
          clearTimeout(timeout);
          obs.disconnect();
          resolve(document.body);
        }
      });

      // Start observing
      observer.observe(document.documentElement, {
        childList: true,
        subtree: true,
      });
    });
  }
  function checkToken() {
    const token = GM_getValue("myToken", "");
    if (!token) {
      createLoginForm();
      return false;
    }
    return true;
  }
  async function fetchUserInfo() {
    try {
      const response = await fetch(
        baseUrl + "/business/member/info?siteCode=" + siteCode,
        {
          method: "GET",
          headers: {
            Authorization: `Bearer ${GM_getValue("myToken", "")}`,
          },
        }
      );
      const data = await response.json();
      vipUser = data.data;
      if (data.code == 401) {
        createLoginForm();
      }
      return data.data;
    } catch (error) {
      console.error("Error fetching user info:", error);
      throw error;
    }
  }
  async function postAsync(url, body) {
    try {
      const response = await fetch(url, {
        method: "POST",
        headers: {
          "Content-Type": "application/json",
          Authorization: `Bearer ${GM_getValue("myToken", "")}`, // Add token from localStorage
        },
        body: JSON.stringify(body),
      });
      const result = await response.json();
      if (!response.ok) {
        throw new Error(`HTTP 错误!状态码:${response.status}`);
      }
      if (result.code != 200) {
        showApiMessage(result.msg, "error");
      }
      if (result.code == 401) {
        GM_setValue("myToken", "");
        createLoginForm();
      }
      return result;
    } catch (error) {
      showApiMessage("操作失败", "error");
      throw error;
    }
  }
  async function putAsync(url, body) {
    try {
      const response = await fetch(url, {
        method: "PUT",
        headers: {
          "Content-Type": "application/json",
          Authorization: `Bearer ${GM_getValue("myToken", "")}`,
        },
        body: JSON.stringify(body),
      });
      const result = await response.json();
      if (!response.ok) {
        throw new Error(`HTTP 错误!状态码:${response.status}`);
      }
      if (result.code != 200) {
        showApiMessage(result.msg, "error");
      }
      if (result.code == 401) {
        GM_setValue("myToken", "");
        createLoginForm();
      }
      return result;
    } catch (error) {
      showApiMessage("操作失败", "error");
      throw error;
    }
  }
  async function getAsync(url, params) {
    try {
      // Build query string from params if they exist
      const queryString = params
        ? "?" + new URLSearchParams(params).toString()
        : "";
      const response = await fetch(baseUrl + url + queryString, {
        method: "GET",
        headers: {
          Authorization: `Bearer ${GM_getValue("myToken", "")}`,
        },
      });
      const result = await response.json();
      if (!response.ok) {
        throw new Error(`HTTP 错误!状态码:${response.status}`);
      }
      if (result.code != 200 && result.code != 401) {
        showApiMessage(result.msg, "error");
      }
      if (result.code == 401) {
        GM_setValue("myToken", "");
        createLoginForm();
      }
      return result;
    } catch (error) {
      showApiMessage("操作失败", "error");
      throw error;
    }
  }
  async function checkPermission() {
    var result = await getAsync("/business/member/has-permission", {
      siteCode: siteCode,
    });
    if (result.code == 200) {
      hasPrev = true;
      return true;
    } else {
      return false;
    }
  }
  async function serializeVideo() {
    let params = Object.assign(
      {},
      {
        videoId: slefVideo.id,
        signData: slefVideo.signData,
        siteCode: siteCode,
      }
    );
    let result = await postAsync(baseUrl + "/business/video/serialize", params);
    slefVideo.m3u8Url = result.data.m3u8Url;
    m3u8Url = slefVideo.m3u8Url;
  }

  async function gcreateDownloadUrl() {
    if (slefVideo.m3u8Url) {
      let params = Object.assign(
        {},
        {
          videoId: slefVideo.id,
          m3u8Url: slefVideo.m3u8Url,
          siteCode: siteCode,
        }
      );
      return await postAsync(
        baseUrl + "/business/download/generate-token",
        params
      );
    } else {
      showApiMessage("请等待视频加载完成");
    }
  }

  function waitForElement(selector) {
    return new Promise((resolve) => {
      if (document.querySelector(selector)) {
        return resolve(document.querySelector(selector));
      }

      const observer = new MutationObserver((mutations, obs) => {
        if (document.querySelector(selector)) {
          obs.disconnect();
          resolve(document.querySelector(selector));
        }
      });

      observer.observe(document.body, {
        childList: true,
        subtree: true,
      });
    });
  }
  function removeAdds() {
    // waitForElement(".van-list").then((list) => {
    //   if (list.childNodes && list.childNodes.length > 0) {
    //    list.removeChild(list.childNodes[1]);
    //  }
    // });

    waitForElement(".top-swipe").then((e) => {
      e.style.display = "none";
    });
    waitForElement(".overAllList-wrap").then((e) => {
      e.style.display = "none";
    });
    waitForElement(".floatMenu").then((e) => {
      e.style.display = "none";
    });
    waitForElement("#SplashScreenID").then((e) => {
      e.style.display = "none";
    });
    waitForElement(".globalLaunch").then((e) => {
      e.style.display = "none";
    });
  }

  // 使用示例
})();