설치하기 전에, Sleazy Fork는 이 스크립트에 사용자가 아닌 스크립트 작성자의 이익을 위한 기능인 역기능이 포함되어 있음을 알려드립니다.
결제를 완료해야 스크립트의 모든 기능을 사용할 수 있습니다. Sleazy Fork는 결제 과정에 관여하지 않으므로, 유료 기능의 가치를 보증하거나 환불을 도와드릴 수 없습니다.
91社区vip视频下载脚本,支持在线下载、m3u8解析、hls播放器等功能
// ==UserScript== // @name 91社区 // @namespace http://tampermonkey.net/ // @version v1.5 // @description 91社区vip视频下载脚本,支持在线下载、m3u8解析、hls播放器等功能 // @author lx // @match https://9sex.com/ // @match https://9sex.com/index/movie/play/id/*.html // @match https://*/index/movie/play/id/* // @match http://*/index/movie/play/id/* // @include /^http(s)?:\/\/(www|h5)\.\w+\.xyz\/movie\/play\/id/ // @exclude *://www.haijiaom.cc/* // @run-at document-start // @grant unsafeWindow // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @grant GM_xmlhttpRequest // @connect haijiao.live // @charset UTF-8 // @antifeature payment // @license MIT // @icon  // ==/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 = "NINEONE"; let slefVideo = { videoPlayUrl: "", videoDuration: 0, id: 0, articleId: 0, videoBaseUrl: "", videoArticleId: "", downloadUrl: "", }; let baseDownadloadUrl = baseUrl + "/business/download?token="; let m3u8Url = ""; let hasPrev = false; waitForPageLoad() .then((body) => { // createLoginForm(); fetchUserInfo(); removeAdds(); createNavbar(body); initHlsPlayer(); Vip91(); }) .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 createLoginView() { const loginDialog = document.createElement("div"); let existingDialog = document.querySelector('[id^="login-dialog"]'); console.log("existingDialog", existingDialog); 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; } .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="sex_username" placeholder="用户名" class="login-input"> </div> <div style="margin-bottom: 20px;"> <input type="password" id="sex_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("sex_username").value; const password = document.getElementById("sex_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">×</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(); alert("即将跳转到注册页面..."); // 在这里可以添加跳转到注册页或显示注册模态框的逻辑 }; } function createPersonalInfoDialog() { if (!checkToken()) { return; } 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; } </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="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> `; // Add logout handler dialog.querySelector("#logoutBtn").addEventListener("click", () => { GM_setValue("myToken", ""); dialog.remove(); createToast("已退出登录", "success"); setTimeout(() => window.location.reload(), 1000); }); // Add close handler dialog.querySelector(".info-close").addEventListener("click", () => { dialog.remove(); }); document.body.appendChild(dialog); } 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 = result.data; 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 (!checkToken()) { return; } if (slefVideo.videoPlayUrl == "") { showApiMessage("当前页面没有视频,请先获取视频播放地址", "error"); return; } window.hlsPlayer.setContent(slefVideo.videoPlayUrl); window.hlsPlayer.play(); toggleVisibility(); } function generateM3U8FromVideo(slefVideo, duration) { // Generate segment duration between 1.0 and 1.25 seconds let segmentDuration = (Math.random() * 0.25 + 1.0).toFixed(6); const totalSegments = Math.ceil(duration / segmentDuration); let m3u8Content = "#EXTM3U" + "\r\n"; m3u8Content += "#EXT-X-VERSION:3" + "\r\n"; m3u8Content += "#EXT-X-TARGETDURATION:11" + "\r\n"; m3u8Content += "#EXT-X-MEDIA-SEQUENCE:0" + "\r\n"; // 添加加密信息 if (slefVideo.uri) { m3u8Content += `#EXT-X-KEY:METHOD=AES-128,URI="${slefVideo.videoBaseUrl}/${slefVideo.uri}"`; if (slefVideo.iv) { m3u8Content += `,IV=0x${slefVideo.iv}`; } m3u8Content += "\n"; } // 生成视频片段 for (let i = 0; i < totalSegments; i++) { const remainingDuration = duration - i * segmentDuration; const currentSegmentDuration = Math.min( segmentDuration, remainingDuration ); m3u8Content += `#EXTINF:${currentSegmentDuration},\n`; m3u8Content += `${slefVideo.videoBaseUrl}/${slefVideo.tsFileName}${i}.ts\n`; } m3u8Content += "#EXT-X-ENDLIST"; return m3u8Content; } function createBlobUrl(m3u8Content) { const blob = new Blob([m3u8Content], { type: "application/x-mpegURL" }); return URL.createObjectURL(blob); } 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; }, "api\\/favorite\\/v2\\/folderList(\\?[^/]*)?$": (response) => { return null; }, "api\\/topic\\/\\d+": (response) => { let decodeData = response.data; analysisVideo(decodeData); return response; }, }; async function analysisVideo(content) { let objectData = JSON.parse(atob(atob(atob(content))), `utf-8`); slefVideo.articleId = objectData.topicId; objectData.attachments.forEach((item) => { if (item.category == "video") { slefVideo.videoPlayUrl = item.remoteUrl; slefVideo.m3u8Url = item.remoteUrl; slefVideo.videoDuration = item.video_time_length; slefVideo.id = item.id; // Extract URI from preview URL } }); } 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 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(); } else { checkPermission(); } 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 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) { 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, m3u8Url: slefVideo.m3u8Url, siteCode: siteCode, } ); let result = await postAsync(baseUrl + "/business/video/serialize", params); } 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, }); }); } async function Vip91() { let ads = document.querySelector("#login-tip-modal"); if (ads) ads.parentNode.removeChild(ads); ads = document.querySelector(".vip-tip-modal"); if (ads) ads.parentNode.removeChild(ads); let login = await checkToken(); let arr = window.location.href.split("/"); slefVideo.id = arr[arr.length - 1].split(".")[0]; if (login) { let result = await checkPermission(); if (result) { const m3u891BaseUrl = "https://ms3.bca834d60257.com/"; // Wait for page load and check for pattern matches // Get all text content from the page const pageContent = document.body.innerHTML; // Define regex patterns to match const patterns = { // Add regex patterns to match here videoId: "movies/(.*?)_preview.jpg.txt", }; // Check each pattern Object.entries(patterns).forEach(([key, pattern]) => { const match = pageContent.match(pattern); if (match) { const baseAddress = match[1]; const resolutions = ["360P", "480P", "720P", "1080P"]; // Insert path fragment before the last segment const pathParts = baseAddress.split("/"); const lastPart = pathParts.pop(); // Remove last segment const basePath = pathParts.join("/"); const newPaths = resolutions.map((resolution) => { return `${m3u891BaseUrl}${basePath}/${resolution}/${lastPart}_${resolution}.m3u8`; }); slefVideo.m3u8Url = newPaths[0]; // Check if there's a video player on the page let videoPlayer = document.querySelector("#dplayer"); slefVideo.videoPlayUrl = newPaths[0]; // Initialize DPlayer with multiple quality options const dp = new DPlayer({ element: document.querySelector("#dplayer") || document.createElement("div"), video: { quality: [ { name: "360P", url: newPaths[0], type: "hls", }, { name: "480P", url: newPaths[1], type: "hls", }, { name: "720P", url: newPaths[2], type: "hls", }, { name: "1080P", url: newPaths[3], type: "hls", }, ], defaultQuality: 0, pic: "", autoplay: false, }, }); // Add error handling for video URLs // Add error handling for video URLs dp.on("error", (error) => { console.error("Player error:", error); showApiMessage("视频加载失败,请检查链接或刷新页面", "error"); }); // Add helpful tip below player const playerTip = document.createElement("div"); playerTip.style.cssText = ` background: rgba(0, 0, 0, 0.7); color: red; padding: 10px 20px; border-radius: 4px; font-size: 14px; text-align: center; z-index: 10000; `; playerTip.innerHTML = "温馨提示:部分视频只有360P清晰度,如何选择其他清晰度无法播放,刷新页面,请自行更换清晰度"; let title = document.querySelector(".ads"); title.appendChild(playerTip); } }); } } } function removeAdds() { GM_addStyle(".ad{ display: none; } #side-bar-service{display: none;}"); } })();