Greasy Fork is available in English.

柒灵全能视频下载器

(目前支持)哔哩哔哩(bilibili)/优酷网(youku)/腾讯视频(qq)/爱奇艺(iqiyi)视频批量下载, 加速下载不卡顿.

À partir de 2020-11-26. Voir la dernière version.

// ==UserScript==
// @name         柒灵全能视频下载器
// @namespace    https://weibo.com/guoxuebiji/profile?is_all=1
// @version      2.0.5
// @description  (目前支持)哔哩哔哩(bilibili)/优酷网(youku)/腾讯视频(qq)/爱奇艺(iqiyi)视频批量下载, 加速下载不卡顿.
// @author       东风
// @date         2020-4-25
// @modified     2020-11-15
// @match        http*://*.bilibili.com/video/*
// @match        http*://*.bilibili.com/bangumi/play/*
// @match        http*://*.bilibili.com/*/favlist*
// @match        http*://v.youku.com/v_show/*
// @match        http*://m.youku.com/alipay_video/*
// @match        http*://v.qq.com/x/cover/*
// @match        http*://m.v.qq.com/x/cover/*
// @match        http*://v.qq.com/x/page/*
// @match        http*://m.v.qq.com/x/page/*
// @match        http*://m.v.qq.com/*
// @match        http*://www.iqiyi.com/v*
// @match        http*://m.iqiyi.com/*
// @match        http*://www.iqiyi.com/*
// @match        http*://m.iqiyi.com/kszt/*
// @match        http*://www.iqiyi.com/kszt/*
// @icon         https://space.bilibili.com/favicon.ico
// @license      BSD 3-Clause License
// @grant        unsafeWindow
// @grant        GM_setClipboard
// @grant        GM_info
// @grant        GM_download
// @grant        GM_registerMenuCommand
// @grant        GM_xmlhttpRequest
// @grant        GM.xmlHttpRequest
// @grant        GM_openInTab
// @grant        GM.openInTab
// @grant        GM_getValue
// @grant        GM.getValue
// @grant        GM_setValue
// @grant        GM.setValue
// ==/UserScript==

