ALIST 外调mpv播放器进行在线播放视频

xc
xc
2025-05-05 / 0 评论 / 5 阅读 / 正在检测是否收录...

概述:

最近对mpv播放器很感兴趣,alist网盘程序虽然下面继承了多个第三方播放器,但是没Mpv,原因就是mpv播放器默认不会将mpv://协议注册到系统中。

正文:

那我们就自己动手吧,大概原理也很简单,让系统知道mpv://协议应该调用mpv播放器进行播放,
另外就是在alist网页中注入相关按钮和生成mpv://协议。

我们先完成系统注册,网上都会提到使用 mpv-handler ,但我试了多次,最终也没成功,我很奇怪,它通过注册表启动mpv-handler.exe,然后这个应用查找本地的config文件,在config文件中,我们要修改本地的mpv播放器路径。

流程

多次尝试,最后总结的成功流程:
先系统注册表,告诉系统mpv://协议,就去启动一个mpv-handler.bat或mpv-handler.exe的文件,因为此时的URL真实地址前面多了一个mpv://,MPV播放器正常来说,还是不识别的,当然,有的系统会被识别,为了统一标准,提高运行成功率,我们需要mpv-handler.bat或mpv-handler.exe将mpv://删除,只保留后面的视频地址给Mpv播放器。

注册mpv-handler.bat或mpv-handler.exe

创建注册表脚本
新建一个文本文件,粘贴以下内容,然后另存为 mpv_protocol.reg(确保保存类型选"所有文件"):

生成mpv-handler.bat

@echo off
setlocal

:: 去掉前缀并解码
set "raw_url=%~1"
set "raw_url=%raw_url:mpv://=%"

