学堂在线 xuetangx.com 批量下载视频及字幕 Aria2脚本

1.提取视频和字幕的链接 2.按照上下文自动生成文件名 3.点击按钮通过 JSON-RPC 调用 aria2 下载至指定文件夹

// ==UserScript==
// @name         学堂在线 xuetangx.com 批量下载视频及字幕 Aria2脚本
// @namespace    mudan_cn
// @version      2019.9.14
// @description  1.提取视频和字幕的链接 2.按照上下文自动生成文件名 3.点击按钮通过 JSON-RPC 调用 aria2 下载至指定文件夹
// @author       mudan_cn
// @match        http://www.xuetangx.com/courses/**
// @match        https://www.xuetangx.com/courses/**
// @license      MIT License
// @run-at       document-idle
// @grant        GM_getValue
// @grant        GM_setValue
// ==/UserScript==

$('div.course_mask').remove();
class Aria2Config {
}
const default_aria2_config = {
    uri: 'http://localhost:6800/jsonrpc',
    token: '',
    storage_directory: 'D:\\\\'
};
let counter = 0;
const id = setInterval(() => {
    try {
        const unit = $('.active.ui-accordion-header')[0].textContent.trim().replace(' ', ' ');
        const section = $('#accordion li.active p:first')[0].firstChild.textContent.trim().replace(' ', ' ');
        const videos = $('.xt_video_player').map((i, x) => x.getAttribute('src')).get();
        const transcripts = $('a[href$="transcript/download"]').map((i, x) => 'http://www.xuetangx.com' + x.getAttribute('href')).get();
        counter++;
        //loop and wait for the next turn
        if (videos.length < transcripts.length && counter < 10)
            return;
        //time out ,give up
        if (videos.length < transcripts.length && counter == 10) {
            clearInterval(id);
            console.error('视频地址加载失败');
        }
        clearInterval(id);
        console.log(`video ready, clear Interval\nvideos:${videos.length}\ntranscripts:${transcripts.length}`);
        console.log(videos);
        console.log(transcripts);
        //get_section_index
        const current_chapter = $('.chapter.is-open');
        const current_section = current_chapter.find('li.active');
        const section_index = ($('.chapter').index(current_chapter) + 1) + '.' + (current_section.parent().children().index(current_section) + 1);
        const video_title_node = $('#seq_content > div > div > div.vert.vert-0 > div > h2');
        //aria2 config settings
        let aria2_config = GM_getValue('aria2_config', default_aria2_config);
        console.log('current saved aria2_config:');
        console.log(aria2_config);
        $(`
            <div>
                <button id="configure_aria2_button" style="margin-bottom: 20px;">配置 aria2</button>
                <div id="configure_aria2_panel" style="display: none;margin: 0.5em;">
                    uri: <input id="aria2_uri" value="${aria2_config.uri}"><br>
                    token: <input id="aria2_token" value="${aria2_config.token}"><br>
                    storage_directory: <input id="aria2_storage_directory" value="${aria2_config.storage_directory}"><br>
                    <button id="save_aria2_settings_button" style="margin-bottom: 20px;">保存</button>
                </div>
            </div>
        `.replace(/ {4,}/g, '')).insertBefore(video_title_node);
        const configure_button = $('#configure_aria2_button');
        const configure_panel = $('#configure_aria2_panel');
        configure_button.on('click', () => {
            configure_panel.show();
            configure_button.css('color', 'black');
        });
        $('#save_aria2_settings_button').on('click', () => {
            aria2_config = {
                uri: $('#aria2_uri').val().toString(),
                token: $('#aria2_token').val().toString(),
                storage_directory: $('#aria2_storage_directory').val().toString()
            };
            GM_setValue('aria2_config', aria2_config);
            configure_button.css('color', 'green');
            configure_panel.hide();
            console.log('new aria2_config saved');
            console.log(aria2_config);
        });
        //build download buttons and actions
        videos.forEach((x, i) => {
            draw_buttons_and_links(x, get_file_name('mp4', videos.length > 1 ? (i + 1) : null), 'mp4');
        });
        transcripts.forEach((x, i) => {
            draw_buttons_and_links(x, get_file_name('srt', transcripts.length > 1 ? (i + 1) : null), 'srt');
        });
        function get_file_name(suffix, index) {
            return `${section_index} ${unit} ${section}${index ? ' ' + index : ''}.${suffix}`;
        }
        function draw_buttons_and_links(uri, file_name, suffix='') {
            $(` <div style="margin-top: 15px;margin-bottom: 15px;">
                    <p style="margin-bottom: 0.3em;">${file_name}</p>
                    <button class="BtnAria" style="margin-top: 10px;display: inline-block;">ARIA2 RPC</button>
                    <button class="BtnBatchdown" style="margin-top: 10px;display: inline-block;">批量下载</button>
                    <a href="${uri}" style="margin-top: 10px;display: inline-block;line-height: 55px;padding-left: 30px;" download="${file_name}">下载链接</a>
                </div>
            `.replace(/ {4,}/g, '')).on('click', 'button.BtnAria', (event) => {
                let success_button = event.target;
                console.log('success_button');
                send_aria2_download_rpc(uri, file_name, GM_getValue('aria2_config', default_aria2_config), (rpc_response) => {
                    try {
                        console.log('rpc sent successfully, server reply:');
                        console.log(rpc_response);
                        if (rpc_response.result) {
                            success_button.style.color = 'green';
                        }
                    }
                    catch (any) {
                        console.error(rpc_response);
                    }
                }, (json_rpc_sent, jqXHR, textStatus, errorThrown) => {
                    console.error('error in send_aria2_download_rpc');
                    console.log(aria2_config);
                    console.log(json_rpc_sent);
                    console.log(jqXHR);
                    alert('aria2 rpc 发送失败,请检查console日志');
                });
            }).on('click', 'button.BtnBatchdown', (event)=>{
                if(!$('#win_batchdown').length){
                    var chapters=$('div.chapter');
                    $(`<div id="win_batchdown" style="background-color:white;position:fixed;z-index:99999;top:1%;left:30%;max-width:40%;max-height:87.7%;overflow:auto">
                           <button onclick="$(this.parentNode).hide()">×</button> <button class="BtnBatchAria">发送到ARIA2</button>
                           <button onclick="$(this.parentNode).find('input').prop('checked',true)" class=all>全选</button>
                           <button onclick="$(this.parentNode).find('input').click()" class=reverse>反选</button><br>
                           <input checked type=checkbox class=mp4>mp4 <input checked type=checkbox class=srt>srt<br>
                       </div>`).appendTo($('body')).on('click', '.BtnBatchAria', function(eve){
                        $('#win_batchdown').find('a').each(function(i,e){
                            if(e.previousElementSibling.checked){
                                e = $(e);
                                $.get(e.attr('href'), function(res){
                                    $($(res).find('div.seq_contents').text()).find('div.video').each(function(i,d){
                                        d = $(d);
                                        if($('input.mp4').prop('checked'))
                                            $.getJSON('http://www.xuetangx.com/videoid2source/' + d.attr('data-ccsource'),function(json){
                                                send_aria2_download_rpc(json.sources.quality20[0],
                                                                        (e.text()+' '+i+d.siblings('h2').text()).replace(':',':')+'.mp4',
                                                                        GM_getValue('aria2_config', default_aria2_config),
                                                                        (res)=>{console.log(res)},(err)=>{console.error(err)});
                                            });
                                        if($('input.srt').prop('checked'))
                                            send_aria2_download_rpc('http://www.xuetangx.com'+d.find('li.video-tracks.video-download-button>a').attr('href'),
                                                                    (e.text()+' '+i+d.siblings('h2').text()).replace(':',':')+'.srt',
                                                                    GM_getValue('aria2_config', default_aria2_config),
                                                                    (res)=>{console.log(res)},(err)=>{console.error(err)});
                                    });
                                },'html');
                            }
                        })
                    });
                    chapters.each(function(i, e){//每个章节
                        var filename, iChapter = i+1, chname = $(e).find('h3>a').text();
                        $(e).find('li>a').each(function(i, e){//每节课程
                            var iLesson = i+1;
                            e = $(e);
                            filename = `${iChapter}.${iLesson} ${chname} ${e.find('p')[0].firstChild.textContent.trim()}`;
                            $(`<input type=checkbox><a target=_blank href="${e.attr('href')}">${filename}</a><br/>`).appendTo($('#win_batchdown'));
                        });
                    });
                }
                else
                    $('#win_batchdown').show();
            }).insertBefore(video_title_node);
        }
    }
    catch (any) {
        console.log(any);
        clearInterval(id);
    }
}, 1000);
//send json-rpc(with cookie) to aria2
function send_aria2_download_rpc(uri, filename, aria2_config, success_callback, error_callback) {
    const json_rpc = {
        id: '',
        jsonrpc: '2.0',
        method: 'aria2.addUri',
        params: [
            `token:${aria2_config.token}`,
            [uri],
            {
                dir: aria2_config.storage_directory,
                out: filename,
                header: ['Cookie: ' + document.cookie]
            }
        ]
    };
    $.ajax({
        url: aria2_config.uri,
        //method:'POST' can only be used after jQuery 2.0
        type: 'POST',
        crossDomain: true,
        processData: false,
        data: JSON.stringify(json_rpc),
        contentType: 'application/json',
        success: success_callback,
        error: (jqXHR, textStatus, errorThrown) => {
            error_callback(json_rpc, jqXHR, textStatus, errorThrown);
        },
    });
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.user.js","sourceRoot":"","sources":["src/index.ts"],"names":[],"mappings":"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","sourcesContent":["// ==UserScript==\n// @name         学堂在线 课程视频及字幕 aria2下载脚本\n// @namespace    https://github.com/ShenHongFei/\n// @version      2017.8.27.19.10\n// @description  1.提取视频和字幕的链接<br>2.按照上下文自动生成文件名<br>3.点击按钮通过 JOSN-RPC 调用 aria2 下载至指定文件夹\n// @author       沈鸿飞\n// @homepageURL  \n// @updateURL    \n// @match        http://www.xuetangx.com/courses/**\n// @match        https://www.xuetangx.com/courses/**\n// @icon         https://github.com/ShenHongFei/xuetangx-aria2-download-userscript/favicon.png\n// @license      MIT License\n// @run-at       document-idle\n// @grant        GM_getValue\n// @grant        GM_setValue\n// @_require      file://E:\\SDK\\xuetangx-aria2-download-userscript\\index.js\n// ==/UserScript==\nclass Aria2Config{\n    uri:string\n    token:string\n    storage_directory:string\n}\n\nconst default_aria2_config:Aria2Config={\n    uri:'http://localhost:6800/jsonrpc',\n    token:'',\n    storage_directory:'D:\\\\\\\\'\n}\ndeclare function GM_getValue(name:string,defaultValue:any):any\ndeclare function GM_setValue(name:string,value:any):any\n\nlet counter=0\nconst id=setInterval(()=>{\n    try{\n        const unit=$('.active.ui-accordion-header')[0].textContent.trim().replace(' ',' ')\n        const section=$('#accordion li.active p:first')[0].firstChild.textContent.trim().replace(' ',' ')\n        const videos=<string[]><any>$('.xt_video_player').map((i,x)=>x.getAttribute('src')).get()\n        const transcripts=<string[]><any>$('a[href$=\"transcript/download\"]').map((i,x)=>'http://www.xuetangx.com'+x.getAttribute('href')).get()\n        \n        counter++\n        //loop and wait for the next turn\n        if(videos.length<transcripts.length&&counter<10) return\n        //time out ,give up\n        if(videos.length<transcripts.length&&counter==10){\n            clearInterval(id)\n            console.error('视频地址加载失败')\n        }\n        clearInterval(id)\n        console.log(`video ready, clear Interval\\nvideos:${videos.length}\\ntranscripts:${transcripts.length}`)\n        console.log(videos)\n        console.log(transcripts)\n        //get_section_index\n        const current_chapter=$('.chapter.is-open')\n        const current_section=current_chapter.find('li.active')\n        const section_index:string=($('.chapter').index(current_chapter)+1)+'.'+(current_section.parent().children().index(current_section)+1)\n        \n        const video_title_node=$('#seq_content > div > div > div.vert.vert-0 > div > h2')\n        \n        //aria2 config settings\n        let aria2_config:Aria2Config=GM_getValue('aria2_config',default_aria2_config)\n        console.log('current saved aria2_config:')\n        console.log(aria2_config)\n        $(`\n            <div>\n                <button id=\"configure_aria2_button\" style=\"margin-bottom: 20px;\">配置 aria2</button>\n                <div id=\"configure_aria2_panel\" style=\"display: none;margin: 0.5em;\">\n                    uri: <input id=\"aria2_uri\" value=\"${aria2_config.uri}\"><br>\n                    token: <input id=\"aria2_token\" value=\"${aria2_config.token}\"><br>\n                    storage_directory: <input id=\"aria2_storage_directory\" value=\"${aria2_config.storage_directory}\"><br>\n                    <button id=\"save_aria2_settings_button\" style=\"margin-bottom: 20px;\">保存</button>\n                </div>\n            </div>\n        `.replace(/ {4,}/g,'')\n        ).insertBefore(video_title_node)\n        const configure_button=$('#configure_aria2_button')\n        const configure_panel=$('#configure_aria2_panel')\n        configure_button.on('click',()=>{\n            configure_panel.show()\n            configure_button.css('color','black')\n        })\n        $('#save_aria2_settings_button').on('click',()=>{\n            aria2_config={\n                uri:$('#aria2_uri').val().toString(),\n                token:$('#aria2_token').val().toString(),\n                storage_directory:$('#aria2_storage_directory').val().toString()\n            }\n            GM_setValue('aria2_config',aria2_config)\n            configure_button.css('color','green')\n            configure_panel.hide()\n            console.log('new aria2_config saved')\n            console.log(aria2_config)\n        })\n        //build download buttons and actions\n        videos.forEach((x,i)=>{\n            draw_buttons_and_links(x,get_file_name('mp4',videos.length>1?(i+1):null))\n        })\n        transcripts.forEach((x,i)=>{\n            draw_buttons_and_links(x,get_file_name('srt',transcripts.length>1?(i+1):null))\n        })\n        \n        function get_file_name(suffix:string,index?:number){\n            return `${section_index} ${unit} ${section}${index?' '+index:''}.${suffix}`\n        }\n        function draw_buttons_and_links(uri:string,file_name:string):void{\n            $(`\n                <div style=\"margin-top: 15px;margin-bottom: 15px;\">\n                    <p style=\"margin-bottom: 0.3em;\">${file_name}</p>\n                    <button style=\"margin-top: 10px;display: inline-block;\">ARIA2 RPC</button>\n                    <a href=\"${uri}\" style=\"margin-top: 10px;display: inline-block;line-height: 55px;padding-left: 30px;\" download=\"${file_name}\">下载链接</a>\n                </div>\n            `.replace(/ {4,}/g,''))\n            .on('click','button',(event)=>{\n                let success_button=event.target\n                console.log('success_button')\n                send_aria2_download_rpc(\n                    uri,\n                    file_name,\n                    GM_getValue('aria2_config',default_aria2_config),\n                    (rpc_response)=>{\n                        try{\n                            console.log('rpc sent successfully, server reply:')\n                            console.log(rpc_response)\n                            if(rpc_response.result){\n                                success_button.style.color='green'\n                            }\n                        }catch(any){console.error(rpc_response)}\n                    },\n                    (json_rpc_sent,jqXHR:JQueryXHR,textStatus:string,errorThrown:string)=>{\n                        console.error('error in send_aria2_download_rpc')\n                        console.log(aria2_config)\n                        console.log(json_rpc_sent)\n                        console.log(jqXHR)\n                        alert('aria2 rpc 发送失败，请检查console日志')\n                    })\n            })\n            .insertBefore(video_title_node)\n        }\n    }catch(any){\n        console.log(any)\n        clearInterval(id)\n    }\n},1000)\n\n//send json-rpc(with cookie) to aria2\nfunction send_aria2_download_rpc(\n    uri:string,\n    filename:string,\n    aria2_config:Aria2Config,\n    success_callback?:(rpc_response:any)=>void,\n    error_callback?:(json_rpc_sent:object,jqXHR:JQueryXHR,textStatus:string,errorThrown:string)=>void\n){\n    const json_rpc={\n        id:'',\n        jsonrpc:'2.0',\n        method:'aria2.addUri',\n        params:[\n            `token:${aria2_config.token}`,\n            [uri],\n            {\n                dir:aria2_config.storage_directory,\n                out:filename,\n                header:['Cookie: '+document.cookie]\n            }\n        ]\n    }\n    $.ajax({\n    url:aria2_config.uri,\n    //method:'POST' can only be used after jQuery 2.0\n    type:'POST',\n    crossDomain:true,\n    processData:false,\n    data:JSON.stringify(json_rpc),\n    contentType:'application/json',\n    success:success_callback,\n    error:(jqXHR:JQueryXHR,textStatus:string,errorThrown:string)=>{\n        error_callback(json_rpc,jqXHR,textStatus,errorThrown)\n    },\n})\n}"]}