// ==UserScript==
// @name                    Play-With-MPV
// @name:zh                 使用 MPV 播放
// @description             使用 MPV 播放网页上的视频
// @namespace               https://github.com/LuckyPuppy514
// @version                 2.1.0
// @commit                  v1.2.1 新增 powershell 脚本升级提醒功能
// @commit                  v1.2.2 修复 youtube 标题带 | 导致错误脚本升级提醒
// @commit                  v1.2.3 修改 imomoe 域名
// @commit                  v1.3.0 新增域名:www.6dm.cc, www.dmla.cc(第一线路:大部分支持,其他线路:小部分支持)
// @commit                  v1.3.0 新增域名:www.dm233.me(线路III:大部分支持,其他线路:大部分不支持)
// @commit                  v1.3.0 代码重构,使用继承方便后续添加网站支持
// @commit                  v1.4.0 b站bug修复:标题带数字,解析出错,修复并优化了获取视频链接的速度
// @commit                  v1.4.0 新增对plex支持(本地:*://*/web/index.html*,远程:https://app.plex.tv/desktop/*)
// @commit                  v1.4.1 修复b站番剧播放目录为列表时,无法获取正确集数的bug
// @commit                  v1.4.2 修复b站番剧播放的bug
// @commit                  v1.4.3 修改cdn为unpkg,某些网络无法访问cdn,导致js加载失败(有问题,请自行修改:unpkg.com => cdn.jsdelivr.net/npm)
// @commit                  v1.4.4 www.dmla.cc 域名变更为:www.dmlaa.com
// @commit                  v1.4.5 ddrk.me 域名变更为:ddys.tv
// @commit                  v1.5.0 代码优化,去除 powershell 脚本,只需添加注册表信息即可
// @commit                  v1.5.1 B站添加 cid 参数,配合 https://github.com/itKelis/MPV-Play-BiliBili-Comments 可实现弹幕功能
// @commit                  v1.5.2 注册表代码升级,支持中文标题
// @commit                  v1.5.3 添加低端影视备用域名
// @commit                  v2.0.0 代码重构:1. 新增对B站av号视频支持;2. B站,油管,低端影视同步网页播放时间;3. 新增MPV路径设置,方便生成注册表;4. 新增Youtube代理设置;5. 减少暂停失败情况
// @commit                  v2.0.1 更新 mpv.net_CM 安装教程链接
// @commit                  v2.0.2 更新 www.6dm.cc 域名为 www.996dm.com
// @commit                  v2.0.3 B站接口变更,画质上限:4K => 8K HDR,音质上限:192K => Dolby Hi-Res
// @commit                  v2.0.4 修复B站 Hi-Res 音频链接抓取错误的问题
// @commit                  v2.0.5 新增巴哈姆特(https://ani.gamer.com.tw)支持
// @commit                  v2.0.6 代码优化;设置代理时,对巴哈姆特也生效
// @commit                  v2.0.8 修复油管全屏图标仍然显示的问题
// @commit                  v2.0.9 界面细节优化
// @commit                  v2.1.0 修复低端影视出现10s广告提醒时,无法抓取链接的问题
// @homepage                https://github.com/LuckyPuppy514/Play-With-MPV
// @author                  LuckyPuppy514
// @copyright               2022, Grant LuckyPuppy514 (https://github.com/LuckyPuppy514)
// @license                 MIT
// @icon                    https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/mpv.png
// @match                   *://www.youtube.com/*
// @include                 https://www.youtube.com/watch/*
// @include                 https://www.bilibili.com/bangumi/play/*
// @include                 https://www.bilibili.com/video/*
// @connect                 api.bilibili.com
// @include                 https://ddys.tv/*
// @include                 https://ddys2.me/*
// @include                 https://www.996dm.com/play/*
// @include                 http://www.996dm.com/play/*
// @include                 http://www.dmlaa.com/play/*
// @include                 https://danmu.yhdmjx.com/*
// @include                 https://www.dm233.me/play/*
// @include                 http://www.dmh8.com/player/*
// @include                 https://www.yhdmp.net/vp/*
// @include                 https://ani.gamer.com.tw/animeVideo.php?*
// @run-at                  document-end
// @require                 https://unpkg.com/js-base64@3.6.1/base64.js
// @require                 https://unpkg.com/jquery@3.2.1/dist/jquery.min.js
// @grant                   GM_setValue
// @grant                   GM_getValue
// @downloadURL none
// ==/UserScript==
'use strict';
// 注册表版本
const REG_VERSION = "20220907";
// 不输出控制台信息
const NO_TERMINAL = true;
// const IS_DEBUG = true;
// function debug(data) {
//     if (IS_DEBUG) {
//         console.log(data);
//     }
// }
const DIV =
    `
    
    
    
`
const CSS =
    `
.pwmpv-close-button {
    position: absolute;
    top: 3px;
    right: 3px;
    height: 25px;
    width: 40px;
    border: none;
    font-size: 18px;
    background-color: rgba(0, 0, 0, 0);
}
.pwmpv-close-button:hover {
    background-color: rgba(0, 0, 0, 0.3);
    cursor: pointer;
}
#pwmpv-button-div {
    display: none;
}
.pwmpv-title-span {
    padding-top: 10px;
    font-size: 15px;
}
#pwmpv-about-button {
    position: fixed;
    bottom: 58px;
    left: 8px;
    z-index: 999998;
    width: 25px;
    height: 25px;
    border: none;
    border-radius: 50%;
    background-size: cover;
    background-color: rgba(255, 255, 255, 0);
    background-image: url(https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/about-pink.png);
}
#pwmpv-about-button:hover {
    bottom: 56px;
    left: 6px;
    z-index: 999999;
    width: 27px;
    height: 27px;
	cursor: pointer;
}
#pwmpv-about-div {
    position: fixed;
    top: 40%;
    left: 50%;
    transform: translate(-50%, -50%);
    z-index: 999999;
    width: 600px;
    height: 300px;
    border: 6px solid rgba(255, 255, 255, 0.5);
    background-color: rgba(234, 122, 153, 0.9);
    display: none;
    flex-direction: column;
	border-radius: 6px;
	align-items: center;
    color: rgba(255, 255, 255, 1);
}
#pwmpv-about-table {
    margin-top: 10px;
    width: 570px;
    height: 240px;
    border-radius: 5px !important;
    border: 3px solid rgba(255, 255, 255, 1) !important;
    text-align: center;
}
#pwmpv-about-table td {
    border: 2px solid rgba(255, 255, 255, 0.5);
}
#pwmpv-about-div a {
    color: rgba(255, 255, 255, 1);
    text-decoration: none;
    font-size: 14px;
}
#pwmpv-play-button {
    position: fixed;
    bottom: 16px;
    left: 20px;
    z-index: 999999;
    width: 50px;
    height: 50px;
    border: none;
    border-radius: 50%;
    background-size: cover;
    background-image: url(https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/mpvnet.png);
    cursor: pointer;
}
#pwmpv-play-button:hover {
    bottom: 14px;
    left: 18px;
    width: 54px;
    height: 54px;
	cursor: pointer;
}
#pwmpv-setting-button {
    position: fixed;
    bottom: 56px;
    left: 58px;
    z-index: 999998;
    width: 28px;
    height: 28px;
    border: none;
    border-radius: 50%;
    background-size: cover;
    background-color: rgba(255, 255, 255, 0);
    background-image: url(https://www.lckp.top/gh/LuckyPuppy514/pic-bed/common/lx-setting.png);    
}
#pwmpv-setting-button:hover {
    bottom: 54px;
    left: 56px;
    z-index: 999999;
    width: 32px;
    height: 32px;
	cursor: pointer;
}
#pwmpv-setting-div {
    position: fixed;
    top: 40%;
    left: 50%;
    transform: translate(-50%, -50%);
    z-index: 999999;
    width: 600px;
    height: 300px;
    border: 6px solid rgba(255, 255, 255, 0.5);
    background-color: rgba(65, 146, 247, 0.9);
    display: none;
    flex-direction: column;
	border-radius: 6px;
	align-items: center;
    color: rgba(255, 255, 255, 1);
}
#pwmpv-setting-table {
    margin-top: 10px;
    width: 570px;
    height: 240px;
    border-radius: 5px !important;
    border: 3px solid rgba(255, 255, 255, 1) !important;
    text-align: center;
    padding: 10px;
}
#pwmpv-setting-table td {
    border: 0px solid rgba(255, 255, 255, 0.5);
    padding-top: 10px;
}
.pwmpv-title-td{
    width: 120px;
    height: 30px;
    border: none;
    font-size: 14px;
}
#pwmpv-setting-table input {
    width: 400px;
    height: 30px;
    border: none;
    outline: none;
    padding-left: 6px;
    border-radius: 3px;
    color: rgba(0, 0, 0, 1);
    background-color: rgba(255, 255, 255, 0.9);
}
#pwmpv-bilibili-codecs-select {
    width: 406px;
    height: 30px;
    border: none;
    outline: none;
    padding-left: 6px;
    border-radius: 3px;
    color: rgba(0, 0, 0, 1);
    background-color: rgba(255, 255, 255, 0.9);
}
#pwmpv-save-button {
    margin-left: 80px;
    width: 300px;
    height: 30px;
    border: none;
    border-radius: 3px;
    color: rgba(255, 255, 255, 1);
    background-color: rgba(0, 255, 50, 0.6);
}
#pwmpv-save-button:hover {
    background-color: rgba(0, 255, 0, 0.8);
    cursor: pointer;
}
.pwmpv-download-enable:hover {
    background-color: rgba(0, 255, 0, 0.8);
    cursor: pointer;
}
.pwmpv-download-disable:hover {
    cursor: pointer;
}
.pwmpv-download-enable {
    margin-left: 10px;
    width: 80px;
    height: 30px;
    border: none;
    border-radius: 3px;
    color: rgba(255, 255, 255, 1);
    background-color: rgba(0, 255, 50, 0.6);
    
}
.pwmpv-download-disable {
    margin-left: 10px;
    width: 80px;
    height: 30px;
    border: none;
    border-radius: 3px;
    color: rgba(255, 255, 255, 1);
    background-color: rgba(0, 0, 0, 0.5);
}
.pwmpv-tips-td {
    color: rgba(255, 0, 0, 1);
    font-size: 14px;
    font-weight: bold;
}
.pwmpv-footer-span {
    margin-top: 10px;
    margin-bottom: 10px;
    color: rgba(255, 255, 255, 1);
}
.pwmpv-footer-span a {
    color: rgba(255, 255, 255, 1);
    text-decoration: none;
    font-size: 14px;
    margin-bottom: 1px;
}
.pwmpv-footer-icon {
    width: 18px;
    height: 18px;
    margin-left: 5px;
    margin-right: 5px;
    margin-bottom: -2px;
}
.pwmpv-support-url-icon {
    width: 30px;
    height: 30px;
    margin-left: 8px;
    margin-right: 8px;
}
.pwmpv-support-url-icon-small {
    width: 25px;
    height: 25px;
    margin-left: 8px;
    margin-right: 8px;
    margin-bottom: 2px;  
}
.pwmpv-support-url-icon-large {
    width: 37px;
    height: 37px;
    margin-left: 8px;
    margin-right: 8px;
    margin-bottom: -4px;
}
`
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 (\`powershell -command \\"Add-Type -AssemblyName System.Web;[System.Web.HTTPUtility]::UrlDecode('%1')\\"\`) DO (SET param=%%F) & SET param=!param:mpv://=! & start /min MPV_PATH !param!\\""
`
// element id
const BUTTON_DIV = "pwmpv-button-div";
const ABOUT_BUTTON_ID = "pwmpv-about-button";
const ABOUT_DIV_ID = "pwmpv-about-div";
const PLAY_BUTTON_ID = "pwmpv-play-button";
const SETTING_BUTTON_ID = "pwmpv-setting-button";
const SETTING_DIV_ID = "pwmpv-setting-div";
const MPV_PATH_INPUT_ID = "pwmpv-mpv-path-input";
const PROXY_INPUT_ID = "pwmpv-proxy-input";
const BILIBILI_CODECS_SELECT_ID = "pwmpv-bilibili-codecs-select";
const SAVE_BUTTON_ID = "pwmpv-save-button";
const DOWNLOAD_BUTTON_ID = "pwmpv-download-button";
// display
const DISPLAY_NONE = "none";
const DISPLAY_FLEX = "flex";
// GM value key
const KEY_MPV_PATH = "MPV_PATH";
const KEY_PROXY = "PROXY";
const KEY_REG_VERSION = "REG_VERSION";
const KEY_BILIBILI_CODECS = "BILIBILI_CODECS";
const BILIBILI_CODECS_HEVC = "12";
function appendHTML() {
    var div = document.createElement("div");
    div.innerHTML = DIV.trim();
    document.body.appendChild(div);
}
function appendCSS() {
    var css = document.createElement("style");
    css.innerHTML = CSS.trim();
    document.head.appendChild(css);
}
var bilibiliCodecs;
var isFullScreen = false;
function addListener() {
    // 关于
    var aboutButton = document.getElementById(ABOUT_BUTTON_ID);
    var aboutDiv = document.getElementById(ABOUT_DIV_ID);
    aboutButton.onclick = function () {
        if (aboutDiv.style.display != DISPLAY_FLEX) {
            aboutDiv.style.display = DISPLAY_FLEX;
            settingDiv.style.display = DISPLAY_NONE;
        } else {
            aboutDiv.style.display = DISPLAY_NONE;
        }
    };
    // 播放
    var playButton = document.getElementById(PLAY_BUTTON_ID);
    playButton.onclick = function () {
        let regVersion = GM_getValue(KEY_REG_VERSION);
        if (!regVersion || regVersion != REG_VERSION) {
            showSettingDiv();
            Toast("🆕 注册表配置有更新,请重新下载并添加注册表信息 🆕");
            return;
        }
        handler.playCurrentVideoWithMPV();
    }
    // 设置
    var settingButton = document.getElementById(SETTING_BUTTON_ID);
    var bilibiliCodecsSelect = document.getElementById(BILIBILI_CODECS_SELECT_ID);
    var saveButton = document.getElementById(SAVE_BUTTON_ID);
    var downloadButton = document.getElementById(DOWNLOAD_BUTTON_ID);
    var settingDiv = document.getElementById(SETTING_DIV_ID);
    var mpvPathInput = document.getElementById(MPV_PATH_INPUT_ID);
    var proxyInput = document.getElementById(PROXY_INPUT_ID);
    settingButton.onclick = function () {
        if (settingDiv.style.display != DISPLAY_FLEX) {
            showSettingDiv();
            aboutDiv.style.display = DISPLAY_NONE;
        } else {
            settingDiv.style.display = DISPLAY_NONE;
        }
    };
    bilibiliCodecs = GM_getValue(KEY_BILIBILI_CODECS);
    if (!bilibiliCodecs) {
        bilibiliCodecs = BILIBILI_CODECS_HEVC;
        GM_setValue(KEY_BILIBILI_CODECS, bilibiliCodecs);
    }
    bilibiliCodecsSelect.onchange = function () {
        bilibiliCodecs = this.value;
        Toast("💡 保存设置后,请刷新当前页面以更新编码 💡", 1500);
    };
    saveButton.onclick = function () {
        let oldMpvPath = GM_getValue(KEY_MPV_PATH);
        let mpvPath = mpvPathInput.value;
        let proxy = proxyInput.value;
        if (!mpvPath) {
            downloadButton.className = "pwmpv-download-disable";
            Toast("⚠️ MPV路径不能为空 ⚠️", 1500);
            return;
        }
        if (/.*[\u4e00-\u9fa5]+.*$/.test(mpvPath)) {
            downloadButton.className = "pwmpv-download-disable";
            Toast("⚠️ MPV路径不能包含中文 ⚠️", 1500)
            return;
        }
        mpvPath = mpvPath.replaceAll("/", "\\");
        mpvPath = mpvPath.replaceAll("\\\\", "\\");
        mpvPath = mpvPath.replaceAll("\\", "\\\\");
        GM_setValue(KEY_MPV_PATH, mpvPath);
        GM_setValue(KEY_PROXY, proxy);
        GM_setValue(KEY_BILIBILI_CODECS, bilibiliCodecs);
        // debug(proxy);
        downloadButton.className = "pwmpv-download-enable";
        if (oldMpvPath != mpvPath) {
            Toast("🔥 请重新添加注册表信息 🔥", 2500);
            downloadButton.click();
        } else {
            Toast("✅  保存成功  ✅", 1500);
        }
    };
    downloadButton.onclick = function () {
        // 生成注册表信息
        var a = document.createElement('a');
        var blob = new Blob([REG.replace(KEY_MPV_PATH, GM_getValue(KEY_MPV_PATH))], { 'type': 'application/octet-stream' });
        a.href = window.URL.createObjectURL(blob);
        a.download = "mpv.reg";
        a.click();
        GM_setValue(KEY_REG_VERSION, REG_VERSION);
    }
    var closeButtons = document.getElementsByClassName("pwmpv-close-button");
    for (let closeButton of closeButtons) {
        closeButton.onclick = function () {
            aboutDiv.style.display = DISPLAY_NONE;
            settingDiv.style.display = DISPLAY_NONE;
        }
    }
    // 全屏
    document.addEventListener("fullscreenchange", () => {
        if (document.fullscreenElement) {
            isFullScreen = true;
            document.getElementById(BUTTON_DIV).style.display = DISPLAY_NONE;
        } else {
            isFullScreen = false;
            handler.checkCurrentVideoUrl();
        }
    });
}
// 显示设置窗口
function showSettingDiv() {
    var downloadButton = document.getElementById(DOWNLOAD_BUTTON_ID);
    var settingDiv = document.getElementById(SETTING_DIV_ID);
    var bilibiliCodecsSelect = document.getElementById(BILIBILI_CODECS_SELECT_ID);
    var mpvPathInput = document.getElementById(MPV_PATH_INPUT_ID);
    var proxyInput = document.getElementById(PROXY_INPUT_ID);
    let mpvPath = GM_getValue(KEY_MPV_PATH);
    let proxy = GM_getValue(KEY_PROXY);
    bilibiliCodecs = GM_getValue(KEY_BILIBILI_CODECS);
    if (mpvPath) {
        mpvPathInput.value = mpvPath;
        downloadButton.className = "pwmpv-download-enable";
    } else {
        downloadButton.className = "pwmpv-download-disable";
    }
    if (proxy) {
        proxyInput.value = proxy;
    }
    bilibiliCodecsSelect.value = bilibiliCodecs;
    settingDiv.style.display = DISPLAY_FLEX;
}
// 显示消息
function Toast(msg, duration) {
    duration = isNaN(duration) ? 3000 : duration;
    var m = document.createElement('div');
    m.innerHTML = msg;
    m.style.cssText = "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;position: fixed;top: 15%;left: 50%;transform: translate(-50%, -50%);z-index: 999999;background: rgba(0, 0, 0, 0.6);font-size: 14px;";
    document.body.appendChild(m);
    setTimeout(function () {
        var d = 0.5;
        m.style.opacity = '0';
        setTimeout(function () { document.body.removeChild(m) }, d * 1000);
    }, duration);
}
// mpv urlprotocol
const MPV_URLPROTOCOL = "mpv://";
// mpv urlprotocol link
class UrlProtocol {
    constructor() {
        this.link = MPV_URLPROTOCOL + '"' + currentVideoUrl + '"';
        this.appendNoTerminal();
        this.needAppendTitle = false;
    }
    // 添加参数
    append(param) {
        this.link = this.link + ' ' + param;
    }
    // 禁止命令行输出及控制
    appendNoTerminal() {
        if (NO_TERMINAL) {
            this.append('--no-terminal');
        }
    }
    // 开始时间(如果 mpv 开启了退出时记住播放状态,则记住状态优先级更高)
    appendStartTime() {
        let startTime = handler.getStartTime();
        if (startTime) {
            this.append('--ss="' + startTime + '"');
        }
    }
    // 标题
    appendTitle() {
        this.needAppendTitle = true;
    }
    // 代理
    appendProxy() {
        let proxy = GM_getValue(KEY_PROXY);
        if (proxy) {
            this.append('--http-proxy=' + proxy + ' --ytdl-raw-options=proxy=[' + proxy + ']');
        }
    }
    // 最终链接
    getLink() {
        if (this.needAppendTitle) {
            // 限制标题长度(url 有长度限制)
            let maxLength = 1900 - this.link.length;
            let title = encodeURIComponent(document.title);
            if (title.length > maxLength) {
                title = title.substring(0, maxLength) + '...';
            }
            this.append('--force-media-title="' + title + '"');
        }
        return this.link;
    }
}
// 网页处理器
var handler;
class Handler {
    // 获取当前视频链接
    getCurrentVideoUrl() { }
    // 获取开始时间
    getStartTime() { return null; }
    // 暂停网页视频
    pauseCurrentVideo() { document.getElementsByTagName("video")[0].pause(); }
    // 获取调用 mpv 链接
    getUrlProtocolLink() {
        let urlProtocol = new UrlProtocol;
        urlProtocol.appendStartTime();
        urlProtocol.appendTitle();
        return urlProtocol.getLink();
    }
    // 校验视频链接是否有效
    checkCurrentVideoUrl() {
        if (this.baseCheckCurrentVideoUrl()){
            if(!isFullScreen){
                document.getElementById(BUTTON_DIV).style.display = DISPLAY_FLEX;
            }
            return true;
        }
        return false;
    }
    // 调用 mpv 播放
    playCurrentVideoWithMPV() {
        window.open(this.getUrlProtocolLink(), "_self");
        let i = 0;
        while (i < 3) {
            i++;
            setTimeout(function () {
                handler.pauseCurrentVideo();
            }, 2000 * i);
        }
    }
    // 根据 class name 获取播放时间
    getStartTimeByClassName(className) {
        let startTimeElements = document.getElementsByClassName(className);
        let length = startTimeElements.length;
        if (length > 0) {
            return startTimeElements[length - 1].innerHTML;
        }
        return null;
    }
    // 视频链接基础校验
    baseCheckCurrentVideoUrl(){
        // debug("current video url: " + currentVideoUrl);
        if (!currentVideoUrl || !currentVideoUrl.startsWith("http") 
            || currentVideoUrl.indexOf("yun.66dm.net") != -1
            || currentVideoUrl.indexOf("www.xmfans.me") != -1
            || currentVideoUrl.indexOf("sod.bunediy.com") != -1) {
            return false;
        }
        return true;
    }
}
// 油管
const YOUTUBE = "www.youtube.com";
class YoutubeHandler extends Handler {
    // 获取当前视频链接
    getCurrentVideoUrl() {
        currentVideoUrl = currentUrl;
        this.checkCurrentVideoUrl();
    }
    // 获取开始时间
    getStartTime() {
        return this.getStartTimeByClassName("ytp-time-current");
    }
    // 获取调用 mpv 链接
    getUrlProtocolLink() {
        let urlProtocol = new UrlProtocol;
        urlProtocol.appendStartTime();
        urlProtocol.appendProxy();
        return urlProtocol.getLink();
    }
    // 校验视频链接是否有效
    checkCurrentVideoUrl() {
        if (currentUrl.indexOf("/watch") == -1 && currentUrl.indexOf("/playlist") == -1) {
            return false;
        }
        return super.checkCurrentVideoUrl();
    }
}
// B站
const BILIBILI = "www.bilibili.com";
// B站 API
const BILIBILI_API = 'https://api.bilibili.com';
// cid 用于传递给 mpv 获取弹幕
var bilibiliCid;
class BilibiliHandler extends Handler {
    // 获取当前视频链接
    getCurrentVideoUrl() {
        let index = currentUrl.indexOf('/video/');
        if (index != -1) {
            // 投稿视频
            let param = "";
            let videoId = currentUrl.substring(index + 7);
            if (videoId.startsWith("BV")) {
                param = "bvid=" + videoId.substring(2, 12);
            } else if (videoId.startsWith("av")) {
                param = "aid=" + videoId.substring(2, 10);
            } else {
                // debug("bilibili video id invalid: " + videoId);
                return;
            }
            // debug("bilibili video id: " + param);
            getBilibiliVideoUrl(param);
        } else {
            // 番剧
            let aElement = document.getElementsByClassName('ep-item cursor visited')[0];
            if (!aElement) {
                aElement = document.getElementsByClassName('ep-item cursor')[0];
            }
            let epid = aElement.getElementsByTagName('a')[0].href;
            epid = epid.substring(epid.indexOf('/ep') + 3);
            epid = epid.substring(0, epid.indexOf('/'));
            // debug('epid: ' + epid);
            let eno = document.getElementsByClassName("ep-list-progress")[0];
            if (!eno) {
                return;
            }
            eno = eno.innerHTML;
            eno = eno.substring(0, eno.indexOf('/'));
            // debug('eno: ' + eno);
            getBilibiliBangumiUrl(epid, eno);
        }
    }
    // 获取开始时间
    getStartTime() {
        let startTime = this.getStartTimeByClassName("bpx-player-ctrl-time-current");
        if (!startTime) {
            startTime = this.getStartTimeByClassName("squirtle-video-time-now");
        }
        return startTime;
    }
    // 获取调用 mpv 链接
    getUrlProtocolLink() {
        let urlProtocol = new UrlProtocol;
        urlProtocol.appendStartTime();
        urlProtocol.appendTitle();
        urlProtocol.append('--audio-file="' + currentAudioUrl + '"');
        urlProtocol.append('--http-header-fields="referer: ' + currentUrl + ',user-agent: ' + navigator.userAgent + '"');
        urlProtocol.append('--script-opts="cid=' + bilibiliCid + '"');
        return urlProtocol.getLink();
    }
}
// 获取B站投稿视频链接
function getBilibiliVideoUrl(param) {
    $.ajax({
        type: "GET",
        url: BILIBILI_API + "/x/web-interface/view?" + param,
        xhrFields: {
            withCredentials: true
        },
        success: function (res) {
            // debug("get acid and cid by avid/bvid result: ");
            // debug(res);
            let avid = res.data.aid;
            let cid = res.data.cid;
            let index = currentUrl.indexOf("?p=");
            if (index != -1 && res.data.pages.length > 1) {
                let p = currentUrl.substring(index + 3);
                let endIndex = p.indexOf("&");
                if (endIndex != -1) {
                    p = p.substring(0, endIndex);
                }
                cid = res.data.pages[p - 1].cid;
            }
            getBilibiliPlayUrl(avid, cid);
        }
    })
}
// 获取B站番剧视频链接
function getBilibiliBangumiUrl(epid, eno) {
    if (!epid || !eno) {
        return;
    }
    $.ajax({
        type: "GET",
        url: BILIBILI_API + "/pgc/view/web/season?ep_id=" + epid,
        xhrFields: {
            withCredentials: true
        },
        success: function (res) {
            // debug("get acid and cid by epid result: ");
            // debug(res);
            let episodes = res.result.episodes;
            if (eno.indexOf('PV') != -1 || eno.indexOf('OP') != -1 || eno.indexOf('ED') != -1) {
                return;
            }
            let episode = episodes[eno - 1];
            getBilibiliPlayUrl(episode.aid, episode.cid);
        }
    })
}
// 获取B站视频播放链接
function getBilibiliPlayUrl(avid, cid) {
    // debug("avid: " + avid);
    // debug("cid: " + cid);
    bilibiliCid = cid;
    let queryBilibiliVideoUrl = "/x/player/playurl?"
        + "qn=&otype=json&fourk=1&fnver=0&fnval=4048"
        + "&avid=" + avid
        + "&cid=" + cid;
    $.ajax({
        type: "GET",
        url: BILIBILI_API + queryBilibiliVideoUrl,
        xhrFields: {
            withCredentials: true
        },
        success: function (res) {
            // debug(res);
            let dash = res.data.dash;
            let hiRes = dash.flac;
            let dolby = dash.dolby;
            if (hiRes && hiRes.audio) {
                // debug("hi-res: on");
                currentAudioUrl = hiRes.audio.baseUrl;
            } else if (dolby && dolby.audio) {
                // debug("dolby: on");
                currentAudioUrl = dolby.audio[0].base_url;
            } else {
                // debug(dash.audio[0].id);
                currentAudioUrl = dash.audio[0].baseUrl;
            }
            let baseUrl = dash.video[0].baseUrl;
            let id = dash.video[0].id;
            let i = 1;
            while (i < dash.video.length) {
                if (dash.video[i].id != id) {
                    break;
                }
                if (dash.video[i].codecid == bilibiliCodecs) {
                    baseUrl = dash.video[i].baseUrl;
                    break;
                }
                i++;
            }
            currentVideoUrl = baseUrl;
            handler.checkCurrentVideoUrl();
        }
    });
}
// 低端影视
const DDRK = "ddys.tv, ddys2.me";
// 低端影视播放状态
var ddrkPlayStatus;
class DdrkHandler extends Handler {
    // 获取当前视频链接
    getCurrentVideoUrl() {
        // 点击播放按钮加载 video 元素
        if (!ddrkPlayStatus) {
            let ddrkPlayButton = document.getElementsByClassName('vjs-big-play-button')[0];
            if (!ddrkPlayButton) {
                // debug("ddrk get play button fail");
                return;
            }
            ddrkPlayButton.click();
            ddrkPlayStatus = true;
        }
        currentVideoUrl = document.getElementById('vjsp_html5_api').src;
        this.checkCurrentVideoUrl();
    }
    // 获取开始时间
    getStartTime() {
        return this.getStartTimeByClassName("vjs-time-tooltip");
    }
}
// 樱花动漫网
const DM6CC = "www.6dm.cc, www.996dm.com";
class Dm6ccHandler extends Handler {
    // 获取当前视频链接
    constructor() {
        super();
        window.addEventListener('message', function (event) {
            currentVideoUrl = event.data;
            this.checkCurrentVideoUrl();
            window.removeEventListener("message", () => { });
        }, false);
    }
    // 暂停网页视频
    pauseCurrentVideo() {
        document.getElementsByTagName("iframe")[2].contentWindow.postMessage("pause", "https://" + YHDMJX);
    }
}
// 风车动漫
const DMLACC = "www.dmlaa.com";
class DmlaccHandler extends Handler {
    // 获取当前视频链接
    constructor() {
        super();
        window.addEventListener('message', function (event) {
            currentVideoUrl = event.data;
            this.checkCurrentVideoUrl();
            window.removeEventListener("message", () => { });
        }, false);
    }
    // 暂停网页视频
    pauseCurrentVideo() {
        document.getElementsByTagName("iframe")[2].contentWindow.postMessage("pause", "https://" + YHDMJX);
    } 
}
// 樱花动漫网和风车动漫实际播放地址
const YHDMJX = "danmu.yhdmjx.com";
class YhdmjxHandler extends Handler {
    constructor() {
        super();
        // 监听父页面暂停指令
        window.addEventListener("message", function (event) {
            if (event.data == "pause") {
                document.getElementsByTagName('video')[0].pause();
            }
        }, false);
    }
    // 获取当前视频链接
    getCurrentVideoUrl() {
        // 发送视频链接到父页面
        currentVideoUrl = document.getElementsByTagName('video')[0].src;
        if (this.checkCurrentVideoUrl()) {
            window.parent.postMessage(currentVideoUrl, "*");
        }
    }
    // 校验视频链接是否有效
    checkCurrentVideoUrl() {
        return this.baseCheckCurrentVideoUrl();
    }
}
// 233动漫网
const DM233 = "www.dm233.me";
class Dm233Handler extends Handler {
    constructor() {
        super();
        this.videoElement = null;
    }
    // 获取当前视频链接
    getCurrentVideoUrl() {
        let iframe = document.getElementById('id_main_playiframe');
        this.videoElement = iframe.contentWindow.document.getElementsByTagName("video")[0];
        let videoUrl = this.videoElement.src;
        if (videoUrl.startsWith("blob:")) {
            videoUrl = iframe.src;
            let startIndex = videoUrl.indexOf('url=http') + 4;
            let endIndex = videoUrl.indexOf('&getplay_url=');
            videoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
        }
        currentVideoUrl = videoUrl;
        this.checkCurrentVideoUrl();
    }
    // 获取开始时间
    getStartTime() {
        return this.getStartTimeByClassName("dplayer-ptime");
    }
    // 暂停网页视频
    pauseCurrentVideo() {
        this.videoElement.pause();
    }
}
// 樱花动漫
const DMH8 = "www.dmh8.com";
class Dmh8Handler extends Handler {
    // 获取当前视频链接
    getCurrentVideoUrl() {
        let iframe = document.getElementsByTagName('iframe')[2];
        let videoUrl = iframe.src;
        let startIndex = videoUrl.indexOf('url=http') + 4;
        let endIndex = videoUrl.indexOf('m3u8') + 4;
        currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
        this.checkCurrentVideoUrl();
    }
    // 获取开始时间
    getStartTime() {
        return this.getStartTimeByClassName("dplayer-ptime");
    }
}
// 樱花动漫
const YHDMP = "www.yhdmp.net";
class YhdmpHandler extends Handler {
    constructor() {
        super();
        this.videoElement = null;
    }
    // 获取当前视频链接
    getCurrentVideoUrl() {
        let iframe = document.getElementById('yh_playfram');
        this.videoElement = iframe.contentWindow.document.getElementsByTagName("video")[0];
        let videoUrl = iframe.src;
        let startIndex = videoUrl.indexOf('url=http') + 4;
        let endIndex = videoUrl.indexOf('&getplay_url=');
        currentVideoUrl = decodeURIComponent(videoUrl.substring(startIndex, endIndex));
        this.checkCurrentVideoUrl();
    }
    // 获取开始时间
    getStartTime() {
        return this.getStartTimeByClassName("dplayer-ptime");
    }
    // 暂停网页视频
    pauseCurrentVideo() {
        this.videoElement.pause();
    }
}
// 巴哈姆特
const GAMER = "ani.gamer.com.tw";
// 巴哈姆特 API
const GAMER_API = "https://ani.gamer.com.tw/ajax/m3u8.php";
class GamerHandler extends Handler {
    // 获取当前视频链接
    getCurrentVideoUrl() {
        let index = currentUrl.indexOf("sn=") + 3;
        if (index == -1) {
            return;
        }
        let sn = currentUrl.substring(index);
        index = sn.indexOf("&");
        if (index != -1) {
            sn = sn.substring(0, index);
        }
        let device = localStorage.ANIME_deviceid;
        // debug("sn: " + sn + ", device: " + device);
        $.ajax({
            type: "GET",
            url: GAMER_API + "?sn=" + sn + "&device=" + device,
            xhrFields: {
                withCredentials: true
            },
            success: function (res) {
                // debug(res);
                currentVideoUrl = JSON.parse(res).src;
                handler.checkCurrentVideoUrl();
            }
        })
    }
    // 获取开始时间
    getStartTime() {
        return this.getStartTimeByClassName("vjs-current-time-display");
    }
    // 获取调用 mpv 链接
    getUrlProtocolLink() {
        let urlProtocol = new UrlProtocol;
        urlProtocol.appendStartTime();
        urlProtocol.appendTitle();
        urlProtocol.appendProxy();
        urlProtocol.append('--http-header-fields="origin: https://ani.gamer.com.tw"');
        return urlProtocol.getLink();
    }
}
// 最大尝试次数
const MAX_TRY_TIME = 8;
// 定时器
var timers;
// 当前页面链接
var currentUrl;
// 当前页面域名
var currentDomain;
// 当前页面视频链接
var currentVideoUrl;
// 当前页面音频链接
var currentAudioUrl;
// 巴哈姆特视频时长
var gamerDurationTime;
// 初始化当前页信息
function initCurrentPageInfo() {
    // debug("init current page info ......");
    document.getElementById(BUTTON_DIV).style.display = DISPLAY_NONE;
    if (timers) {
        for (let timer of timers) {
            // debug("clear timer");
            clearTimeout(timer);
        }
    }
    currentUrl = window.location.href;
    currentDomain = window.location.host;
    currentVideoUrl = "";
    ddrkPlayStatus = false;
}
// 创建处理器
function createHandler() {
    // debug("start create handler: " + currentDomain);
    if (BILIBILI.indexOf(currentDomain) != -1) {
        handler = new BilibiliHandler();
    } else if (DDRK.indexOf(currentDomain) != -1) {
        handler = new DdrkHandler();
    } else if (YOUTUBE.indexOf(currentDomain) != -1) {
        handler = new YoutubeHandler();
    } else if (DM6CC.indexOf(currentDomain) != -1) {
        handler = new Dm6ccHandler();
    } else if (DMLACC.indexOf(currentDomain) != -1) {
        handler = new DmlaccHandler();
    } else if (YHDMJX.indexOf(currentDomain) != -1) {
        handler = new YhdmjxHandler();
    } else if (DM233.indexOf(currentDomain) != -1) {
        handler = new Dm233Handler();
    } else if (DMH8.indexOf(currentDomain) != -1) {
        handler = new Dmh8Handler();
    } else if (YHDMP.indexOf(currentDomain) != -1) {
        handler = new YhdmpHandler();
    } else if (GAMER.indexOf(currentDomain) != -1) {
        handler = new GamerHandler();
    }
}
// 刷新视频链接
function refreshCurrentVideoUrl() {
    // debug("refresh current video url: " + currentVideoUrl);
    // debug("current url: " + currentUrl);
    timers = new Array();
    let tryTime = 0;
    while (tryTime < MAX_TRY_TIME) {
        timers[tryTime] = setTimeout(function () {
            if (!handler.checkCurrentVideoUrl()) {
                handler.getCurrentVideoUrl();
            }
            // debug("timer done");
        }, tryTime * 2000 + 700);
        tryTime = tryTime + 1;
    }
}
// 页面变更监听器
function pageChangeListener() {
    // debug("page change listener");
    let needRefresh = false;
    let newCurrentUrl = window.location.href;
    if (currentUrl != newCurrentUrl) {
        needRefresh = true;
    }
    // 巴哈姆特
    if (!needRefresh && GAMER.indexOf(currentDomain) != -1) {
        let oldGamerDurationTime = gamerDurationTime;
        let durationDiv = document.getElementsByClassName("vjs-duration-display")[0];
        if (durationDiv) {
            gamerDurationTime = durationDiv.innerHTML;
            if (oldGamerDurationTime && oldGamerDurationTime != gamerDurationTime) {
                needRefresh = true;
            }
        }
    }
    if (needRefresh) {
        // debug("page change");
        initCurrentPageInfo();
        refreshCurrentVideoUrl();
    }
}
// 添加组件和监听器
appendHTML();
appendCSS();
addListener();
// 初始化页面信息
initCurrentPageInfo();
// 创建处理器
createHandler();
// 刷新视频链接
refreshCurrentVideoUrl();
// 定时监听页面变化
setInterval(pageChangeListener, 700);