:: 确保 URL 是有效的(修复 https// 到 https://)
set "raw_url=%raw_url:https//=https://%"

:: 打印 URL 和路径以调试
echo Incoming URL: %~1
echo Decoded URL: %raw_url%
echo MPV Path: "D:\mpv\mpv\mpv.exe"

:: 启动 mpv
start "" "D:\mpv\mpv\mpv.exe" "%raw_url%"

:: 检查是否成功启动
if %errorlevel% neq 0 (
    echo Failed to start MPV! Error code: %errorlevel%
) else (
    echo MPV started successfully!
)

生成mpv-handler.exe
首先要安装好python
其次安装模块

pip install pyinstaller
pip install pyinstaller

系统有多个python记得前面加

python -m

mpv-handler.py

( 请根据需要修改D:\mpv\mpv\mpv.exe)

import sys
import subprocess

def fix_url(url):
    """ 修复 URL 格式,例如将 https// 修正为 https:// """
    url = url.replace('mpv://', '')  # 去掉前缀
    url = url.replace('https//', 'https://')  # 修正协议部分
    return url

if len(sys.argv) > 1:
    raw_url = sys.argv[1]
    fixed_url = fix_url(raw_url)  # 修复 URL
    mpv_path = r"D:\mpv\mpv\mpv.exe"
    
    # 启动 MPV 播放器并传入修复后的 URL
    subprocess.Popen([mpv_path, fixed_url])

编译

pyinstaller --onefile mpv-handler.py

出错就强制编译

C:\Users\Admin\AppData\Local\Programs\Python\Python312\Scripts\pyinstaller.exe --onefile mpv-handler.py

新生成的exe文件在:dist\mpv-handler.exe

修改alist页面

大概alist管理页面,在全局的自定义中,添加以下代码。
ALIST管理页面

<script>
(function() {
    'use strict';

    // --- Configuration ---
    const MPV_PROTOCOL = 'mpv://';
    const BUTTON_CONTAINER_SELECTOR = 'div.hope-flex[class*="igXrpAn-css"]'; // More flexible selector
    const EXISTING_BUTTON_SELECTOR = 'a.hope-anchor[href*="://"]';
    const MPV_BUTTON_ID = 'alist-external-mpv-button';
    // *** IMPORTANT: Make sure this path is correct OR use Base64 ***
    const MPV_ICON_SRC = ''; // Or '_BASE64_STRING'
    const DEBUG_MODE = true; // Set to true for more console logs

    // --- Logging Function ---
    function log(...args) {
        if (DEBUG_MODE) {
            console.log('[AList MPV Button]', ...args);
        }
    }

    log("Script Initializing...");

    // --- Helper Functions ---
    function extractRawUrl(href) {
        if (!href) return null;
        log("Attempting to extract raw URL from:", href);
        const specificSchemes = [
            { prefix: 'iina://weblink?url=', isBase64: false },
            { prefix: 'potplayer://', isBase64: false },
            { prefix: 'vlc://', isBase64: false },
            { prefix: 'nplayer-', isBase64: false },
            { prefix: 'omniplayer://weblink?url=', isBase64: false },
            { prefix: 'figplayer://weblink?url=', isBase64: false },
            { prefix: 'infuse://x-callback-url/play?url=', isBase64: false },
            { prefix: 'filebox://play?url=', isBase64: false },
            { prefix: 'intent:', isBase64: false },
            { prefix: 'iplay://play/any?type=url&url=', isBase64: true }
        ];

        for (const scheme of specificSchemes) {
            if (href.startsWith(scheme.prefix)) {
                let urlPart = href.substring(scheme.prefix.length);

                 // Handle intent URLs
                 if (scheme.prefix === 'intent:') {
                    const intentEndIndex = urlPart.indexOf('#Intent');
                    if (intentEndIndex !== -1) {
                        urlPart = urlPart.substring(0, intentEndIndex);
                    }
                 }

                // --- 修改开始: 先移除后缀,然后解码 ---
                urlPart = urlPart.replace(/:0$/, ''); // 移除潜在的 :0 后缀

                if (scheme.isBase64) {
                    try {
                        urlPart = atob(urlPart);
                        log("Decoded Base64 URL:", urlPart);
                    } catch (e) {
                        console.error("Error decoding Base64 URL:", e, urlPart);
                        return null;
                    }
                } else {
                    // 对非 Base64 的 URL 进行 URL 解码
                    try {
                        urlPart = decodeURIComponent(urlPart);
                        log("URL-decoded part:", urlPart);
                    } catch (e) {
                         console.error("Error URL-decoding part:", e, urlPart);
                         // 如果解码失败(比如 URL 本身有问题),也认为无效
                         return null;
                    }
                }
                // --- 修改结束 ---

                // 现在检查解码后的 URL
                if (urlPart.startsWith('http://') || urlPart.startsWith('https://')) {
                     log("Extracted URL:", urlPart);
                     return urlPart;
                } else {
                     log("Processed part doesn't look like a valid URL:", urlPart);
                }
            }
        }

        // Fallback (也需要解码)
        const genericMatch = href.match(/(https?:\/\/.*)/);
        if (genericMatch && genericMatch[1]) {
            try {
                const fallbackUrl = decodeURIComponent(genericMatch[1].replace(/:0$/, ''));
                 log("Extracted URL using fallback (decoded):", fallbackUrl);
                 if (fallbackUrl.startsWith('http://') || fallbackUrl.startsWith('https://')) {
                      return fallbackUrl;
                 }
            } catch (e) {
                 console.error("Error URL-decoding fallback URL:", e, genericMatch[1]);
            }
        }

        console.warn("Could not extract raw URL from:", href);
        return null;
    }

    function createMpvButton(rawUrl) {
        if (!rawUrl) return null;
        log("Creating MPV button for URL:", rawUrl);
        const mpvUrl = MPV_PROTOCOL + rawUrl;

        const link = document.createElement('a');
        link.id = MPV_BUTTON_ID;
        // Use specific classes found in the example, be careful if they change
        link.className = 'hope-anchor hope-c-iHuheP hope-c-PJLV hope-c-PJLV-idrWMwW-css';
        link.href = mpvUrl;
        link.target = "_blank";
        link.style.marginLeft = '4px'; // Add a little space

        const img = document.createElement('img');
        // Use specific classes found in the example
        img.className = 'hope-image hope-c-PJLV hope-c-PJLV-idGrHCo-css';
        img.src = MPV_ICON_SRC;
        img.alt = "MPV";
        img.title = "使用 MPV 播放";
        // Add basic error handling for the image
        img.onerror = () => {
             console.error(`Failed to load MPV icon: ${MPV_ICON_SRC}`);
             img.alt = "MPV (icon error)"; // Show text if icon fails
             img.style.width = '20px'; // Give it some size even without icon
             img.style.height = '20px';
             img.style.border = '1px solid red'; // Make it visible
         };

        link.appendChild(img);
        return link;
    }

    function addMpvButtonToContainer(container) {
        if (!container) {
            log("addMpvButtonToContainer called with null container");
            return;
        }
        if (container.querySelector(`#${MPV_BUTTON_ID}`)) {
            log("MPV button already exists in container:", container);
            return; // Already added
        }
        log("Attempting to add MPV button to container:", container);

        const existingButton = container.querySelector(EXISTING_BUTTON_SELECTOR);
        if (!existingButton || !existingButton.href) {
            log("No existing button with href found in container.");
            return;
        }

        const rawUrl = extractRawUrl(existingButton.href);
        if (!rawUrl) {
            log("Failed to extract raw URL from existing button:", existingButton.href);
            return;
        }

        const mpvButton = createMpvButton(rawUrl);
        if (mpvButton) {
            // Append the button
            container.appendChild(mpvButton);
            log("MPV button added successfully.");
        } else {
             log("Failed to create MPV button element.");
        }
    }

    // --- MutationObserver Logic ---
    const observerCallback = (mutationsList, observer) => {
        log("MutationObserver detected changes...");
        for (const mutation of mutationsList) {
            if (mutation.type === 'childList') {
                mutation.addedNodes.forEach(node => {
                    if (node.nodeType === Node.ELEMENT_NODE) {
                        // Check if the added node itself is the container
                        if (node.matches && node.matches(BUTTON_CONTAINER_SELECTOR)) {
                            log("Container node added:", node);
                            addMpvButtonToContainer(node);
                        }
                        // Check if the added node contains the container
                        else if (node.querySelectorAll) {
                            const containers = node.querySelectorAll(BUTTON_CONTAINER_SELECTOR);
                            if (containers.length > 0) {
                                log("Container node found within added node:", node);
                                containers.forEach(addMpvButtonToContainer);
                            }
                        }
                    }
                });
                 // Also check removed nodes in case the container is re-rendered quickly
                 // This might be overkill, but could help in some frameworks
                 mutation.removedNodes.forEach(node => {
                      if (node.nodeType === Node.ELEMENT_NODE && node.matches && node.matches(BUTTON_CONTAINER_SELECTOR)) {
                          log("Container node removed, will re-check on next addition.");
                      }
                 });
            }
            // Optional: Observe attribute changes if the container exists but buttons are added later
            // else if (mutation.type === 'attributes' && mutation.target.matches(BUTTON_CONTAINER_SELECTOR)) {
            //     log("Attributes changed on container, re-checking:", mutation.target);
            //     addMpvButtonToContainer(mutation.target);
            // }
        }
    };

    // Start observing
    const observer = new MutationObserver(observerCallback);
    // Observe body, but consider observing a more specific, stable parent if possible
    observer.observe(document.body, { childList: true, subtree: true });
    log("Observer started.");

    // --- Initial Check ---
    log("Performing initial check for existing containers...");
    document.querySelectorAll(BUTTON_CONTAINER_SELECTOR).forEach(container => {
        log("Found existing container on load:", container);
        addMpvButtonToContainer(container);
    });
    log("Initial check complete.");

})();
</script>

总结:

mpv-handler.bat和mpv-handler.exe二选一就可以了,在win系统中exe文件要比bat文件更可靠一些,但bat文件修改起来更方便一些,按需选择,最后记得更新mpv_protocol.reg的位置信息。

我已经打包好了这些文件,确保你的mpv.exe和我一致的话(D:\mpv\mpv\mpv.exe),拿来直接用就可以了。

0

评论 (0)

取消
使用 Typecho 建站,并搭配 joe 主题(有修改)
蜀ICP备2022005623号-2 川公网安备 51012202001212号
本站已运行 00000000
Copyright © 1970 ~ Xcshare All rights reserved.