// ==UserScript== // @name Play-With-MPV // @name:zh 使用 MPV 播放 // @namespace https://github.com/LuckyPuppy514 // @version 3.8.1 // @author LuckyPuppy514 // @copyright 2023, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514) // @license MIT // @description 使用 mpv 播放网页中的视频,并支持 potplayer 及自定义播放器 // @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://www.bilibili.com/festival/* // @match https://www.bilibili.com/list/* // @match https://live.bilibili.com/* // @match https://www.ixigua.com/* // @match https://yun.nxflv.com/?url=* // @match https://ddys.art/* // @match https://ddys.pro/* // @include *://*.libvio.* // @match https://*.chinaeast2.cloudapp.chinacloudapi.cn/*?url=* // @match https://*.cfnode1.xyz/*?url=* // @match https://www.nivod.tv/* // @match https://www.pkmkv.com/py/* // @match https://www.pkmkv.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.926dm.com/play/* // @match *://www.dmlaa.com/play/* // @match *://www.qdmsh.com/play/* // @match *://www.ntdm8.com/play/* // @match https://danmu.yhdmjx.com/*?url=* // @match https://dick.xfani.com/watch/* // @match https://dick.xfani.com/addons/dp/player/* // @match https://player.moedot.net/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://*.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://*.yzzy-tv1.com/* // @match https://*.yzzy-tv-cdn.com/* // @match https://www.bdys10.com/* // @match https://www.haitu.tv/* // @include *://*alist* // @include *://*:5244* // @match *://*/*.mp4 // @match *://*/*.mkv // @match https://www.dora-family.com/Resource:TV // @match https://www.olehdtv.com/* // @match *://tkznp.com/vodplay/* // @match *://www.tkznp.com/vodplay/* // @match *://www.tkznp1.com/vodplay/* // @match *://www.tkznp2.com/vodplay/* // @match *://www.tkznp3.com/vodplay/* // @match *://www.tkznp4.com/vodplay/* // @match *://www.tkznp5.com/vodplay/* // @match *://www.tkznp6.com/vodplay/* // @match https://vip.ckllk.com/?url=* // @match https://www.hdmoli.com/* // @match https://play.qwertwe.top/xplay/?url=* // @match https://www.anfuns.cc/play/* // @match https://www.anfuns.cc/vapi/* // @match https://www.youtube.com/* // @match https://odysee.com/* // @match https://rumble.com/* // @match https://www.bitchute.com/* // @match https://ani.gamer.com.tw/animeVideo.php?sn=* // @match https://hanime1.me/watch?v=* // @match https://jable.tv/videos/* // @match https://ok.ru/* // @match https://tver.jp/* // @match https://www.lckp.top/play-with-mpv/index.html // @match https://www.douyin.com/ // @match https://www.douyin.com/video/* // @match https://www.douyin.com/discover?modal_id=* // @match https://www.mengfan.tv/play/* // @match https://video1.beijcloud.com/player/?url=* // @match https://www.tucao.cam/play/* // @match https://mypikpak.com/drive/* // @match https://www.icourse163.org/learn/* // @match https://www.iole.tv/* // @match https://www.zhihu.com/zvideo/* // @match *://www.susudm8.com/* // @match *://susudyy.com/* // @match *://buding3.com/* // @match *://buding6.com/* // @match *://v2.shenjw.com:*/wap.php?url=* // @match *://u88.xigua88ok.com:*/wap.php?url=* // @match *://test3.gqyy8.com:*/f/aliplayer.php?url=* // @match *://v.mksec.cn/* // @include *://*dsh*.com/* // @match https://www.twitch.tv/* // @connect api.bilibili.com // @connect api.live.bilibili.com // @require https://unpkg.com/jquery@3.2.1/dist/jquery.min.js // @require https://unpkg.com/md5@2.3.0/dist/md5.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", mpv: { path: "", regVersion: "20230514", }, potplayer: { path: "", regVersion: "20230514", }, proxy: "", bestQuality: "2160p", bilibiliCodecs: 12, playAuto: 0, closeAuto: 0, syncStartTime: 0, subtitlePrefer: "zh-Hans", customPlayer: { name: "custom", params: { videoUrl: 'iina://weblink?url=${EvideoUrl}', audioUrl: '', subtitleUrl: '', title: '', startTime: '', referer: '', origin: '', proxy: '', other: '' } }, version: "20230702" }; 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: 3500, refresh: 600, reportInterval: 600, pauseInterval: 2000, showButton: 5000, } // 尝试次数 var tryTime = 0; const TRY_TIME = { maxPause: 5, maxParse: 8 }; // 播放器配置 const PLAYER = { mpv: { name: "mpv", params: { videoUrl: 'mpv://"${videoUrl}"', audioUrl: ' --audio-file="${audioUrl}"', subtitleUrl: ' --sub-file="${subtitleUrl}"', 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', subtitleUrl: ' /sub="${subtitleUrl}"', title: ' /title="${title}"', startTime: ' /seek=${startTime}', referer: ' /referer="${referer}"', origin: ' /headers="origin: ${origin}"', proxy: ' /user_agent="${proxy}"' } }, custom: { name: "custom", params: undefined } } // 页面信息 var page = { 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`, infoButton: `${PREFIX}-info-button`, infoDiv: `${PREFIX}-info-div`, infoTable: `${PREFIX}-info-table`, playButton: `${PREFIX}-play-button`, settingButton: `${PREFIX}-setting-button`, settingDiv: `${PREFIX}-setting-div`, settingTable: `${PREFIX}-setting-table`, playerRadio: `${PREFIX}-player-radio`, softwarePathInput: `${PREFIX}-software-path-input`, proxyInput: `${PREFIX}-proxy-input`, bestQualityRadio: `${PREFIX}-best-quality-radio`, bilibiliCodecsRadio: `${PREFIX}-bilibili-codecs-radio`, saveButton: `${PREFIX}-save-button`, downloadButton: `${PREFIX}-download-button`, deleteButton: `${PREFIX}-delete-button`, playAutoInput: `${PREFIX}-play-auto-input`, closeAutoInput: `${PREFIX}-close-auto-input`, syncStartTimeInput: `${PREFIX}-sync-start-time-input`, syncStartTimeSpan: `${PREFIX}-sync-start-time-span`, infoDiv: `${PREFIX}-info-div`, infoTable: `${PREFIX}-info-table`, subtitlePreferRadio: `${PREFIX}-subtitle-prefer-radio`, customPlayerButton: `${PREFIX}-custom-player-button`, customPlayerTable: `${PREFIX}-custom-player-table`, videoUrlParamInput: `${PREFIX}-video-url-param-input`, audioUrlParamInput: `${PREFIX}-audio-url-param-input`, subtitleUrlParamInput: `${PREFIX}-subtitle-url-param-input`, titleParamInput: `${PREFIX}-title-param-input`, startTimeParamInput: `${PREFIX}-start-time-param-input`, proxyParamInput: `${PREFIX}-proxy-param-input`, refererParamInput: `${PREFIX}-referer-param-input`, originParamInput: `${PREFIX}-origin-param-input`, nxParserIframe: `${PREFIX}-nx-parser-iframe` } // 组件 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}-switch-label-class`, sliderSpan: `${PREFIX}-slider-span-class`, roundSpan: `${PREFIX}-round-span-class`, readOnly: `${PREFIX}-read-only-class`, footerA: `${PREFIX}-footer-a-class`, infoInput: `${PREFIX}-info-input-class`, } // 消息类型 const TOAST_TYPE = { info: "info", warn: "warn", error: "error" } // 图标 const ICON_BASE64 = { custom: "url('')", back: "url('')", } 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); } #${ID.loadingDiv} .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; position: fixed; bottom: 0; left: 0; cursor: pointer; z-index: 99999; width: 90px; height: 90px; } #${ID.buttonDiv}:hover .${CLASS.button} { visibility: visible !important; } .${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}, #${ID.infoDiv} { position: fixed; top: 40%; left: 50%; transform: translate(-50%, -50%); z-index: 999999999; width: 900px; height: 520px; background-color: rgb(65, 146, 247); display: none; flex-direction: column; border-radius: 6px; align-items: center; color: rgba(0, 0, 0, .7); font-family: "微软雅黑"; } #${ID.infoDiv} { background-color: rgb(234, 122, 153) !important; } #${ID.infoTable}, #${ID.settingTable}, #${ID.customPlayerTable} { width: 800px; height: 460px; border-radius: 5px !important; border: 3px solid rgba(255, 255, 255, 1) !important; text-align: left; border-collapse: unset !important; display: flex; justify-content: center; padding-top: 3px; padding-bottom: 12px; line-height: 20px !important; font-weight: normal !important; } #${ID.customPlayerTable} { display: none; } #${ID.infoTable} td, #${ID.settingTable} td, #${ID.customPlayerTable} td { font-size: 14px; border: 0px solid rgba(255, 255, 255, 0.5); padding: 19px 0px 0px 0px !important; } #${ID.infoTable} td { padding-top: 16.5px !important; } .${CLASS.titleSpan} { padding-top: 12.5px !important;; padding-bottom: 12.5px !important;; font-size: 16px; font-weight: bold; color: rgba(255, 255, 255, 1) !important; } .${CLASS.closeButton} { position: absolute; top: 4px; right: 8px; height: 25px; width: 25px; border: none; font-size: 17px !important; font-weight: normal !important; background-color: rgba(0, 0, 0, 0); line-height: 0px; border-radius: 3px; transform: scale(1.32, 1); color: rgba(255, 255, 255, 1); } .${CLASS.closeButton}:hover { font-size: 20px; background-color: rgba(255, 255, 255, .5); cursor: pointer; } .${CLASS.tipSpan} { font-size: 12px; color: yellow; position: fixed; } .${CLASS.titleTd} { position: relative; width: 80px; height: 30px; border: none; font-size: 14px; text-align: center; color: rgba(255, 255, 255, 1) !important; cursor: default; } #${ID.infoTable} input, #${ID.settingTable} input, #${ID.customPlayerTable} input { font-size: 12px !important; width: 500px; height: 26px; border: none; outline: none; text-indent: 5px; padding: 0px !important; border-radius: 0px !important; color: rgba(0, 0, 0, 1); background-color: rgba(255, 255, 255, 1); cursor: auto; display: flex !important; margin-top: 1px !important; margin-bottom: 1px !important; border-collapse: unset !important; } #${ID.infoTable} input:hover, #${ID.settingTable} input:hover, #${ID.customPlayerTable} input:hover, #${ID.infoTable} input:focus-visible, #${ID.settingTable} input:focus-visible, #${ID.customPlayerTable} input:focus-visible { box-shadow: none; } #${ID.settingTable} input::placeholder, #${ID.customPlayerTable} input::placeholder { font-size: 12px; color: rgba(0, 0, 0, .3); } #${ID.saveButton} { font-size: 14px; margin-left: 83px; width: 300px; height: 30px; border: none; border-radius: 3px; color: rgba(255, 255, 255, 1); background-color: rgba(0, 255, 0, .7); } #${ID.downloadButton} { font-size: x-small; margin-left: 10px; width: 100px; height: 30px; border: none; border-radius: 3px; color: rgba(255, 255, 255, 1); background-color: rgba(0, 255, 0, .7); } #${ID.deleteButton} { text-decoration: none; font-size: x-small; width: 80px; height: 30px; border: none; border-radius: 3px; color: rgba(255, 255, 255, 1); background-color: rgba(0,0,0,0); } #${ID.saveButton}:hover, #${ID.downloadButton}:hover { opacity: .8; background-color: rgba(0, 255, 0, .8); cursor: pointer; } #${ID.deleteButton}:hover { opacity: .8; background-color: rgba(0,0,0,0); cursor: pointer; } .${CLASS.footerSpan} { margin-top: 8px !important; margin-bottom: 8px !important; color: rgba(255, 255, 255, 1); } #${ID.infoDiv} a, .${CLASS.footerSpan} a { color: rgba(255, 255, 255, 1); text-decoration: none; font-size: 14px !important; font-weight: normal !important; margin-bottom: 1px; display: inline-block; } /* switch */ .${CLASS.switchLabel} { position: relative; display: inline-block; width: 50px; height: 21px; margin-top: 3px; } .${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; } #${ID.settingDiv} input:checked + .${CLASS.sliderSpan} { background-color: rgba(0, 255, 0, .7); } #${ID.settingDiv} input:focus + .${CLASS.sliderSpan} { box-shadow: 0 0 1px rgba(0, 255, 0, .7); } #${ID.settingDiv} 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, 255, 255, .3) !important; background-color: rgba(0, 0, 0, .3) !important; cursor: default !important; opacity: 1 !important; } .${CLASS.readOnly}::placeholder { color: rgba(255, 255, 255, .3) !important; } #${ID.infoButton} { bottom: 52px; left: 3px; width: 25px; height: 25px; background-image: url(); } #${ID.infoButton}:hover { bottom: 50px; left: 1px; width: 29px; height: 29px; } #${ID.settingDiv} .tabs { display: flex; width: 501px; height: 28px; } #${ID.settingDiv} .tabs>.tab { flex: 1; display: flex; border: 1px solid rgb(65,146,247); margin: 0; } #${ID.settingDiv} .tabs>.tab:after { background: rgba(0, 0, 0, 0); color: rgba(0, 0, 0, 0) !important; } #${ID.settingDiv} .tab>.tab-input { width: 0 !important; height: 0 !important; margin: 0 !important; display: none !important; } #${ID.settingDiv} .tab>.tab-box { padding: 4px 0px 0.5px 0px; width: 100%; height: 22px; text-align: center; transition: 0.3s; background: rgba(255, 255, 255, 1); font-size: 12px; font-weight: normal !important; display: table !important; color: #000000B3; } #${ID.settingDiv} .tab>.tab-box:hover { opacity: .8; cursor: pointer; } #${ID.settingDiv} .tab>.tab-input:checked+.tab-box { color: rgba(255, 255, 255, 1); background: rgba(0, 255, 0, .7); } #${ID.customPlayerButton}:hover:after, #${ID.settingDiv} .${CLASS.titleTd}:hover:after, .${CLASS.footerA}:hover:after { position: absolute; font-size: 12px; left: 0px; top: -3px; padding: 5px 5px 5px 5px !important; background-color: rgba(0, 0, 0, 0.6); color: rgba(255, 255, 255, 1); content: attr(data-tip); text-align: center; z-index: 999999; width: auto !important; height: auto !important; white-space: nowrap; } #${ID.customPlayerButton}:hover:after { left: -10px !important; top: -28px !important; } .${CLASS.footerA} { position: relative; } .${CLASS.footerA}:hover:after { left: 0px !important; top: -27px !important; } #${ID.customPlayerButton} { position: absolute; right: 122px; top: 76px; width: 22px; height: 22px; border: none; cursor: pointer; background-image: ${ICON_BASE64.custom}; z-index: 999999; } `; const HTML = `
Play-With-MPV
本 页 视 频 信 息
视频标题
视频链接
音频链接
字幕链接
referer
origin
🆕 升级 🆕 🧭 网站导航 🧭 🌟 项目源码 🌟 👻 反馈 👻
🆕 © 2023 LuckyPuppy514 👻
Play-With-MPV
播放软件
软件路径
代理设置
最高画质
视频编码
首选字幕
自动播放
自动关闭
同步时间
视频参数
音频参数
字幕参数
标题参数
时间参数
代理参数
referer
origin
🆕 © 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\\\${PLAYER_NAME}] @="\${PLAYER_NAME} Protocol" "URL Protocol"="" [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\DefaultIcon] @="" [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\shell] @="" [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\shell\\open] @="" [HKEY_CLASSES_ROOT\\\${PLAYER_NAME}\\shell\\open\\command] @="C:\\\\Windows\\\\System32\\\\WindowsPowerShell\\\\v1.0\\\\powershell.exe -WindowStyle Hidden -Command \\"& {Add-Type -AssemblyName System.Web;$PARAMS=([System.Web.HTTPUtility]::UrlDecode('%1') -replace '^\${PLAYER_NAME}://'); Start-Process -FilePath \\\\\\\"\${SOFTWARE_PATH}\\\\\\\" -ArgumentList $PARAMS}\\"" ` const REG_DELETE = `Windows Registry Editor Version 5.00 [-HKEY_CLASSES_ROOT\\\${PLAYER_NAME}] ` 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 settingTable = document.getElementById(ID.settingTable); let softwarePathInput = document.getElementById(ID.softwarePathInput); let proxyInput = document.getElementById(ID.proxyInput); let playAutoInput = document.getElementById(ID.playAutoInput); let closeAutoInput = document.getElementById(ID.closeAutoInput); let syncStartTimeInput = document.getElementById(ID.syncStartTimeInput); let syncStartTimeSpan = document.getElementById(ID.syncStartTimeSpan); let downloadButton = document.getElementById(ID.downloadButton); let deleteButton = document.getElementById(ID.deleteButton); let saveButton = document.getElementById(ID.saveButton); let closeButtons = document.getElementsByClassName(CLASS.closeButton); let infoButton = document.getElementById(ID.infoButton); let infoDiv = document.getElementById(ID.infoDiv); let customPlayerTable = document.getElementById(ID.customPlayerTable); let customPlayerButton = document.getElementById(ID.customPlayerButton); let videoUrlParamInput = document.getElementById(ID.videoUrlParamInput); let audioUrlParamInput = document.getElementById(ID.audioUrlParamInput); let subtitleUrlParamInput = document.getElementById(ID.subtitleUrlParamInput); let titleParamInput = document.getElementById(ID.titleParamInput); let startTimeParamInput = document.getElementById(ID.startTimeParamInput); let proxyParamInput = document.getElementById(ID.proxyParamInput); let refererParamInput = document.getElementById(ID.refererParamInput); let originParamInput = document.getElementById(ID.originParamInput); let infoInputs = document.getElementsByClassName(CLASS.infoInput); switchStatus(downloadButton, false); // 播放按钮 playButton.onclick = function () { if (currentConfig.player == PLAYER.mpv.name || currentConfig.player == PLAYER.potplayer.name) { let message = undefined; if (!currentConfig[currentConfig.player].path) { message = "请先进行设置"; } else if (!currentConfig[currentConfig.player].regVersion) { message = "请先下载注册表"; } else if (currentConfig[currentConfig.player].regVersion != DEFAULT_CONFIG[currentConfig.player].regVersion) { message = "注册表有更新,请重新下载注册表"; } if (message) { toast(message, TOAST_TYPE.warn); settingDiv.style.display = "none"; settingButton.click(); return; } } loading(true); try { playButtonClickLimit(); handler.play(); if (currentConfig.closeAuto == 1 && page.url !== "https://www.lckp.top/play-with-mpv/index.html") { setTimeout(() => { if (history.length === 1) { window.location.href = "about:blank"; window.top.close(); } else { history.back(); } }, 1000); } else { handler.pause(); } } catch (error) { toast("出错辣 ...... 😓", TOAST_TYPE.error); console.log(error); } loading(false); } // 播放快捷键 CTRL + P window.onkeydown = async function () { if (event.ctrlKey && event.keyCode === 80 && !event.shiftKey) { event.preventDefault(); playButton.click(); } } // 设置按钮 settingButton.onclick = function () { let display = settingDiv.style.display; if (display == "flex") { settingDiv.style.display = "none"; } else { infoDiv.style.display = "none"; settingDiv.style.display = "flex"; // 加载配置 softwarePathInput.value = currentConfig[currentConfig.player].path; proxyInput.value = currentConfig.proxy; $(`input:radio[name="${ID.bestQualityRadio}"][value="${currentConfig.bestQuality}"]`).prop('checked', true); $(`input:radio[name="${ID.bilibiliCodecsRadio}"][value="${currentConfig.bilibiliCodecs}"]`).prop('checked', true); $(`input:radio[name="${ID.playerRadio}"][value="${currentConfig.player}"]`).prop('checked', true); playAutoInput.checked = currentConfig.playAuto == 1 ? true : false; closeAutoInput.checked = currentConfig.closeAuto == 1 ? true : false; syncStartTimeInput.checked = currentConfig.syncStartTime == 1 ? true : false; $(`input:radio[name="${ID.subtitlePreferRadio}"][value="${currentConfig.subtitlePrefer}"]`).prop('checked', true); switchPlayer($(`input:radio[name="${ID.playerRadio}"]:checked`).val()); } } // 播放器选择框 $(`input:radio[name="${ID.playerRadio}"]`).change(function () { switchPlayer(this.value); }); // 保存按钮 saveButton.onclick = function () { let playerChecked = $(`input:radio[name="${ID.playerRadio}"]:checked`).val(); if (playerChecked == PLAYER.mpv.name || playerChecked == PLAYER.potplayer.name) { let oldSoftwarePath = currentConfig[playerChecked].path; let newSoftwarePath = softwarePathInput.value; if (!newSoftwarePath) { toast("软件路径不能为空", TOAST_TYPE.error); return; } if (/.*[\u4e00-\u9fa5]+.*/g.test(newSoftwarePath)) { toast("软件路径不能包含中文", TOAST_TYPE.error); return; } newSoftwarePath = newSoftwarePath.replace(/[\\|/]+/g, "//"); if (!newSoftwarePath.endsWith(".com") && !newSoftwarePath.endsWith(".exe")) { if (!newSoftwarePath.endsWith("//")) { newSoftwarePath = newSoftwarePath + "//"; } if (playerChecked == PLAYER.mpv.name) { if (newSoftwarePath.toLowerCase().indexOf("mpvnet") != -1 || newSoftwarePath.toLowerCase().indexOf("mpv.net") != -1) { newSoftwarePath = newSoftwarePath + "mpvnet.exe"; } else { newSoftwarePath = newSoftwarePath + "mpv.exe"; } } else if (playerChecked == PLAYER.potplayer.name) { newSoftwarePath = newSoftwarePath + "PotPlayerMini64.exe"; } } softwarePathInput.value = newSoftwarePath; currentConfig[playerChecked].path = newSoftwarePath; switchStatus(downloadButton, softwarePathInput.value ? true : false); if (oldSoftwarePath != newSoftwarePath) { currentConfig[playerChecked].regVersion = "00000000"; } } currentConfig.proxy = proxyInput.value; currentConfig.bestQuality = $(`input:radio[name="${ID.bestQualityRadio}"]:checked`).val(); currentConfig.bilibiliCodecs = $(`input:radio[name="${ID.bilibiliCodecsRadio}"]:checked`).val(); currentConfig.player = playerChecked; currentConfig.subtitlePrefer = $(`input:radio[name="${ID.subtitlePreferRadio}"]:checked`).val(); currentConfig.playAuto = playAutoInput.checked ? 1 : 0; currentConfig.closeAuto = closeAutoInput.checked ? 1 : 0; currentConfig.syncStartTime = syncStartTimeInput.checked ? 1 : 0; GM_setValue(KEY.config, currentConfig); if (playAutoInput.checked && closeAutoInput.checked) { toast("保存成功,如需修改配置请前往导航页"); } else { toast("保存成功"); } if (currentConfig.playAuto == 1) { playButtonClickLimit(); } if (document.getElementById("iptv") && localStorage.category == "iptv") { localStorage.player = JSON.stringify(PLAYER[currentConfig.player]); document.getElementById("iptv").click(); } init(); } // 下载按钮 downloadButton.onclick = function () { let playerChecked = $(`input:radio[name="${ID.playerRadio}"]:checked`).val(); currentConfig[playerChecked].regVersion = DEFAULT_CONFIG[playerChecked].regVersion; GM_setValue(KEY.config, currentConfig); let reg = REG.replace("${SOFTWARE_PATH}", currentConfig[playerChecked].path); reg = reg.replace(/\$\{PLAYER_NAME\}/g, playerChecked); let a = document.createElement('a'); let blob = new Blob([reg], { 'type': 'application/octet-stream' }); a.href = window.URL.createObjectURL(blob); a.download = `${playerChecked}-install.reg`; a.click(); } deleteButton.onclick = function () { let playerChecked = $(`input:radio[name="${ID.playerRadio}"]:checked`).val(); currentConfig[playerChecked].regVersion = "00000000"; GM_setValue(KEY.config, currentConfig); let reg = REG_DELETE.replace(/\$\{PLAYER_NAME\}/g, playerChecked); let a = document.createElement('a'); let blob = new Blob([reg], { 'type': 'application/octet-stream' }); a.href = window.URL.createObjectURL(blob); a.download = `${playerChecked}-delete.reg`; a.click(); } // 关闭按钮 for (let closeButton of closeButtons) { closeButton.onclick = function () { settingDiv.style.display = "none"; infoDiv.style.display = "none"; } } // 信息按钮 infoButton.onclick = function () { let display = infoDiv.style.display; if (display == "flex") { infoDiv.style.display = "none"; } else { settingDiv.style.display = "none"; infoDiv.style.display = "flex"; let title = handler.media.title; infoInputs[0].value = title ? title : document.title; infoInputs[1].value = handler.media.videoUrl; infoInputs[2].value = handler.media.audioUrl; infoInputs[3].value = handler.media.subtitleUrl; infoInputs[4].value = handler.media.referer; infoInputs[5].value = handler.media.origin; for (const infoInput of infoInputs) { if (infoInput.value) { infoInput.style.cursor = "pointer"; infoInput.onclick = function () { this.select(); navigator.clipboard.writeText(this.value); toast("已复制到剪贴板"); } } } } } let bestQualityRadios = document.getElementsByName(ID.bestQualityRadio); let bilibiliCodecsRadios = document.getElementsByName(ID.bilibiliCodecsRadio); let subtitlePreferRadios = document.getElementsByName(ID.subtitlePreferRadio); // 切换播放器 function switchPlayer(player) { player = PLAYER[player]; // mpv 和 potplayer 专属 if (player.name == PLAYER.mpv.name || player.name == PLAYER.potplayer.name) { switchStatus(softwarePathInput, true); softwarePathInput.value = currentConfig[player.name].path; if (softwarePathInput.value) { switchStatus(downloadButton, true); } else { switchStatus(downloadButton, false); } } else { switchStatus(softwarePathInput, false); switchStatus(downloadButton, false); } // 代理 let flag = player.params.proxy ? true : false; switchStatus(proxyInput, flag); // 音频 flag = player.params.audioUrl ? true : false; for (const radio of bestQualityRadios) { switchStatus(radio, flag); } for (const radio of bilibiliCodecsRadios) { switchStatus(radio, flag); } // 字幕 flag = player.params.subtitleUrl ? true : false; for (const radio of subtitlePreferRadios) { switchStatus(radio, flag); } // 时间 flag = player.params.startTime ? true : false; switchStatus(syncStartTimeSpan, flag); switchStatus(syncStartTimeInput, flag); } // 全屏 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); } // 自定义播放器按钮 customPlayerButton.onclick = function () { if (customPlayerTable.style.display == "flex") { if (!videoUrlParamInput.value) { toast("视频参数不能为空", TOAST_TYPE.error); return; } currentConfig.customPlayer.params.videoUrl = videoUrlParamInput.value; currentConfig.customPlayer.params.audioUrl = audioUrlParamInput.value; currentConfig.customPlayer.params.subtitleUrl = subtitleUrlParamInput.value; currentConfig.customPlayer.params.title = titleParamInput.value; currentConfig.customPlayer.params.startTime = startTimeParamInput.value; currentConfig.customPlayer.params.proxy = proxyParamInput.value; currentConfig.customPlayer.params.referer = refererParamInput.value; currentConfig.customPlayer.params.origin = originParamInput.value; PLAYER.custom = currentConfig.customPlayer; GM_setValue(KEY.config, currentConfig); switchPlayer($(`input:radio[name="${ID.playerRadio}"]:checked`).val()); settingTable.style.display = "flex"; customPlayerTable.style.display = "none"; customPlayerButton.style.backgroundImage = ICON_BASE64.custom; customPlayerButton.dataset.tip = "设置自定义播放器"; } else { videoUrlParamInput.value = currentConfig.customPlayer.params.videoUrl; audioUrlParamInput.value = currentConfig.customPlayer.params.audioUrl; subtitleUrlParamInput.value = currentConfig.customPlayer.params.subtitleUrl; titleParamInput.value = currentConfig.customPlayer.params.title; startTimeParamInput.value = currentConfig.customPlayer.params.startTime; proxyParamInput.value = currentConfig.customPlayer.params.proxy; refererParamInput.value = currentConfig.customPlayer.params.referer; originParamInput.value = currentConfig.customPlayer.params.origin; settingTable.style.display = "none"; customPlayerTable.style.display = "flex"; customPlayerButton.style.backgroundImage = ICON_BASE64.back; customPlayerButton.dataset.tip = "保存并返回"; } } } // 切换元素状态 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); currentConfig = JSON.parse(JSON.stringify(DEFAULT_CONFIG)); currentConfig.mpv.regVersion = ""; currentConfig.potplayer.regVersion = ""; if (!oldConifg) { GM_setValue(KEY.config, currentConfig); } else { if (oldConifg.version != DEFAULT_CONFIG.version) { for (const key in oldConifg) { currentConfig[key] = oldConifg[key]; } if (!currentConfig.mpv.path && currentConfig.mpvPath) { currentConfig.mpv.path = currentConfig.mpvPath; delete currentConfig['mpvPath']; } currentConfig.version = DEFAULT_CONFIG.version; GM_setValue(KEY.config, currentConfig); } else { currentConfig = oldConifg; GM_setValue(KEY.config, currentConfig); } } PLAYER.custom = currentConfig.customPlayer; } class Media { constructor() { this.title = ""; this.videoUrl = ""; this.audioUrl = ""; this.subtitleUrl = ""; 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 (!this.title) { this.setTitle(document.title); } let nxParserIframe = document.getElementById(ID.nxParserIframe); if (nxParserIframe) { document.body.removeChild(nxParserIframe); } if (document.getElementById(ID.buttonDiv)) { document.getElementById(ID.buttonDiv).style.display = "flex"; if (currentConfig.playAuto == 1) { document.getElementById(ID.playButton).click(); } document.getElementById(ID.infoButton).style.visibility = "visible"; document.getElementById(ID.settingButton).style.visibility = "visible"; document.getElementById(ID.playButton).style.visibility = "visible"; setTimeout(() => { document.getElementById(ID.infoButton).style.visibility = "hidden"; document.getElementById(ID.settingButton).style.visibility = "hidden"; document.getElementById(ID.playButton).style.visibility = "hidden"; }, TIME.showButton); } } } setAudioUrl(audioUrl) { this.audioUrl = audioUrl; } setSubtitleUrl(subtitleUrl) { this.subtitleUrl = subtitleUrl; } 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) && videoUrl != localStorage.iptvUrl) { let m3u8 = ""; $.ajax({ type: "GET", url: videoUrl, async: false, success: function (res) { m3u8 = res; } }); if (m3u8 && m3u8.indexOf("png") != -1) { console.log("Play-With-MPV:m3u8 链接无法播放:" + videoUrl); return false; } } if (videoUrl.startsWith("https://www.mp4")) { return false; } return true; } console.log(`Play-With-MPV:链接无效:${videoUrl}`); return false; } } class BaseHandler { constructor() { loadConfig(); this.media = new Media(); this.media.setProxy(currentConfig.proxy); 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"; } } initCheck() { return window.location.href != page.url; } 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 { let videos = document.getElementsByTagName("video"); for (let i = 0; i < videos.length; i++) { if (videos[i].currentTime != 0) { this.media.setStartTime(videos[i].currentTime); break; } } } } let value = this.media[key]; if (value) { let param = this.player.params[key]; let matchKey = '${' + key + '}'; while (param.indexOf(matchKey) != -1) { param = param.replace(matchKey, value); } matchKey = '${E' + key + '}'; while (param.indexOf(matchKey) != -1) { param = param.replace(matchKey, encodeURIComponent(value)); } matchKey = '${D' + key + '}'; while (param.indexOf(matchKey) != -1) { param = param.replace(matchKey, decodeURIComponent(value)); } link = link + param; } } if (this.media.title) { let maxLength = 1950 - link.length; let title = encodeURIComponent(this.media.title); if (title.length > maxLength) { title = title.substring(0, maxLength) + '...'; } let param = this.player.params.title; param = param.replace('${title}', title); link = link + param; } 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]; } } // 诺讯解析 nxParser() { handler.addIframeListener(); let nxParserIframe = document.getElementById(ID.nxParserIframe); if (!nxParserIframe) { nxParserIframe = document.createElement("iframe"); nxParserIframe.id = ID.nxParserIframe; nxParserIframe.src = `https://yun.nxflv.com/?url=${page.url}`; document.body.appendChild(nxParserIframe); } } } // 获取B站视频播放链接 async function getBilibiliPlayUrl(avid, cid) { if (handler.player.name == PLAYER.mpv.name) { handler.media.setOther(`--script-opts="cid=${cid}"`); } if (!handler.player.params.audioUrl || await getBilibiliVideoDash(avid, cid) == -1) { getBilibiliVideoDurl(avid, cid); } if (currentConfig.subtitlePrefer != "off") { getBilibiliVideoSubtitle(avid, cid); } } // 获取B站 DASH 格式视频 async function getBilibiliVideoDash(avid, cid) { let result = 0; await $.ajax({ type: "GET", url: `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=4048&avid=${avid}&cid=${cid}`, xhrFields: { withCredentials: true }, async: false, success: function (res) { if (!res.data) { toast("Play-With-MPV 获取视频失败,如未登录请先登录并刷新页面", TOAST_TYPE.error); tryTime = TRY_TIME.maxParse; return; } let videoUrl = undefined; let audioUrl = undefined; let dash = res.data.dash; if (!dash) { result = -1; return; } 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 if (dash.audio) { 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); result = 1; } }); return result; } // 获取B站 FLV / MP4 格式视频 function getBilibiliVideoDurl(avid, cid) { $.ajax({ type: "GET", url: `https://api.bilibili.com/x/player/playurl?qn=120&otype=json&fourk=1&fnver=0&fnval=128&avid=${avid}&cid=${cid}`, xhrFields: { withCredentials: true }, async: false, success: function (res) { if (!res.data) { toast("Play-With-MPV 获取视频失败,如未登录请先登录并刷新页面", TOAST_TYPE.error); tryTime = TRY_TIME.maxParse; return; } handler.media.setVideoUrl(res.data.durl[0].url); } }); } // 获取B站视频字幕 function getBilibiliVideoSubtitle(avid, cid) { $.ajax({ type: "GET", url: `https://api.bilibili.com/x/player/v2?aid=${avid}&cid=${cid}`, xhrFields: { withCredentials: true }, async: false, success: function (res) { if (res.code == 0 && res.data.subtitle && res.data.subtitle.subtitles.length > 0) { let subtitles = res.data.subtitle.subtitles; let url = "https:" + subtitles[0].subtitle_url; let lan = subtitles[0].lan; for (const subtitle of subtitles) { if (currentConfig.subtitlePrefer.startsWith("zh") && subtitle.lan.startsWith("zh")) { url = "https:" + subtitle.subtitle_url; lan = subtitle.lan; } if (subtitle.lan == currentConfig.subtitlePrefer) { url = "https:" + subtitle.subtitle_url; lan = subtitle.lan; break; } } handler.media.setSubtitleUrl(`https://www.lckp.top/common/bilibili/jsonToSrt/?url=${url}&lan=${lan}`); } } }); } // 获取B站视频 aid 和 cid function getBilibiliVideoId() { let hasInitialState = false; try { if (__INITIAL_STATE__) { hasInitialState = true; } } catch (error) { hasInitialState = false; } if (!hasInitialState) { return undefined; } let video = undefined; if (__INITIAL_STATE__.epInfo) { video = __INITIAL_STATE__.epInfo; } else if (__INITIAL_STATE__.videoData) { video = __INITIAL_STATE__.videoData; } else if (__INITIAL_STATE__.videoInfo) { video = __INITIAL_STATE__.videoInfo; } let aid = video.aid; let cid = video.cid; let p = __INITIAL_STATE__.p; if (p && p > 1) { cid = __INITIAL_STATE__.cidMap[aid].cids[p]; } let videoId = { aid: aid, cid: cid }; return videoId; } 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 }, ytdlp: { "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 // ✅ https://www.bilibili.com/bangumi/play/ep319063 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() { // 直接从数据中获取 aid 和 cid let videoId = getBilibiliVideoId(); if (videoId && videoId.aid && videoId.cid) { getBilibiliPlayUrl(videoId.aid, videoId.cid); return; } // 从元素提取 epid 请求接口获取 aid 和 cid let epid = page.url.match(/ep(\d+)/); if (epid && epid[1]) { epid = epid[1]; } else { 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 // ✅ https://www.bilibili.com/video/BV17Z4y117Qm // ✅ https://www.bilibili.com/list/ml1806211634?oid=822115390&bvid=BV1Fg4y1p7Qe name: "B站投稿", regex: /^https:\/\/www\.bilibili\.com\/(video\/|list.*)(BV|av).*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.media.setReferer("https://www.bilibili.com"); } initCheck() { if (super.initCheck()) { let newPageUrl = window.location.href; let oldPageUrl = page.url; let regex = /(&|\?)vd_source=\w+/; if (regex.test(newPageUrl.replace(oldPageUrl, ""))) { page.url = newPageUrl; return false; } return true; } return false; } async parse() { // 直接从数据中获取 aid 和 cid let videoId = getBilibiliVideoId(); if (videoId && videoId.aid && videoId.cid) { getBilibiliPlayUrl(videoId.aid, videoId.cid); return; } // 通过 bvid/avid 请求接口获取 aid 和 cid let param = undefined; let bvid = page.url.match(/BV([0-9a-zA-Z]+)/); if (bvid && bvid[1]) { param = `bvid=${bvid[1]}`; } else { let avid = page.url.match(/av([0-9]+)/); if (avid && avid[1]) { param = `aid=${avid[1]}`; } } if (!param) { return; } $.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://www.bilibili.com/festival/2023bnj?bvid=BV17G4y1X7vQ name: "B站节日", regex: /^https:\/\/www\.bilibili\.com\/festival\/.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.media.setReferer("https://www.bilibili.com"); } initCheck() { if (super.initCheck()) { return true; } let oldvideoId = this.videoId; let newvideoId = getBilibiliVideoId(); if (oldvideoId && oldvideoId.cid != newvideoId.cid) { return true; } return false; } async parse() { let videoId = getBilibiliVideoId(); if (videoId && videoId.aid && videoId.cid) { this.videoId = videoId; getBilibiliPlayUrl(videoId.aid, videoId.cid); return; } else { toast("Play-With-MPV 读取视频数据失败,请尝试清理B站缓存后刷新重试", TOAST_TYPE.error, 5000); tryTime = TRY_TIME.maxParse; } } }, }, { // ✅ https://live.bilibili.com/7777 name: "B站直播", home: [ "https://live.bilibili.com" ], regex: /^https:\/\/live\.bilibili\.com\/\d+.*/g, handler: class Handler extends BaseHandler { async parse() { let iframes = document.getElementsByTagName("iframe"); let roomid = undefined; for (let iframe of iframes) { let roomids = iframe.src.match(/^https:\/\/live\.bilibili\.com.*(roomid=\d+|blanc\/\d+).*/); if (roomids && roomids[1]) { roomid = roomids[1].match(/\d+/)[0]; break; } } if (!roomid) { console.log("Play-With-MPV:找不到 roomid:" + roomid); return; } 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://www.ixigua.com/ name: "西瓜视频", home: [ "https://www.ixigua.com" ], regex: /^https:\/\/www\.ixigua\.com\/\d.*/g, handler: class Handler extends BaseHandler { constructor() { super(); this.media.setReferer("https://www.ixigua.com/"); } async parse() { let that = this; $.ajax({ type: "GET", url: page.url, async: false, success: function (res) { try { let _SSR_HYDRATED_DATA = (new Function("return " + res.match(/