// ==UserScript==
// @name 网易云下载
// @description 歌单里好多歌用以前的脚本都404了,于是简单修改了下。
// @version 20180616
// @author 糖果君
// @include http*://music.163.com/*
// @grant unsafeWindow
// @grant GM_xmlhttpRequest
// @grant GM.xmlHttpRequest
// @require https://greasyfork.org/scripts/26727-网易云音乐歌曲封面下载/code/网易云音乐歌曲封面下载.user.js
// @require https://greasyfork.org/scripts/34555-greasemonkey-4-polyfills/code/Greasemonkey%204%20Polyfills.js
// @namespace
// ==/UserScript==
//参考 https://greasyfork.org/zh-CN/scripts/10548-网易云音乐下载
//以及更早的 https://greasyfork.org/zh-CN/scripts/1099-netease-music-download
var window=unsafeWindow||window.wrappedJSObject;
var api = {
//参考 https://greasyfork.org/zh-CN/scripts/10582-网易云音乐高音质支持
getTrackURL: function(dfsId) {
var byte1 = '3go8&$8*3*3h0k(2)2';
var byte2 = dfsId + '';
var byte3 = [];
for (var i = 0; i < byte2.length; i++) {
byte3[i] = byte2.charCodeAt(i) ^ byte1.charCodeAt(i % byte1.length);
}
byte3 = byte3.map(function(i) {
return String.fromCharCode(i);
}).join('');
var results = window.CryptoJS.MD5(byte3).toString(window.CryptoJS.enc.Base64).replace(/\//g, '_').replace(/\+/g, '-');
var url = 'http://p2.music.126.net/' + results + '/' + byte2;
return url;
},
autoSign: function (type, cookie) {
if (document.cookie.indexOf(cookie) == - 1) {
api.sign(type, function (result) {
if (result.code == - 2 || result.code == 200) {
var cookieime = new Date();
cookieime.setTime(24 * 60 * 60 * 1000 + new Date(new Date() .toDateString()) .getTime() - 1);
document.cookie = cookie+ ';expires=' + cookieime.toGMTString();
}
});
};
},
encrypt_request: function(callback, url, data) {
// 这个好像不重要。var token = window.document.cookie.split('__csrf') [1].split(';') [0].substring(1);
var token = '';
data.csrf_token = token;
var req = new XMLHttpRequest();
req.open('POST', window.location.origin+url+ token, true);
req.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
req.setRequestHeader('X-Real-IP', '27.38.4.87');
//解决海外问题。
req.onload = function() {
callback(JSON.parse(this.responseText));
};
//参考 https://github.com/darknessomi/musicbox/wiki/网易云音乐新版WebAPI分析。
var pubKey = '010001';
var modulus = '00e0b509f6259df8642dbc35662901477df22677ec152b5ff68ace615bb7b725152b3ab17a876aea8a5aa76d2e417629ec4ee341f56135fccf695280104e0312ecbda92557c93870114af6c9d05c4f7f0c3685b7a46bee255932575cce10b424d813cfe4875d3e82047b97ddef52741d546b8e289dc6935b3ece0462db0a22b8e7';
var nonce = '0CoJUm6Qyw8W8jud';
var result = window.asrsea(JSON.stringify(data), pubKey, modulus, nonce);
req.send('params=' + encodeURIComponent(result.encText) + '&encSecKey=' + encodeURIComponent(result.encSecKey));
},
detail: function(songId, callback) {
var url = '/weapi/v3/song/detail';
var data = {
c:
JSON.stringify([{
id: songId
}
])
};
this.encrypt_request(callback, url, data);
},
album: function (albumId, callback) {
var url = '/api/album/' + albumId;
var data = {
};
this.encrypt_request(callback, url, data);
},
lrc: function(songId, callback) {
var url = '/weapi/song/lyric';
var data = {
id: songId,
lv: -1,
tv: -1
};
this.encrypt_request(callback, url, data);
},
newsong: function(songId, callback) {
var url = '/weapi/song/enhance/player/url';
var data = {
ids: [songId],
br: 999000,
};
this.encrypt_request(callback, url, data);
},
mv: function(mvId, callback) {
var url = '/weapi/mv/detail/';
var data = {
id: mvId,
};
this.encrypt_request(callback, url, data);
},
search: function(songinfo, callback) {
var url = '/weapi/search/pc';
var data = {
s: songinfo,
limit: 1,
type: 1,
offset: 0,
};
this.encrypt_request(callback, url, data);
},
sign: function (type, callback) {
var url = '/weapi/point/dailyTask';
var data = {
type: type
};
this.encrypt_request(callback, url, data);
},
xiamisearch: function (key, callback) {
GM.xmlHttpRequest({
method: 'GET',
url: 'http://music-api-jwzcyzizya.now.sh/api/search/song/xiami?&limit=1&page=1&key=' + key,
onload: function (response) {
callback(JSON.parse(response.responseText));
}
});
},
};
var innerFrame = document.querySelector('iframe');
var pages = [
{
url: 'music.163.com/#/song?id=',
handler: function() {
var songId = location.href.match(/id=([0-9]+)/)[1];
var downloadLine = this.createDownloadLine(songId);
var innerFrameDoc = innerFrame.contentWindow.document;
var albumNode = innerFrameDoc.querySelectorAll('p.des.s-fc4')[1];
var parentNode = albumNode.parentNode;
parentNode.insertBefore(downloadLine, albumNode.nextElementSibling);
},
createDownloadLine: function(songId) {
var disableStyle = function(link) {
link.text += '(无)';
link.style.color = 'gray';
link.style.textDecoration = 'none';
link.style.cursor = 'auto';
};
var setUrlAndSize = function (mp3Link, Music, albumId,songinfo) {
if (Music) {
var href = Music.url;
if (href) {
mp3Link.href = href;
mp3Link.text += (Music.size / 1024 / 1024) .toFixed(1) + 'M';
return ;
}
else {/*其实旧接口还是能用,带.mp3等后缀名会直接403 去掉后缀名就可以
api.album(albumId, function (result) {
if (result.album && result.album.songs)
{
var songs = result.album.songs;
for (var i = 0; i < songs.length; i++)
{
if (songs[i].id == songId) {
var song = songs[i];
var music = song.hMusic || song.mMusic || song.lMusic || song.bMusic;
var mp3url;
var dfsId = music.dfsId_str || music.dfsId;
if (music && dfsId != 0) {
mp3url = api.getTrackURL(dfsId);
mp3Link.text += (music.size / 1024 / 1024) .toFixed(1) + 'M(旧)';
} else if (!song.mp3Url.endsWith('==/0')) {
mp3url = song.mp3Url;
}
if (mp3url) {
mp3Link.href = mp3url;
}
else {*/
api.xiamisearch(songinfo, function (result) {
if (result.songList) {
console.log(result);
mp3Link.href = result.songList[0].file;
mp3Link.text += '(虾米)';
} else {
disableStyle(mp3Link);
}
});/*
}
}
}
}
});
*/}
}
};
var setLyric = function(LycLink, result) {
var LrC = '';
var lrc = result.lrc;
var tlrc = result.tlyric;
var num = 0;
if (lrc && lrc.lyric) {
LrC += lrc.lyric + '\n';
num+=1;}
if (tlrc && tlrc.lyric) {
LrC += tlrc.lyric;
num+=2;}
if (num != 0) {
var html = '';
switch (num) {
case 1:html="(原)";break;
case 2:html="(译)";break;
case 3:html="(合)";break;
}
LycLink.href = 'data:text/plain;charset=utf-8,' + encodeURIComponent(LrC);
LycLink.innerHTML += html;
} else {
disableStyle(LycLink);
}
};
var newMp3Link = this.createLink('歌曲');
var lyricLink = this.createLink('歌词');
var mvLink = this.createLink('mv');
var picLink = this.createLink('封面');
api.detail(songId, function (result) {
var song = result.songs[0];
if (song.mv) {
api.mv(song.mv, function (result) {
var mv = result.data.brs;
mvLink.href = mv[720] || mv[480] || mv[240];
});
}
else {
disableStyle(mvLink);
};
if (song.al.pic_str || song.al.pic) {
var img = innerFrame.contentWindow.document.querySelector(".j-img");
picLink.href = img.dataset["src"] = api.getTrackURL(song.al.pic_str || song.al.pic);
img.src = img.dataset["src"] + "?param=130y130";
} else {
disableStyle(picLink);
};
var alia = song.alia[0] ? '(' + song.alia + ')' : '';
var ars = '-';
for (var i = 0; i < song.ar.length; i++) {
ars += (song.ar) [i].name + '/';
}
var songinfo = song.name + alia + ars;
var alid = song.al.id;
api.newsong(songId, function (result) {
var song = result.data[0];
setUrlAndSize(newMp3Link, song, alid,songinfo);
});
});
api.lrc(songId, function (result) {
setLyric(lyricLink, result);
});
var container = this.createLineContainer('下载');
container.appendChild(newMp3Link);
container.appendChild(lyricLink);
container.appendChild(mvLink);
container.appendChild(picLink);
return container;
},
createLink: function(label) {
var link = document.createElement('a');
link.innerHTML = label;
link.className = 's-fc7';
link.style.marginRight = '10px';
link.href = 'javascript:void(0);';
link.target = '_blank';
return link;
},
createLineContainer: function(label) {
var container = document.createElement('p');
container.className = 'desc s-fc4';
container.innerHTML = label + ':';
container.style.margin = '10px 0';
return container;
},
},
];
if (innerFrame) {
innerFrame.addEventListener('load', function () {
var i, page;
for (i = 0; i < pages.length; i += 1) {
page = pages[i];
if (location.href.indexOf(page.url) != -1) {
page.handler();
}
}
});
}
document.cookie = 'os=ios';
api.autoSign(0,'appsign=true');
api.autoSign(1,'websign=true');