(function () {
    'use strict';
    //==========utils=====================================================================
    //加载css文件
    function addCSS(href) {
        var link = document.createElement('link');
        link.type = 'text/css';
        link.rel = 'stylesheet';
        link.href = href;
        document.getElementsByTagName("head")[0].appendChild(link);
    }
    //加载js文件
    function addJS(src, cb) {
        var script = document.createElement("script");
        script.type = "text/javascript";
        script.src = src;
        console.log("addJS",script)
        document.getElementsByTagName('head')[0].appendChild(script);
        script.onload = typeof cb === "function" ? cb : function () {};
    }

    // 加载css字符串
    function GMaddStyleString(css) {
        var myStyle = document.createElement('style');
        myStyle.textContent = css;
        var doc = document.head || document.documentElement;
        doc.appendChild(myStyle);
    }

    function AddHtml(html) {
        document.body.insertAdjacentHTML('afterEnd', html);
    }


    // 改用ZUI //https://www.openzui.com/
    addJS("https://cdnjs.cloudflare.com/ajax/libs/zui/1.9.2/lib/jquery/jquery.js", function () {
        console.log("-------------------load jq-------------------")
        window.$ = $.noConflict();
        addCSS("https://cdnjs.cloudflare.com/ajax/libs/zui/1.9.2/css/zui.min.css");
        addJS("https://cdnjs.cloudflare.com/ajax/libs/zui/1.9.2/js/zui.min.js");
    });


    function GetFileName(url) {
        var Business=url.split("/");
        return Business[Business.length-1];
    }

    // String.prototype.TextFilter=function(){
    //     var pattern=new RegExp("[`~%!@#^=''?~!@#¥……&——‘”“'?*()(),,。.、]"); //[]内输入你要过滤的字符
    //     var rs="";
    //     for(var i=0;i<this.length;i++){
    //         rs+=this.substr(i,1).replace(pattern,'');
    //     }
    //     return rs;
    // }

    // 把空格和斜杠转换成下划线
    function Trim(str)
    {
        // str = str.TextFilter()
        var result = str.replace(/\s/g,"_");
        result = result.replace(/\//g,"_");
        result = result.replace(/\\/g,"_");
        result = result.replace(/&/g,"-");
        result = result.replace(/"/g,"");
        return result;
    }

    // 去掉标题后缀
    function FormatTitle(str)
    {
        var title = Trim(document.title)
        var n = title.lastIndexOf(str);
        if (n >= 0) {
            title = title.substring(0,n);
        }
        return title;
    }   

    function ShowTips(str) {
        new $.zui.Messager(str, {
            type: 'success', // 定义颜色主题
            time:2000
        }).show();
    }
    //字符串是否包含子串
    function isContains(str, substr) {
        //str是否包含substr
        return str.indexOf(substr) >= 0;
    }

    // 把网页获取的对象转换成数组
    function objToArray(x) {
        var list = [];
        console.log(x)
        for (var i = 0; i < x.length; i++) {
            list[i] = x[i];
        }
        return list
    }

    function ShowSelect(senddata) {
        window.g_senddata = senddata
        console.log("ShowSelect", JSON.stringify(senddata))
        var select_window = $("#select_window")
        console.log(select_window)
        if (!select_window[0]) {
            var html = 
            `
            <div class="modal fade" id="select_window" style="z-index:10000">
              <div class="modal-dialog modal-lg">
                <div class="modal-content">
                  <div class="modal-header">
                    <button type="button" class="close" data-dismiss="modal"><span aria-hidden="true">×</span><span class="sr-only">关闭</span></button>
                    <h4 class="modal-title">请选择要下载的视频</h4>
                  </div>
                  <div class="modal-body">
                        <form id='select_form' class="layui-form" action="" lay-filter="example">
                        </form>
                  </div>
                  <div class="modal-footer center">
                    <button type="button" class="btn btn-primary" id="btn_download_all">下载选中</button>
                    <button type="button" class="btn btn-primary" id="btn_select_all">取消/全选</button>
                    <button type="button" class="btn btn-primary" id="btn_download_self">仅下载本视频</button>
                    <button type="button" class="btn btn-default" data-dismiss="modal" >关闭</button>
                  </div>
                </div>
              </div>
            </div>
            `
            AddHtml(html)//style="display:none"    
            // select_form = $("#select_form") 

            $(function() {
                $("#btn_download_self").click(function(){
                    RealDownload(DownloadSelfInfo())
                });
            }); 

            $(function() {
                $("#btn_download_all").click(function(){
                    var list = $("[id='listItem']:checkbox")
                    var res = []
                    for (var i = 0; i < list.length; i++) {
                        if (list[i].checked) {
                            res.push(window.g_senddata[i])
                        }
                    }
                    console.log(res)
                    RealDownload(res)
                });
            }); 

             $(function() {
                $("#btn_select_all").click(function(){
                    var list = $("[id='listItem']:checkbox")
                    var flag = true
                    if (list[0] && list[0].checked) {
                        flag = false
                    }
                    console.log("flag" , flag)
                    console.log(list)
                    for (var i = 0; i < list.length; i++) {
                        list[i].checked = flag
                    }
                });
            });

        }
        else{
            if ($("#select_window").is(":visible")) {
                $("#select_window").hide()
                // alert("存在")
                return
            }
        }
        var select_form = $("#select_form")

        //弹出表格
        var s_content = `
            <form class="layui-form" action="" lay-filter="example">
              <div class="layui-form-item" pane="">
                <label class="layui-form-label">视频列表</label>`

        for (var i = 0; i < senddata.length; i++) {
            senddata[i].id = i+1
            var checkboxid = "select_checkbox_id"+i
            s_content = s_content + `<div class="checkbox-primary"><input type="checkbox" checked="checked" id="listItem"><label for="`+checkboxid+`">`+ senddata[i].fileName +`</label></div>`

        }

        s_content = s_content + `</div></form>`
        console.log(s_content)
        select_form[0].innerHTML = s_content

        $('#select_window').modal({
            scrollInside : true,
            moveable : "inside",
            show     : true
        })
    }

    function Download(urls) 
    { 
        ShowSelect(urls)
    };
    //==========以下是与下载器通讯=====================================================================
    var host = '127.0.0.1'
    ,port = '5678';
    
    function wsmessage(evt) 
    { 
        console.log(evt);
        var received_msg = evt.data;
        console.log("收到服务器的信息", received_msg);
        // ws.send(JSON.stringify(["https://www.bilibili.com/video/BV19E411D78Q?p=6","https://www.bilibili.com/video/BV19E411D78Q?p=3"]));
        // console.log(JSON.parse(received_msg));
        // 发送成功{id: 2333333, jsonrpc: "2.0", result: "a6ff40d33524229a"}
        // 开始下载{jsonrpc: "2.0", method: "aria2.onDownloadStart", params: [[gid: "a6ff40d33524229a"]]}
        // 下载完成{jsonrpc: "2.0", method: "aria2.onDownloadComplete", params: [[gid: "a6ff40d33524229a"]]}
        // 下载出错{jsonrpc: "2.0", method: "aria2.onDownloadError", params: [[gid: "a6ff40d33524229a"]]}

        ShowTips("任务发送成功");
    };
        
    function wsclose() 
    { 
        console.log("连接关闭");
        // layui.use('layer', function(){
        //   var layer = layui.layer;
        //     layer.msg("连接关闭", {
        //         icon: 1
        //     });
        // }); 
    };

    function RealDownload(url, out, dir)
    {
        console.log("RealDownload",url)
        // var json = MakeSendData(url, out, dir)
        var ws = new WebSocket("ws://" + host + ":" + port + "/jsonrpc");

        function wsopen()
        {
            console.log("连接下载服务器");
            // var json = MakeSendData("http://aria2c.com/usage.html", "test.html", "./dow")
            // ws.send(JSON.stringify(["https://www.bilibili.com/video/BV19E411D78Q?p=5","https://www.bilibili.com/video/BV19E411D78Q?p=3"]));
            ws.send(JSON.stringify(url));

            setTimeout(function() {
                ws.close()
            }, 10000)
            
        };

        setTimeout(function() {
            if (ws.readyState===1) {
                //连接成功什么事情都不用处理
            }else{
                ShowTips("连接下载器失败,请确认开启下载器");
            }
        }, 1000)

        ws.onopen =  wsopen;
        ws.onmessage = wsmessage;
        ws.onclose =  wsclose;
    };

    //===============================================================================

    // 拷贝我的收藏视频网址
    function CopyFavlistUrls() {

        var title = "我的收藏"

        var x = document.getElementsByClassName("fav-video-list clearfix content")[0].children;
        console.log(title)

        var list = [];
        console.log(x)
        for (var i = 0; i < x.length; i++) {
            console.log(i,x[i].attributes["class"].nodeValue)
            if (x[i].attributes["class"].nodeValue != "small-item disabled") {
                list[i] = x[i];
            }
        }

        console.log(list)
        // zoomfile
        if (list) {

            GM_setClipboard( list.map(function (pin) {
                return "https:" + pin.children[0].attributes["href"].nodeValue + "\r\n";
            }).join(""));

            return list.map(function (pin) {
                var item = {"url": "https:" + pin.children[0].attributes["href"].nodeValue};
                item.folder = "我的收藏"

                item.islist = false;
                return item;
            })
        }else
        {
            ShowTips("找不到视频");
        }
    }

    // 拷贝播放列表视频网址
    function CopyVedioUrls() {
        var title = Trim(document.querySelector('meta[property="og:title"]').getAttribute('content'));//FormatTitle("_哔哩哔哩(゜-゜)つロ干杯~-bilibili")

        var list_box = document.getElementsByClassName("list-box")
        if (list_box.length == 0) {
            return DownloadSelfInfo()
        }

        var x = document.getElementsByClassName("list-box")[0].children;
        console.log(title)

        var list = [];
        console.log(x)
        for (var i = 0; i < x.length; i++) {
            list[i] = x[i];
        }

        console.log(list)
        if (list) {
            return list.map(function (pin) {
                var item = {"url": "https://www.bilibili.com" + pin.children[0].attributes["href"].nodeValue}
                item.folder = Trim(title)
                item.fileName = pin.children[0].title
                item.islist = false;
                return item;
            })
        }else
        {
            ShowTips("找不到视频");
        }
    }
    // 拷贝番剧播放列表视频网址
    function CopyBangumiUrls() {
        jQuery.ajax({
            url: window.location.href,
            async: false,
            success: function (res) {
                // console.log(res);
                var info = res

                var n = info.lastIndexOf("<script>window.__INITIAL_STATE__");
                if (n >= 0) {
                    info = info.substring(n+"<script>window.__INITIAL_STATE__=".length);
                    var n2 = info.indexOf(";(function(){var s;(s=document.currentScript||document.scripts[document.scripts.length-1]).parentNode.removeChild(s);}());</script>");
                    // console.log("1111",n, n2,info)
                    info = info.substring(0,n2);
                }
                var bili_state = JSON.parse(info)
                if (bili_state) {

                    console.log(bili_state)
                    var epList = bili_state.epList 
                    var title = bili_state.mediaInfo.title
                    var ssType = bili_state.ssType 
                    // console.log(bili_state.epList)
                    // console.log("epList =", epList, "mediaInfo = ", mediaInfo)
                    // for (var i = 0; i < epList.length; i++) {
                    //     console.log(i, epList[i])
                    // }

                    // if (ssType==1) { //番剧

                        var urls = epList.map(function (ep) {
                            var item = {
                                "url": "https://www.bilibili.com/bangumi/play/ep"+ ep.id, 
                                "titleFormat":Trim(ep.titleFormat), 
                                "longTitle":Trim(ep.longTitle),
                                "fileName":Trim(ep.titleFormat+"_"+ep.longTitle)
                            }
                            if (ep.longTitle == "") {
                                item.fileName = Trim(ep.titleFormat)
                            }
                            item.folder = Trim(title)
                            item.islist = false;
                            return item;
                        })
                        Download(urls)
                    // }else
                    // {// 电影
                    //     DownloadSelf(title, false)
                    // }
                }else
                {
                    var title = document.querySelector('meta[property="og:title"]').getAttribute('content');
                    DownloadSelf(title, true)
                }
            },
            omplete: function (data) {
                if (data.status === 200) {
                }
                else {
                    ShowTips("系统错误:暂时无法连接服务器")
                    var title = document.querySelector('meta[property="og:title"]').getAttribute('content');
                    DownloadSelf(title, true)
                }
            }
        });
    }

    function DownloadSelf(title, islist, youtube) {
        Download(DownloadSelfInfo(title, islist, youtube))
    }

    function DownloadSelfInfo(title, islist, youtube) {
        var item = {"url": window.location.href}
        // item.folder = Trim(title); // 不能有空格,有时候下载失败是因为有空格
        if (!title) {
            title = Trim(document.title)
        }
        if (!islist) {
            islist = false
        }
        item.fileName = Trim(title)
        item.islist = islist;
        var host = getEffectiveHost()
        if (host == "youku") {
            item.youtube = true
        }
        return [item]
    }


    // 解析bilibili网站
    function ParseBilibiliUrl() {
        var arrUrl = window.location.pathname.split('/');
        console.log("arrUrl = ", arrUrl);
        if(!arrUrl[1]) return;
        if (arrUrl[1] == "video") 
            return CopyVedioUrls();
        else if (arrUrl[1] == "bangumi") 
            return CopyBangumiUrls();
        else if (arrUrl[2] == "favlist") 
            return CopyFavlistUrls();
        else
        {
            return DownloadSelfInfo()
        }
    }

    // 解析优酷网站
    function ParseYoukuUrl() {
        var arrUrl = window.location.pathname.split('/');
        console.log("arrUrl = ", arrUrl);

        if(!arrUrl[1]) return;
        if (arrUrl[1] == "v_show") 
        {
            var title = Trim(document.querySelector('meta[name="irAlbumName"]').getAttribute('content')); 
            var l = document.getElementsByClassName("anthology-content")
            if (l.length == 2) {
                var x = document.getElementsByClassName("anthology-content")[0].children;
                var list = objToArray(x)
                if (list) {
                    return list.map(function (pin) {
                        var item = {"url": pin.children[0].attributes["href"].nodeValue}
                        item.saveFileName = Trim(pin.title) //优酷下载的名称没有第几集, 所以需要重命名
                        item.fileName = item.saveFileName
                        item.folder = Trim(title)
                        item.islist = false;
                        item.youtube = true
                        return item;
                    })
                }
            }        
        }

        return DownloadSelfInfo(null, false, true)
    }

    // 解析腾讯视频网站
    function ParseQQUrl() {
        var arrUrl = window.location.pathname.split('/');
        console.log("arrUrl = ", arrUrl);

        if(!arrUrl[1]) return;
        if (arrUrl[1] == "x") 
        {
            var player_title = document.getElementsByClassName("player_title")
            if (player_title.length==0) {
                
            }else{

                var title = Trim(document.getElementsByClassName("player_title")[0].children[0].text); //player_title
                var x = document.getElementsByClassName("figure_list _hot_wrapper")[0].children; 
                var list = objToArray(x)
                if (list) {
                    return list.map(function (pin) {
                        var item = {"url": "https://v.qq.com" + pin.children[0].attributes["href"].nodeValue}
                        item.fileName = Trim(pin.attributes["data-title"].nodeValue) 
                        item.folder = Trim(title)
                        item.islist = false;
                        return item;
                    })
                }
            }        
        }

        return DownloadSelfInfo()
    }

    // 解析爱奇艺网站
    function ParseIqiyiUrl() {
        var title = Trim(document.querySelector('meta[name="irAlbumName"]').getAttribute('content')); 
        var x = document.getElementsByClassName("select-title"); //专辑
        var list = objToArray(x)

        if (list.length > 0) {
            return list.map(function (pin) {
                var item = {"url": "https:" + pin.children[1].attributes["href"].nodeValue}
                item.saveFileName = Trim(pin.children[0].innerText+"_"+pin.children[1].text) 
                item.fileName = item.saveFileName
                item.folder = Trim(title)
                item.islist = false;
                return item;
            })
        }else
        {
            return DownloadSelfInfo()
        }            
    }

    //获取可使用域名
    function getEffectiveHost() {
        var host = window.location.host;
        if (!host) {
            host = document.domain;
        }
        if (!host) {
            host = "bilibili.com";
        }
        if (isContains(host, "bilibili")) {
            host = "bilibili";
        } else if (isContains(host, "youku")) {
            host = "youku";
        }else if (isContains(host, "iqiyi")) {
            host = "iqiyi";
        } else if (isContains(host, "qq.com")) {
            host = "qq";
        } else {
            host = "bilibili";
        }
        return host;
    }

    function ParseUrl() {   
        // console.log("----------------------222222",$.fn.jquery)
        var host = getEffectiveHost()
        if (host == "bilibili") {
            return ParseBilibiliUrl()
        } else if (host == "youku") {
            return ParseYoukuUrl()
        } else if (host == "iqiyi") {
            return ParseIqiyiUrl()
        } else if (host == "qq") {
            return ParseQQUrl()
        }
        else {
            return DownloadSelfInfo()
        }
    }


    /*
        主入口,分出不同模块:用户、画板,监听并刷新URL
    */
    window.onload =function() {

            var btn = document.getElementsByClassName("btnDownload")
            if (btn.length > 0) {
                // btn.innerHTML = tmpHtml
                return
            }

            GMaddStyleString(`#download_movie_box {cursor:pointer; position:fixed; top:` + 60 + `px; left:` + 0 + `px; width:0px; background-color:#2E9AFE; z-index:2147483647; font-size:20px; text-align:left;}
                #download_movie_box .item_text {width:28px; padding:4px 0px; text-align:center;}
                #download_movie_box .item_text img {width:35px; height:35px; display:inline-block; vertical-align:middle;}
                `);

            // var $ = $ || window.$;
            var ImgBase64Data = ""

            var html = 
            `<div id='download_movie_box' class="btnDownload">
                <div class='item_text'>
                    <img src='`+ ImgBase64Data +`' title='下载视频' id="downloadVideos"/>
                </div>
            </div>`;

            document.body.insertAdjacentHTML('afterEnd', html);

            document.getElementById("downloadVideos").onclick = function () {
                var urls = ParseUrl();
                console.log("urls = ",urls)
                if (urls) {
                    Download(urls)
                }
            };
        // },3000)
    }

    // GM_setValue("mytset","mytset---------------------------")
    // console.log(GM_getValue("mytset"))
})();