// ==UserScript==
// @name 慕课网 下载视频(失效)
// @namespace https://github.com/Ahaochan/Tampermonkey
// @version 0.2.7
// @description 获取视频下载链接,使用方法:进入任意课程点击下载即可。如http://www.imooc.com/learn/814。慕课网已废弃v1和v2接口, 全面启用HLS, 此脚本失效, 详情看脚本内说明。github:https://github.com/Ahaochan/Tampermonkey,欢迎star和fork。
// @author Ahaochan
// @match http://www.imooc.com/learn/*
// @match https://www.imooc.com/learn/*
// @grant GM_xmlhttpRequest
// @grant GM_setClipboard
// @require http://code.jquery.com/jquery-1.11.0.min.js
// ==/UserScript==
(function () {
'use strict';
// 最新视频下载方法
// 例如下载http://www.imooc.com/video/14351,点击F12,点击Network,筛选XHR,找到medium.hxk。复制神秘代码58c65fc3e520e5677f8b457a。
// 下载地址就是http://v3.mukewang.com/584e2423b3fee3bb558b7896/H.mp4。
// 估计是通过http://www.imooc.com/course/14351/medium.m3u8?cdn=aliyun返回的数据,通过某种解密方式获得的神秘代码
// 思路在这,个人水平不够,修复不了,有能力希望能fork并pull request一下。。。
// 2017年10月3日
// 现在慕课网已经把http://v1.mukewang.com和http://v2.mukewang.com的DNS解析停掉了。
// 看来已经全面启用HLS传输视频流。可以看到http://m.imooc.com/course/3725/high.m3u8?cdn=aliyun应该就是获取m3u8文件的链接。
// 而且慕课网对m3u8文件进行了加密, 奇怪的是, 每次刷新的加密后m3u8文件字符串都不一样, 但是请求的ts文件是一样的url。不知道慕课网是怎么做到的。
// 上面的解决方案也只能解决一部分视频的下载。
// 还是本人技术不够, 所以放弃此脚本的维护。
/**--------------------------------获取下载链接---------------------------------------------*/
var videoes = [];
var $medias = $('.mod-chapters').find('a.J-media-item');
var total = $medias.length;
var len = total;
//添加提示标签
$('.course-menu').append($('<li><a href="javascript:void(0)"><span id="downTip">慕课网下载脚本已失效</span></a></li>'));
return; // 中止此脚本运行
if (!isLogin) {
$('#downTip').text('视频下载异常,点击进行登录')
.click(function () {
var clickEvent = document.createEvent('MouseEvents');
clickEvent.initEvent('click', true, true);
document.getElementById('js-signin-btn').dispatchEvent(clickEvent);
});
return;
}
//遍历获取下载链接
$medias.each(function (key, value) {
var vid = $(this).parent().attr('data-media-id');
var name = $(this).text();
var pattern = /\(\d\d:\d\d\)/;
if (!pattern.test(name)) {
total--;
if (key === len - 1 && !total) {
$('#downTip').text('无视频下载');
}
return;
}
name = name.replace(/\(\d{2}:\d{2}\)/, '').replace(/\s/g, '');
v1(vid, name, $(this));
//v2(vid, name, $(this));
//v3(vid, name, $(this));
});
/**--------------------------------获取下载链接---------------------------------------------*/
/**--------------------------------视频下载解析接口-----------------------------------------*/
/** v1接口,强制转换为v1接口 */
function v1(vid, name, item) {
$.getJSON('/course/ajaxmediainfo/?mid=' + vid + '&mode=flash', function (response) {
console.log('加载数据:' + vid);
if (response.data.result.mpath instanceof Array) {
var url = response.data.result.mpath[0].replace('http://v2', 'http://v1');
parseVideo(vid, name, url, item);
} else {
$('#downTip').text('不支持新版视频, 若要下载请查看源码中的注释');
}
});
}
/** v2接口,只能解析v1,v2(已废弃) */
function v2(vid, name, item) {
$.getJSON('/course/ajaxmediainfo/?mid=' + vid + '&mode=flash', function (response) {
var url = response.data.result.mpath[0];
parseVideo(vid, name, url, item);
});
}
/** v3接口,解析v1,v2,v3(已废弃) */
function v3(vid, name, item) {
GM_xmlhttpRequest({
method: 'GET',
url: 'http://m.imooc.com/video/' + vid,
onload: function (response) {
var pattern = /(http.+mp4)/;
var url = response.responseText.match(pattern)[0];
parseVideo(vid, name, url, item);
}
});
}
/**--------------------------------视频下载解析接口-----------------------------------------*/
/**--------------------------------处理数据-------------------------------------------------*/
function parseVideo(vid, name, url, item) {
var urlL = url.replace('H.mp4', 'M.mp4').replace('M.mp4', 'L.mp4');
var urlM = url.replace('H.mp4', 'M.mp4').replace('L.mp4', 'M.mp4');
var urlH = url.replace('L.mp4', 'M.mp4').replace('M.mp4', 'H.mp4');
var video = {
vid: vid,
name: name,
url: [urlL, urlM, urlH]
};
videoes.push(video);
//添加单个下载链接
var $link = $('<a href="' + video.url[clarityType] + '" style="position:absolute;right:100px;top:0;" target="_blank">下载</a>');
$link.bind('DOMNodeInserted', function () {
$(this).find('i').remove();
});//移除子标签
item.after($link);
//添加全部下载链接
if (videoes.length === total) {
$('#downTip').text('视频下载(' + total + '),已复制到剪贴板');
videoes.sort(function (a, b) {
if (a.name > b.name) return 1;
else if (a.name < b.name) return -1;
else return 0;
});
//显示
$('#downloadBox').append($('<div style="margin-top:24px;">' +
'<div style="border:1px solid #b7bbbf;box-sizing:border-box;border-radius:2px;">' +
'<textarea id="downLoadArea" style="width:97%;min-height:100px;padding:8px;color:#555;resize:none;line-height:18px;"></textarea>' +
'</div>')
);
textAreaChange();
}
}
/**--------------------------------处理数据-------------------------------------------------*/
/**--------------------------------导出设置-------------------------------------------------*/
var clarityType = 2;
var outTextType = 'idm';
$('div.mod-tab-menu').after(
$('<div id="downloadBox" class="course-brief">' +
'<div style="float:left;margin-right:30px;">' +
'<h4 style="font-weight:700;font-size: 16px;marginTop:10px">下载清晰度 : </h4>' +
'<label for="lowClarity" >普清(L)</label><input type="radio" id="lowClarity" name="clarity" value="0" />' +
'<label for="middleClarity">高清(M)</label><input type="radio" id="middleClarity" name="clarity" value="1" />' +
'<label for="highClarity" >超清(H)</label><input type="radio" id="highClarity" name="clarity" value="2" checked="checked" />' +
'</div>' +
'<div>' +
'<h4 style="font-weight:700;font-size: 16px;marginTop:10px">导出格式 : </h4>' +
'<label for="rawOutText" >raw </label><input type="radio" id="rawOutText" name="outText" value="raw"/>' +
'<label for="idmOutText" >idm </label><input type="radio" id="idmOutText" name="outText" value="idm" checked="checked" />' +
'<label for="xmlOutText" >xml </label><input type="radio" id="xmlOutText" name="outText" value="xml" />' +
'<label for="jsomOutText">json</label><input type="radio" id="jsomOutText" name="outText" value="json"/><br/>' +
'</div>' +
'</div>')
);
$('input:radio').css('margin', 'auto 50px auto 3px');//设置单选框
$('input:radio[name=clarity]').change(function () {
clarityType = this.value;
textAreaChange();
});
$('input:radio[name=outText]').change(function () {
outTextType = this.value;
textAreaChange();
});
function textAreaChange() {
var downloadTextArea = getTextLinks(clarityType, outTextType);
GM_setClipboard(downloadTextArea);
$('#downloadBox').find('textarea').text(downloadTextArea);
}
/**--------------------------------导出设置-------------------------------------------------*/
/**--------------------------------格式化下载链接用以显示---------------------------------*/
function getTextLinks(clarityType, outTextType) {
if (outTextType === 'json') return JSON.stringify(videoes);
else {
var str = '';
for (var i in videoes) {
if (outTextType === 'xml') {
str += '\t<video>\n\t\t<url>' + videoes[i].url[clarityType] + '</url>\n\t\t<name>' + videoes[i].name + '</name>\n\t</video>\n';
} else if (outTextType === 'raw') {
str += videoes[i].url[clarityType] + '\n';
} else {//idm
str += 'filename=' + videoes[i].name + '&fileurl=' + videoes[i].url[clarityType] + '\n';
}
}
if (outTextType === 'xml') str = '<?xml version="1.0" encoding="utf-8" ?>\n<videoes>\n' + str + '</videoes>';
return str;
}
}
/**--------------------------------格式化下载链接用以显示---------------------------------*/
})();