// ==UserScript== // @name Play-With-MPV // @name:zh 使用 MPV 播放 // @namespace https://github.com/LuckyPuppy514 // @version 3.0.1 // @author LuckyPuppy514 // @copyright 2023, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514) // @license MIT // @description 使用 MPV 播放网页上的视频 // @homepage https://github.com/LuckyPuppy514/Play-With-MPV // @icon https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/mpv.png // @match https://www.bilibili.com/bangumi/play/* // @match https://www.bilibili.com/video/* // @match https://live.bilibili.com/* // @match https://ddys.art/* // @match https://ddys.pro/* // @match https://libvio.fun/play/* // @match https://libvio.me/play/* // @match https://www.libvio.me/play/* // @match https://sh-data-s02.chinaeast2.cloudapp.chinacloudapi.cn/*?url=* // @match https://p.cfnode1.xyz/*?url=* // @match https://www.nivod.tv/* // @match https://www.pkmp4.com/py/* // @match https://www.pkmp4.com/addons/dplayer/?url=* // @match https://www.btnull.org/py/* // @match https://www.btnull.to/py/* // @match https://www.btnull.nu/py/* // @match https://www.btnull.in/py/* // @match *://www.996dm.com/play/* // @match *://www.dmlaa.com/play/* // @match *://www.qdmsh.com/play/* // @match https://danmu.yhdmjx.com/*?url=* // @match https://dick.xfani.com/watch/* // @match https://dick.xfani.com/addons/dp/player/* // @match https://m3.moedot.net/muiplayer/?url=* // @match https://www.mgnacg.com/bangumi/* // @match https://play.mknacg.top:8585/* // @match https://www.omofun.top/index.php/vod/play/id/* // @match https://player.omofun.top/?url=* // @match https://spdcat.net/vodplay/* // @match https://spdcat.net/addons/dp/player/* // @match http://www.dm88.me/player/* // @match https://jianghu.live2008.com/*?url=* // @match https://www.kk151.com/play/* // @match https://jx.m3u8.tv/jiexi/?url=* // @match https://jx.wolongzywcdn.com:65/m3u8.php?url=* // @match https://www.m3u8.tv.cdn.8old.cn/jx.php?url=* // @match https://jx.wujinkk.com/dplayer/?url=* // @match https://www.ikdmjx.com/?url=* // @match https://hls.kuaibofang.com/?url=* // @match https://jx.jxbdzyw.com/m3u8/?url=* // @match https://hdzyk.com/?m=* // @match https://1080zyk1.com/?m=* // @match https://1080zyk2.com/?m=* // @match https://1080zyk3.com/?m=* // @match https://1080zyk4.com/?m=* // @match https://1080zyk5.com/?m=* // @match https://vip.zykbf.com/?url=* // @match https://www.bdys01.com/* // @match *://*/*.mp4 // @match *://*/*.mkv // @match https://www.youtube.com/* // @match https://ani.gamer.com.tw/animeVideo.php?sn=* // @connect api.bilibili.com // @connect api.live.bilibili.com // @require https://unpkg.com/jquery@3.2.1/dist/jquery.min.js // @grant GM_setValue // @grant GM_getValue // @run-at document-end // @downloadURL none // ==/UserScript== 'use strict'; const INFO = ` ▶️🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽🔽◀️ ▶️ ◀️ ▶️ Play-With-MPV ◀️ ▶️ ◀️ ▶️ https://github.com/LuckyPuppy514/Play-With-MPV ◀️ ▶️ ◀️ ▶️ © 2023 LuckyPuppy514 ◀️ ▶️ ◀️ ▶️🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼🔼◀️ `; // gm key const KEY = { config: "config" } // 默认配置 const DEFAULT_CONFIG = { player: "mpv", mpvPath: "", proxy: "", bestQuality: "2160p", bilibiliCodecs: 12, playAuto: 0, syncStartTime: 0, regVersion: "20221230", version: "20221230" }; var currentConfig; // 视频链接匹配正则 const VIDEO_URL_REGEX = /https?:\/\/(?![^"^']*http)[^"^']+(\.|%2e)(m3u8|m3u|mp4|mkv|flv|avi)(|\?[\w&]+)/g; // 父子页面方法名 const METHOD = { pause: "PAUSE", report: "REPORT" }; // 时间 ms const TIME = { out: 3000, toast: 3000, refresh: 600, reportInterval: 600, pauseInterval: 2000 } // 尝试次数 var tryTime = 0; const TRY_TIME = { maxPause: 5, maxParse: 8 }; // 播放器配置 const PLAYER = { mpv: { name: "mpv", params: { videoUrl: 'mpv://"${videoUrl}" ', audioUrl: '--audio-file="${audioUrl}" ', title: '--force-media-title="${title}" ', startTime: '--start=${startTime} ', referer: '--http-header-fields="referer: ${referer}" ', origin: '--http-header-fields="origin: ${origin}" ', proxy: '--http-proxy=${proxy} --ytdl-raw-options=proxy=[${proxy}] ', other: '${other} ' } }, potplayer: { name: "potplayer", params: { videoUrl: 'potplayer://${videoUrl} /current ', title: '/title="${title}" ', startTime: '/seek=${startTime} ', referer: '/referer="${referer}" ', origin: '/headers="origin: ${origin}" ', proxy: '/user_agent="${proxy}" ' } } } // 页面信息 var page = { title: undefined, host: undefined, url: undefined, isFullScreen: false, }; // 处理器 var handler; // 前缀 const PREFIX = "pwm"; // 组件 id const ID = { loadingDiv: `${PREFIX}-loading-div`, toastDiv: `${PREFIX}-toast-div`, buttonDiv: `${PREFIX}-button-div`, aboutButton: `${PREFIX}-about-button`, playButton: `${PREFIX}-play-button`, settingButton: `${PREFIX}-setting-button`, settingDiv: `${PREFIX}-setting-div`, settingTable: `${PREFIX}-setting-table`, playerSelect: `${PREFIX}-player-input`, mpvPathInput: `${PREFIX}-mpv-path-input`, proxyInput: `${PREFIX}-proxy-input`, bestQualitySelect: `${PREFIX}-best-quality-select`, bilibiliCodecsSelect: `${PREFIX}-bilibili-codecs-select`, saveButton: `${PREFIX}-save-button`, downloadButton: `${PREFIX}-download-button`, playAutoInput: `${PREFIX}-play-auto-input`, syncStartTimeInput: `${PREFIX}-sync-start-time-input`, aboutDiv: `${PREFIX}-about-div`, aboutTable: `${PREFIX}-about-table`, } // 组件 class const CLASS = { button: `${PREFIX}-button`, titleSpan: `${PREFIX}-title-span-class`, titleTd: `${PREFIX}-title-td-class`, closeButton: `${PREFIX}-cloase-button-class`, tipSpan: `${PREFIX}-tip-span-class`, footerSpan: `${PREFIX}-footer-span-class`, switchLabel: `${PREFIX}-footer-label-class`, sliderSpan: `${PREFIX}-slider-span-class`, roundSpan: `${PREFIX}-round-span-class`, customSelect: `${PREFIX}-custom-select-class`, readOnly: `${PREFIX}-read-only-class`, } // 消息类型 TOAST_TYPE = { info: "info", warn: "warn", error: "error" } const CSS = ` #${ID.loadingDiv} { display: none; position: fixed; bottom: 50%; left: 50%; z-index: 999999; transform: translate(-50%, -50%); background-color: rgba(255, 255, 255, 0); } .spinner { width: 40px; height: 40px; background-color: rgba(255, 255, 255, 1); -webkit-animation: sk-rotateplane 1.2s infinite ease-in-out; animation: sk-rotateplane 1.2s infinite ease-in-out; } @-webkit-keyframes sk-rotateplane { 0% { -webkit-transform: perspective(120px) } 50% { -webkit-transform: perspective(120px) rotateY(180deg) } 100% { -webkit-transform: perspective(120px) rotateY(180deg) rotateX(180deg) } } @keyframes sk-rotateplane { 0% { transform: perspective(120px) rotateX(0deg) rotateY(0deg); -webkit-transform: perspective(120px) rotateX(0deg) rotateY(0deg) } 50% { transform: perspective(120px) rotateX(-180.1deg) rotateY(0deg); -webkit-transform: perspective(120px) rotateX(-180.1deg) rotateY(0deg) } 100% { transform: perspective(120px) rotateX(-180deg) rotateY(-179.9deg); -webkit-transform: perspective(120px) rotateX(-180deg) rotateY(-179.9deg); } } #${ID.toastDiv} { display: none; position: fixed; bottom: 80%; left: 50%; max-width: 60%; min-width: 150px; padding: 0 14px; height: 40px; color: rgb(255, 255, 255); line-height: 40px; text-align: center; border-radius: 4px; transform: translate(-50%, -50%); z-index: 999999; background: rgba(0, 255, 0, .9); font-size: 14px; font-weight: blod; } /* 按钮 */ ${ID.buttonDiv} { display: none; } .${CLASS.button} { position: fixed; cursor: pointer; z-index: 99999; border: none; border-radius: 50%; background-size: cover; background-color: rgba(255, 255, 255, 0); } #${ID.playButton} { bottom: 10px; left: 16px; width: 50px; height: 50px; background-image: url(); } #${ID.playButton}:hover { bottom: 8px; left: 14px; width: 54px; height: 54px; } /* 设置 */ #${ID.settingButton} { bottom: 50px; left: 56px; width: 28px; height: 28px; background-image: url(); } #${ID.settingButton}:hover { bottom: 48px; left: 52px; width: 32px; height: 32px; } #${ID.settingDiv} { position: fixed; top: 40%; left: 50%; transform: translate(-50%, -50%); z-index: 99999; width: 780px; height: 410px; border: 6px solid rgba(255, 255, 255, 0.5); background-color: rgba(0, 113, 255, 1); display: none; flex-direction: column; border-radius: 6px; align-items: center; color: rgba(255, 255, 255, 1); font-family: "微软雅黑"; } #${ID.settingTable} { margin-top: 10px; width: 680px; height: 330px; border-radius: 5px !important; border: 3px solid rgba(255, 255, 255, 1) !important; text-align: left; border-collapse: unset !important; } #${ID.settingTable} td { font-size: 14px; border: 0px solid rgba(255, 255, 255, 0.5); padding-top: 18px; } .${CLASS.titleSpan} { padding-top: 15px; font-size: 16px; font-weight: bold; } .${CLASS.closeButton} { position: absolute; top: 4px; right: 3px; height: 25px; width: 40px; border: none; font-size: 17px; background-color: rgba(0, 0, 0, 0); line-height: 0px; border-radius: 3px; } .${CLASS.closeButton}:hover { font-size: 20px; background-color: rgba(255, 255, 255, .5); cursor: pointer; } .${CLASS.tipSpan} { font-size: xx-small; color: yellow; position: fixed; padding-left: 5px; padding-top: 9px; } .${CLASS.titleTd} { width: 100px; height: 30px; border: none; font-size: 14px; padding-right: 15px; text-align: right; } #${ID.settingTable} input { font-size: 12px; width: 420px; height: 26px; border: none; outline: none; padding-left: 6px; border-radius: 2px; color: rgba(0, 0, 0, 1); background-color: rgba(255, 255, 255, 1); cursor: auto; display: inline-block !important; margin-top: 1px !important; margin-bottom: 1px !important; } #${ID.settingTable} input::placeholder { font-size: 12px; color: rgba(0, 0, 0, .6); } #${ID.saveButton} { font-size: 14px; margin-left: 185px; width: 300px; height: 30px; border: none; border-radius: 3px; color: rgba(255, 255, 255, 1); background-color: rgba(255, 255, 255, .6); } #${ID.downloadButton} { font-size: x-small; margin-left: 10px; width: 80px; height: 30px; border: none; border-radius: 3px; color: rgba(255, 255, 255, 1); background-color: rgba(255, 255, 255, .6); } #${ID.saveButton}:hover, #${ID.downloadButton}:hover { background-color: rgba(0, 255, 0, .7); cursor: pointer; } .${CLASS.customSelect} { position: relative; display: inline-block; width: 90px; height: 25px; border: none; outline: none; padding-left: 6px; border-radius: 3px; color: rgba(0, 0, 0, 1); background-color: rgba(255, 255, 255, 1); cursor: pointer; font-size: 12px; } .${CLASS.footerSpan} { margin-top: 10px; margin-bottom: 10px; color: rgba(255, 255, 255, 1); } .${CLASS.footerSpan} a { color: rgba(255, 255, 255, 1); text-decoration: none; font-size: 14px; margin-bottom: 1px; display: inline-block; } /* switch */ .${CLASS.switchLabel} { position: relative; display: inline-block; width: 50px; height: 21px; } .${CLASS.switchLabel} input { opacity: 0; width: 0 !important; height: 0 !important; } .${CLASS.sliderSpan} { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background-color: rgba(255, 255, 255, .6); -webkit-transition: .4s; transition: .4s; } .${CLASS.sliderSpan}:before { position: absolute; content: ""; height: 13px; width: 13px; left: 4px; bottom: 4px; background-color: rgba(255, 255, 255, 1); -webkit-transition: .4s; transition: .4s; } input:checked + .${CLASS.sliderSpan} { background-color: rgba(0, 255, 0, .7); } input:focus + .${CLASS.sliderSpan} { box-shadow: 0 0 1px rgba(0, 255, 0, .7); } input:checked + .${CLASS.sliderSpan}:before { -webkit-transform: translateX(29px); -ms-transform: translateX(29px); transform: translateX(29px); } .${CLASS.sliderSpan}.${CLASS.roundSpan} { border-radius: 34px; } .${CLASS.sliderSpan}.${CLASS.roundSpan}:before { border-radius: 50%; } .${CLASS.readOnly} { color: rgba(255, 0, 0, .9) !important; background-color: rgba(255, 0, 0, .9) !important; cursor: default !important; } #${ID.aboutButton} { bottom: 52px; left: 3px; width: 25px; height: 25px; background-image: url(); } #${ID.aboutButton}:hover { bottom: 50px; left: 1px; width: 29px; height: 29px; } `; const HTML = `
Play-With-MPV
播放软件 选择 potplayer 时:软件路径,最高画质,视频编码无效
软件路径 mpv.exe 完整路径
代理设置 油管和巴哈专属
最高画质 油管和B站专属 视频编码 B站专属
自动播放 解析成功自动播放 同步时间 同步网页播放时间
© 2023 LuckyPuppy514 🆕 👻
`; const REG = `Windows Registry Editor Version 5.00 [HKEY_LOCAL_MACHINE\\SOFTWARE\\Policies\\Google\\Chrome] "ExternalProtocolDialogShowAlwaysOpenCheckbox"=dword:00000001 [HKEY_LOCAL_MACHINE\\SOFTWARE\\Policies\\Microsoft\\Edge] "ExternalProtocolDialogShowAlwaysOpenCheckbox"=dword:00000001 [HKEY_CLASSES_ROOT\\mpv] @="mpv Protocol" "URL Protocol"="" [HKEY_CLASSES_ROOT\\mpv\\DefaultIcon] @="" [HKEY_CLASSES_ROOT\\mpv\\shell] @="" [HKEY_CLASSES_ROOT\\mpv\\shell\\open] @="" [HKEY_CLASSES_ROOT\\mpv\\shell\\open\\command] @="cmd /V:ON /C \\"FOR /F \\"tokens=* USEBACKQ\\" %%F IN (\`C:\\\\Windows\\\\System32\\\\WindowsPowerShell\\\\v1.0\\\\powershell.exe -command \\"Add-Type -AssemblyName System.Web;[System.Web.HTTPUtility]::UrlDecode('%1')\\"\`) DO (SET param=%%F) & SET param=!param:mpv://=! & start /min \${MPV_PATH} !param!\\"" ` function appendCSS() { let css = document.createElement("style"); css.innerHTML = CSS.trim(); document.head.appendChild(css); } function appendHTML() { let div = document.createElement("div"); div.innerHTML = HTML.trim(); document.body.appendChild(div); } function loading(visiable) { let loadingDiv = document.getElementById(ID.loadingDiv); if (visiable) { loadingDiv.style.display = "flex"; setTimeout(() => { if (loadingDiv.style.display == "flex") { document.getElementById(ID.loadingDiv).style.display = "none"; toast("超时辣 ...... 😓", TOAST_TYPE.error); } }, TIME.out); } else { loadingDiv.style.display = "none"; } } function toast(message, type, duration) { type = type ? type : TOAST_TYPE.info; duration = isNaN(duration) ? TIME.toast : duration; let toastDiv = document.getElementById(ID.toastDiv); toastDiv.innerHTML = message; toastDiv.style.display = "block"; if (type == TOAST_TYPE.info) { toastDiv.style.backgroundColor = "rgba(75, 180 ,54, 1)"; } else if (type == TOAST_TYPE.warn) { toastDiv.style.backgroundColor = "rgba(190, 190, 70, 1)"; } else if (type == TOAST_TYPE.error) { toastDiv.style.backgroundColor = "rgba(210, 51, 35, 1)"; } setTimeout(() => { toastDiv.style.display = "none"; }, duration); } function addListener() { let buttonDiv = document.getElementById(ID.buttonDiv); let playButton = document.getElementById(ID.playButton); let settingButton = document.getElementById(ID.settingButton); let settingDiv = document.getElementById(ID.settingDiv); let playerSelect = document.getElementById(ID.playerSelect); let mpvPathInput = document.getElementById(ID.mpvPathInput); let proxyInput = document.getElementById(ID.proxyInput); let bestQualitySelect = document.getElementById(ID.bestQualitySelect); let bilibiliCodecsSelect = document.getElementById(ID.bilibiliCodecsSelect); let playAutoInput = document.getElementById(ID.playAutoInput); let syncStartTimeInput = document.getElementById(ID.syncStartTimeInput); let downloadButton = document.getElementById(ID.downloadButton); let saveButton = document.getElementById(ID.saveButton); let closeButtons = document.getElementsByClassName(CLASS.closeButton); let aboutButton = document.getElementById(ID.aboutButton); switchStatus(downloadButton, false); // 播放按钮 playButton.onclick = function () { if (currentConfig.player == PLAYER.mpv.name) { let message = undefined; if (!currentConfig.mpvPath) { message = "请先进行设置"; } else if (!currentConfig.regVersion) { message = "请先下载注册表"; } else if (currentConfig.regVersion != DEFAULT_CONFIG.regVersion) { message = "注册表有更新,请重新下载注册表"; } if (message) { toast(message, TOAST_TYPE.warn); settingDiv.style.display = "none"; settingButton.click(); return; } } loading(true); try { playButtonClickLimit(); handler.play(); handler.pause(); } catch (error) { toast("出错辣 ...... 😓", TOAST_TYPE.error); console.log(error); } loading(false); } // 设置按钮 settingButton.onclick = function () { let display = settingDiv.style.display; if (display == "flex") { settingDiv.style.display = "none"; } else { settingDiv.style.display = "flex"; // 加载配置 playerSelect.value = currentConfig.player; mpvPathInput.value = currentConfig.mpvPath; proxyInput.value = currentConfig.proxy; bestQualitySelect.value = currentConfig.bestQuality; bilibiliCodecsSelect.value = currentConfig.bilibiliCodecs; playAutoInput.checked = currentConfig.playAuto == 1 ? true : false; syncStartTimeInput.checked = currentConfig.syncStartTime == 1 ? true : false; switchPlayer(playerSelect.value); } } // 播放器选择框 playerSelect.onchange = function () { switchPlayer(this.value); } // 保存按钮 saveButton.onclick = function () { let oldMpvPath = currentConfig.mpvPath; let newMpvPath = mpvPathInput.value; if (playerSelect.value == PLAYER.mpv.name) { if (!newMpvPath) { toast("软件路径不能为空", TOAST_TYPE.error); return; } if (/.*[\u4e00-\u9fa5 ]+.*/g.test(newMpvPath)) { toast("软件路径不能包含中文或空格", TOAST_TYPE.error); return; } newMpvPath = newMpvPath.replace(/[\\|/]+/g, "//"); if (!newMpvPath.endsWith(".com") && !newMpvPath.endsWith(".exe")) { if (!newMpvPath.endsWith("//")) { newMpvPath = newMpvPath + "//"; } if (newMpvPath.toLowerCase().indexOf("mpvnet") != -1 || newMpvPath.toLowerCase().indexOf("mpv.net") != -1) { newMpvPath = newMpvPath + "mpvnet.exe"; } else { newMpvPath = newMpvPath + "mpv.exe"; } } mpvPathInput.value = newMpvPath; currentConfig.mpvPath = newMpvPath; currentConfig.bestQuality = bestQualitySelect.value; currentConfig.bilibiliCodecs = bilibiliCodecsSelect.value; switchStatus(downloadButton, mpvPathInput.value ? true : false); } currentConfig.proxy = proxyInput.value; currentConfig.player = playerSelect.value; currentConfig.playAuto = playAutoInput.checked ? 1 : 0; currentConfig.syncStartTime = syncStartTimeInput.checked ? 1 : 0; GM_setValue(KEY.config, currentConfig); if (oldMpvPath != newMpvPath) { toast("软件路径已修改,请重新下载注册表", TOAST_TYPE.warn); } else { toast("保存成功"); } playButtonClickLimit(); init(); } // 下载按钮 downloadButton.onclick = function () { currentConfig.regVersion = DEFAULT_CONFIG.regVersion; GM_setValue(KEY.config, currentConfig); console.log(DEFAULT_CONFIG); console.log(currentConfig); var a = document.createElement('a'); var blob = new Blob([REG.replace("${MPV_PATH}", currentConfig.mpvPath)], { 'type': 'application/octet-stream' }); a.href = window.URL.createObjectURL(blob); a.download = "mpv.reg"; a.click(); } // 关闭按钮 for (let closeButton of closeButtons) { closeButton.onclick = function () { settingDiv.style.display = "none"; } } // 关于按钮 aboutButton.onclick = function () { window.open("https://www.lckp.top/play-with-mpv/index.html", "_blank"); } // 切换播放器 function switchPlayer(player) { if (player == PLAYER.mpv.name) { switchStatus(mpvPathInput, true); switchStatus(bestQualitySelect, true); switchStatus(bilibiliCodecsSelect, true); if (mpvPathInput.value) { switchStatus(downloadButton, true); } } else if (player == PLAYER.potplayer.name) { switchStatus(mpvPathInput, false); switchStatus(bestQualitySelect, false); switchStatus(bilibiliCodecsSelect, false); switchStatus(downloadButton, false); } } // 全屏 document.addEventListener("fullscreenchange", () => { if (document.fullscreenElement) { page.isFullScreen = true; buttonDiv.style.display = "none"; } else { page.isFullScreen = false; if (handler.media.videoUrl) { buttonDiv.style.display = "flex"; } } }); // 限制播放按钮点击频率 function playButtonClickLimit() { playButton.disabled = true; setTimeout(() => { playButton.disabled = false; }, TIME.pauseInterval); } } // 切换元素状态 function switchStatus(element, flag) { if (flag) { element.readOnly = false; element.disabled = false; element.classList.remove(CLASS.readOnly); } else { element.readOnly = true; element.disabled = true; element.classList.add(CLASS.readOnly); } } function sleep(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } // 加载配置 function loadConfig() { let oldConifg = GM_getValue(KEY.config); if (!oldConifg) { currentConfig = copy(DEFAULT_CONFIG); currentConfig.regVersion = ""; GM_setValue(KEY.config, currentConfig); } else { if (oldConifg.version != DEFAULT_CONFIG.version) { currentConfig = copy(DEFAULT_CONFIG); for (const key in oldConifg) { config[key] = oldConifg[key]; } currentConfig.version = DEFAULT_CONFIG.version; GM_setValue(KEY.config, currentConfig); } else { currentConfig = oldConifg; GM_setValue(KEY.config, currentConfig); } } } // 复制 function copy(oldBean) { let newBean = {}; for (const key in oldBean) { newBean[key] = oldBean[key]; } return newBean; } class Media { constructor() { this.title = ""; this.videoUrl = ""; this.audioUrl = ""; this.startTime = ""; this.referer = ""; this.origin = ""; this.proxy = ""; this.other = ""; } setTitle(title) { this.title = title; } setVideoUrl(videoUrl) { if (this.check(videoUrl)) { this.videoUrl = videoUrl; if (document.getElementById(ID.buttonDiv)) { document.getElementById(ID.buttonDiv).style.display = "flex"; if (currentConfig.playAuto == 1) { document.getElementById(ID.playButton).click(); } } } } setAudioUrl(audioUrl) { this.audioUrl = audioUrl; } setStartTime(startTime) { this.startTime = Math.floor(startTime); } setReferer(referer) { this.referer = referer; } setOrigin(origin) { this.origin = origin; } setProxy(proxy) { this.proxy = proxy; } setOther(other) { this.other = other; } // 检查视频链接是否有效 check(videoUrl) { if (videoUrl && videoUrl.startsWith("http")) { if (videoUrl.match(/(\.m3u|\.m3u8)/g)) { let m3u8 = ""; $.ajax({ type: "GET", url: videoUrl, async: false, success: function (res) { m3u8 = res; } }); if (m3u8 && m3u8.indexOf("png") != -1) { console.log("m3u8 链接无法播放:" + videoUrl); return false; } } if (videoUrl.startsWith("https://www.mp4")) { return false; } return true; } console.log(`链接无效:${videoUrl}`); return false; } } class BaseHandler { constructor() { loadConfig(); this.media = new Media(); for (const key in PLAYER) { if (PLAYER[key].name == currentConfig.player) { this.player = PLAYER[key]; break; } } if (window.self == window.top) { if (!document.getElementById(ID.buttonDiv)) { console.log(INFO); appendCSS(); appendHTML(); addListener(); } document.getElementById(ID.buttonDiv).style.display = "none"; } this.media.setTitle(document.title); } async parse() { } pause() { let videos = document.getElementsByTagName("video"); if (videos && videos.length > 0) { let i = 0; while (i < TRY_TIME.maxPause) { setTimeout(function () { for (let video of videos) { video.pause(); } }, TIME.pauseInterval * i); i++; } } else if (this.iframe) { this.iframe.postMessage({ method: METHOD.pause }, "*"); } } play() { let link = ""; for (let key in this.player.params) { if (key == "title") { continue; } if (key == "startTime") { if (currentConfig.syncStartTime != 1) { continue; } else if (!this.media.startTime) { let video = document.getElementsByTagName("video")[0]; if (video) { this.media.setStartTime(video.currentTime); } } } let value = this.media[key]; if (value) { let param = this.player.params[key]; do { param = param.replace('${' + key + '}', value); } while (param.indexOf('${' + key + '}') != -1) link = link + param; } } if (this.media.title) { let maxLength = 2000 - link.length; let title = encodeURIComponent(this.media.title); if (title.length > maxLength) { title = title.substring(0, maxLength) + '...'; } link = link + this.player.params.title.replace('${title}', title); } window.open(link, "_self"); } // 监听子页面事件 addIframeListener() { let that = this; window.addEventListener("message", function (event) { that.iframe = event.source; if (event.data.method == METHOD.pause) { that.pause(); } else if (event.data.method == METHOD.report) { that.media.setStartTime(event.data.media.startTime); if (!that.media.videoUrl) { that.media.setVideoUrl(event.data.media.videoUrl); } } }, false); } // 监听顶层页面事件 addTopListener() { let that = this; window.addEventListener("message", function (event) { if (event.data.method == METHOD.pause) { that.pause(); } }, false); // 定时上报当前视频信息 setInterval(() => { let video = document.getElementsByTagName("video")[0]; if (video) { this.media.setStartTime(video.currentTime); } window.top.postMessage({ method: METHOD.report, media: that.media }, "*"); }, TIME.reportInterval); } // yt-dlp 支持网站解析器 ytDlpParser() { return page.url; } // video 元素解析器 videoParser() { let videos = document.getElementsByTagName("video"); for (let video of videos) { let url = video.src; if (url && url.startsWith("http")) { return url; } } } // iframe 元素解析器 iframeParser() { let iframes = document.getElementsByTagName("iframe"); for (let iframe of iframes) { let urls = iframe.src.match(VIDEO_URL_REGEX); if (urls && urls.length > 0) { return urls[0]; } } } // html 解析器 htmlParser() { let urls = document.body.innerHTML.match(VIDEO_URL_REGEX); if (urls && urls.length > 0) { return urls[0]; } } // script 解析器 scriptParser() { for (let script of document.scripts) { let urls = script.innerHTML.match(VIDEO_URL_REGEX); if (urls && urls.length > 0) { return urls[0]; } } } // url 解析器 urlParser() { let urls = page.url.match(VIDEO_URL_REGEX); if (urls && urls.length > 0) { return urls[0]; } } } // 获取B站视频播放链接 function getBilibiliPlayUrl(avid, cid) { let fnval = 128; if (handler.player.params.audioUrl) { fnval = 4048; } $.ajax({ type: "GET", url: `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=${fnval}&avid=${avid}&cid=${cid}`, xhrFields: { withCredentials: true }, async: false, success: function (res) { handler.media.setOther(`--script-opts="cid=${cid}"`); if (handler.player.params.audioUrl) { let videoUrl = undefined; let audioUrl = undefined; let dash = res.data.dash; let hiRes = dash.flac; let dolby = dash.dolby; if (hiRes && hiRes.audio) { audioUrl = hiRes.audio.baseUrl; } else if (dolby && dolby.audio) { audioUrl = dolby.audio[0].base_url; } else { audioUrl = dash.audio[0].baseUrl; } let i = 0; while (i < dash.video.length && dash.video[i].id > BEST_QUALITY.bilibili[currentConfig.bestQuality]) { i++; } videoUrl = dash.video[i].baseUrl; let id = dash.video[i].id; while (i < dash.video.length) { if (dash.video[i].id != id) { break; } if (dash.video[i].codecid == currentConfig.bilibiliCodecs) { videoUrl = dash.video[i].baseUrl; break; } i++; } handler.media.setAudioUrl(audioUrl); handler.media.setVideoUrl(videoUrl); } else { handler.media.setVideoUrl(res.data.durl[0].url); } } }); } // function getBilibiliSubtitle(aid, cid) { // $.ajax({ // type: "GET", // url: `https://api.bilibili.com/x/player/v2?aid=${aid}&cid=${cid}`, // xhrFields: { // withCredentials: true // }, // async: true, // success: function (res) { // let subtitleUrl = res.data.subtitle.subtitles[0].subtitle_url; // console.log(subtitleUrl); // } // }); // } const BEST_QUALITY = { bilibili: { "unlimited": 127, "2160p": 126, "1440p": 116, "1080p": 116, "720p": 74, "480p": 32 }, bilibiliLive: { "unlimited": 4, "2160p": 4, "1440p": 4, "1080p": 4, "720p": 3, "480p": 2 }, youtube: { "unlimited": "", "2160p": "--ytdl-format=bestvideo[height<=?2160]%2Bbestaudio/best", "1440p": "--ytdl-format=bestvideo[height<=?1440]%2Bbestaudio/best", "1080p": "--ytdl-format=bestvideo[height<=?1080]%2Bbestaudio/best", "720p": "--ytdl-format=bestvideo[height<=?720]%2Bbestaudio/best", "480p": "--ytdl-format=bestvideo[height<=?480]%2Bbestaudio/best" } } var websiteList = [ { // ✅ https://www.bilibili.com/bangumi/play/ep508404 name: "B站影视", home: [ "https://www.bilibili.com" ], regex: /^https:\/\/www\.bilibili\.com\/bangumi\/play\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.media.setReferer("https://www.bilibili.com"); } async parse() { let epid = undefined; let epidElement = document.getElementsByClassName("ep-item cursor visited")[0]; if (!epidElement) { epidElement = document.getElementsByClassName('ep-item cursor')[0]; } if (epidElement) { epid = epidElement.getElementsByTagName('a')[0].href.match(/ep(\d+)/)[1]; } else { epidElement = document.getElementsByClassName("squirtle-pagelist-select-item active squirtle-blink")[0]; if (epidElement) { epid = epidElement.dataset.value; } } if (!epid) { return; } $.ajax({ type: "GET", url: `https://api.bilibili.com/pgc/view/web/season?ep_id=${epid}`, xhrFields: { withCredentials: true }, async: false, success: function (res) { let currentEpisode; let section = res.result.section; if (!section) { section = new Array(); } section.push({ episodes: res.result.episodes }); for (let i = section.length - 1; i >= 0; i--) { let episodes = section[i].episodes; for (const episode of episodes) { if (episode.id == epid) { currentEpisode = episode; break; } } if (currentEpisode) { break; } } getBilibiliPlayUrl(currentEpisode.aid, currentEpisode.cid); } }) } }, }, { // ✅ https://www.bilibili.com/video/BV1Hd4y1k7Vb // ✅ https://www.bilibili.com/video/av2 name: "B站投稿", home: [ "https://www.bilibili.com" ], regex: /^https:\/\/www\.bilibili\.com\/video\/(BV|av).*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.media.setReferer("https://www.bilibili.com"); } async parse() { let param = undefined; let videoId = page.url.substring(31); if (videoId.startsWith("BV")) { param = "bvid=" + videoId.match(/BV([0-9a-zA-Z]+)/)[1]; } else { param = "aid=" + videoId.match(/av([0-9]+)/)[1]; } $.ajax({ type: "GET", url: `https://api.bilibili.com/x/web-interface/view?${param}`, xhrFields: { withCredentials: true }, async: false, success: function (res) { let aid = res.data.aid; let cid = res.data.cid; let index = page.url.indexOf("?p="); if (index != -1 && res.data.pages.length > 1) { let p = page.url.substring(index + 3); let endIndex = p.indexOf("&"); if (endIndex != -1) { p = p.substring(0, endIndex); } cid = res.data.pages[p - 1].cid; } getBilibiliPlayUrl(aid, cid); } }) } }, }, { // ✅ https://live.bilibili.com/7777 name: "B站直播", home: [ "https://live.bilibili.com" ], regex: /^https:\/\/live\.bilibili\.com\/.*/g, handler: class Handler extends BaseHandler { async parse() { let iframe = document.getElementsByTagName("iframe")[0]; if (!iframe) { return; } let url = iframe.src; let index = url.indexOf("roomid="); if (index == -1) { return; } let roomid = url.substring(index + 7); roomid = roomid.substring(0, roomid.indexOf("&")); let that = this; $.ajax({ type: "GET", url: `https://api.live.bilibili.com/room/v1/Room/playUrl?quality=${BEST_QUALITY.bilibiliLive[currentConfig.bestQuality]}&cid=${roomid}`, async: false, xhrFields: { withCredentials: true }, success: function (res) { that.media.setVideoUrl(res.data.durl[0].url); } }); } }, }, { // ✅ https://ddys.art/bleach-thousand-year-blood-war name: "低端影视", home: [ "https://ddys.art", "https://ddys.pro" ], regex: /^https:\/\/(ddys\.art|ddys\.pro)\/.*/g, handler: class Handler extends BaseHandler { async parse() { let video = document.getElementsByTagName("video")[0]; if (video.paused) { document.getElementsByClassName('vjs-big-play-button')[0].click(); } this.media.setVideoUrl(this.videoParser()); let playing = document.getElementsByClassName("wp-playlist-playing")[0]; if (playing) { let episode = playing.textContent.replace(/(\n|\t|\d\.)/g, ""); if (episode != " 全") { this.media.title = document.getElementsByClassName("post-title")[0].textContent + episode + " - 低端影视"; } } } }, }, { // ✅ https://libvio.fun/play/714634-1-11.html name: "LIBVIO", home: [ "https://libvio.fun", "https://libvio.me", "https://www.libvio.me" ], regex: /^https:\/\/(libvio\.fun|libvio\.me|www\.libvio\.me)\/play\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "LIBVIO播放器", regex: /^https:\/\/(sh-data-s02\.chinaeast2\.cloudapp\.chinacloudapi\.cn|p\.cfnode1\.xyz)\/.*php\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(urls); } } }, { // ✅ https://www.nivod.tv/UXEwMmLqnUjHG5e4MwmlvmVnWiAJ9rIQ-RofV7wPhhed3uoi50mYsftLPq4mYyIhB-720-0-0-play.html?x=1 name: "泥视频", home: [ "https://www.nivod.tv", ], regex: /^https:\/\/www\.nivod\.tv\/.*play\.html?.*/g, handler: class Handler extends BaseHandler { async parse() { this.media.setVideoUrl(__dp.options.video.url); this.media.setTitle(document.title); } } }, { // ✅ https://www.pkmp4.com/py/268677-2-11.html name: "片库", home: [ "https://www.pkmp4.com", ], regex: /^https:\/\/www\.pkmp4\.com\/py\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } async parse() { this.media.setVideoUrl(player_aaaa.url); } } }, { name: "片库播放器", regex: /^https:\/\/www\.pkmp4\.com\/addons\/dplayer\/\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } } }, { // ✅ https://www.btnull.org/py/BBnLd_9.html?167094 name: "无名小站", home: [ "https://www.btnull.org", "https://www.btnull.to", "https://www.btnull.nu", "https://www.btnull.in", ], regex: /^https:\/\/(www.btnull.org|www.btnull.to|www.btnull.nu|www.btnull.in)\/py\/.*/g, handler: class Handler extends BaseHandler { async parse() { this.media.setVideoUrl(this.htmlParser()); } } }, { // ✅ https://www.996dm.com/play/6792-1-91.html // ✅ http://www.996dm.com/play/7800-1-10.html // ✅ http://www.dmlaa.com/play/7696-1-10.html // ✅ http://www.qdmsh.com/play/7663-1-10.html name: "樱花动漫网", home: [ "https://www.996dm.com", "http://www.996dm.com", "http://www.dmlaa.com", "http://www.qdmsh.com" ], regex: /^https?:\/\/(www\.996dm\.com|www\.dmlaa\.com|www\.qdmsh\.com)\/play\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "樱花动漫网播放器", regex: /^https:\/\/danmu\.yhdmjx\.com\/.*php\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(this.videoParser()); } } }, { // ✅ https://dick.xfani.com/watch/582/1/11.html name: "稀饭动漫", home: [ "https://dick.xfani.com" ], regex: /^https:\/\/dick\.xfani\.com\/watch\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "稀饭动漫播放器", regex: /(^https:\/\/dick\.xfani\.com\/addons\/dp\/player\/.*|^https:\/\/m3\.moedot\.net\/muiplayer\/\?url=.*)/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { if (config.url.indexOf(".m3u8") > 0 || config.url.indexOf(".mp4") > 0 || config.url.indexOf(".flv") > 0) { this.media.setVideoUrl(config.url); } else { let that = this; $.ajax({ type: "POST", url: "api_currentConfig.php", data: { "url": config.url, "time": config.time, "key": config.key, "title": config.title }, async: false, success: function (res) { if (res.code == "200") { that.media.setVideoUrl(res.url); } } }); } } } }, { // ✅ https://www.mgnacg.com/bangumi/426-6-12 name: "橘子动漫", home: [ "https://www.mgnacg.com" ], regex: /^https:\/\/www\.mgnacg\.com\/bangumi\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); this.media.setReferer("https://play.mknacg.top:8585"); } } }, { name: "橘子动漫播放器", regex: /^https:\/\/play\.mknacg\.top:8585\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(this.urlParser()); } } }, { // ✅ https://www.omofun.top/index.php/vod/play/id/17295/sid/2/nid/7.html name: "OmoFun", home: [ "https://www.omofun.top" ], regex: /^https:\/\/www\.omofun\.top\/index.php\/vod\/play\/id\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "OmoFun播放器", regex: /^https:\/\/player\.omofun\.top\/\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(this.urlParser()); } } }, { // ✅ https://spdcat.net/vodplay/135443-1-23 name: "迅猫动漫", home: [ "https://spdcat.net" ], regex: /^https:\/\/spdcat\.net\/vodplay\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "迅猫动漫播放器", regex: /^https:\/\/spdcat\.net\/addons\/dp\/player\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(config.url); setTimeout(() => { let conplay = document.getElementsByClassName("conplay-jump")[0]; if (conplay) { conplay.click(); } }, 1000); } } }, { // ✅ http://www.dm88.me/player/8480-0-11.html name: "樱花动漫", home: [ "http://www.dm88.me" ], regex: /^http:\/\/www\.dm88\.me\/player\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "樱花动漫播放器", regex: /^https:\/\/jianghu\.live2008\.com\/.*\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(url); } } }, { // ✅ https://www.kk151.com/play/15257-2-11.html name: "动漫之家", home: [ "https://www.kk151.com" ], regex: /^https:\/\/www\.kk151\.com\/play\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "动漫之家播放器", regex: /^https:\/\/(www\.ikdmjx\.com\/|jx\.wolongzywcdn\.com:65\/m3u8\.php|hls\.kuaibofang\.com\/|jx\.jxbdzyw\.com\/m3u8\/|www\.m3u8\.tv\.cdn\.8old\.cn\/m3u8tv1127\/api\.php|jx\.wujinkk\.com\/dplayer\/|jx\.m3u8\.tv\/jiexi\/)\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(this.urlParser()); } } }, { // ✅ https://hdzyk.com/?m=vod-play-id-27537-src-2-num-11.html // ✅ https://1080zyk2.com/?m=vod-play-id-27537-src-2-num-11.html name: "优质资源库", home: [ "https://hdzyk.com", "https://1080zyk1.com", "https://1080zyk2.com", "https://1080zyk3.com", "https://1080zyk4.com", "https://1080zyk5.com" ], regex: /^https:\/\/(hdzyk\.com|1080zyk[1-5]\.com)\/\?m=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addIframeListener(); } } }, { name: "优质资源库播放器", regex: /^https:\/\/vip\.zykbf\.com\/\?url=.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.addTopListener(); } async parse() { this.media.setVideoUrl(this.urlParser()); } } }, { // ✅ https://www.bdys01.com/play/22729-8.htm name: "哔嘀影视", home: [ "https://www.bdys01.com" ], regex: /^https:\/\/www\.bdys01\.com\/.*play\/.*/g, handler: class Handler extends BaseHandler { async parse() { this.media.setVideoUrl(this.videoParser()); if (!this.media.videoUrl) { this.media.setVideoUrl(this.htmlParser()); if (!this.media.videoUrl) { this.media.setVideoUrl(this.scriptParser()); } } } } }, { name: "AList", regex: /^https?:\/\/(alist[^\/]+|[^\/]+:5244)\/.*\.(mp4|mkv)/g, handler: class Handler extends BaseHandler { async parse() { let url = this.videoParser(); let index = url.indexOf("?"); if (index != -1) { url = url.substring(0, index + 1) + encodeURIComponent(url.substring(index + 1)); } this.media.setVideoUrl(url); this.media.setTitle(document.title); } } }, { // ✅ https://www.youtube.com/watch?v=IkGuTYaTsLo name: "YouTube", home: [ "https://www.youtube.com" ], regex: /^https:\/\/www\.youtube\.com\/(watch|playlist)\?.*/g, handler: class Handler extends BaseHandler { async parse() { if (page.url != "https://www.youtube.com/") { this.media.setTitle(""); this.media.setProxy(currentConfig.proxy); this.media.setOther(BEST_QUALITY.youtube[currentConfig.bestQuality]); this.media.setVideoUrl(this.ytDlpParser()); } } }, }, { // ✅ https://ani.gamer.com.tw/animeVideo.php?sn=32227 name: "巴哈姆特", home: [ "https://ani.gamer.com.tw" ], regex: /^https:\/\/ani\.gamer\.com\.tw\/animeVideo.php\?sn=.*/g, handler: class Handler extends BaseHandler { async parse() { this.media.setOrigin("https://ani.gamer.com.tw"); let index = page.url.indexOf("sn=") + 3; if (index == -1) { return; } let sn = page.url.substring(index); index = sn.indexOf("&"); if (index != -1) { sn = sn.substring(0, index); } let device = localStorage.ANIME_deviceid; let that = this; let res; $.ajax({ type: "GET", url: `https://ani.gamer.com.tw/ajax/m3u8.php?sn=${sn}&device=${device}`, async: false, xhrFields: { withCredentials: true }, success: function (json) { res = JSON.parse(json); } }) if (res.error) { if (res.error.code == 1015) { let oldDuration = document.getElementsByClassName("vjs-duration-display")[0].innerHTML; let newDuration = oldDuration; while (oldDuration == newDuration) { await sleep(1000); newDuration = document.getElementsByClassName("vjs-duration-display")[0].innerHTML; } tryTime = 0; } } else { that.media.setProxy(currentConfig.proxy); that.media.setVideoUrl(res.src); } } }, }, ]; // 初始化 async function init() { // 加载页面信息 page = { title: document.title, host: window.location.host, url: window.location.href, isFullScreen: false }; // 生成 handler for (let i = 0; i < websiteList.length; i++) { if (page.url.match(websiteList[i].regex)) { handler = new websiteList[i].handler(); break; } } // 尝试解析页面视频 if (handler) { tryTime = 0; while (tryTime < TRY_TIME.maxParse) { await sleep(tryTime * 1000 + 700); if (!handler.media.videoUrl) { try { await handler.parse(); } catch (error) { console.log('解析失败:' + error); } } tryTime++; } } } // 开始执行 init(); setInterval(() => { if (window.location.href != page.url) { init(); } }, TIME.refresh);