// ==UserScript== // @name 网易云音乐:云盘快传(含周杰伦)|歌曲下载&转存云盘|云盘匹配纠正|听歌量打卡|高音质试听 // @description 无需文件云盘快传歌曲(含周杰伦)、歌曲下载&转存云盘(可批量)、云盘匹配纠正、快速完成300首听歌量打卡任务、选择更高音质试听(支持超清母带,默认无损)、歌单歌曲排序(时间、红心数、评论数)、限免VIP歌曲下载上传、云盘音质提升、本地文件上传云盘、云盘导入导出。 // @namespace https://github.com/Cinvin/myuserscripts // @version 3.6.0 // @author cinvin // @license MIT // @match https://music.163.com/* // @icon  // @grant GM_xmlhttpRequest // @grant GM_download // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant unsafeWindow // @require https://fastly.jsdelivr.net/npm/sweetalert2@11.7.32/dist/sweetalert2.all.min.js // @require https://fastly.jsdelivr.net/npm/ajax-hook@3.0.3/dist/ajaxhook.min.js // @require https://fastly.jsdelivr.net/npm/jsmediatags@3.9.7/dist/jsmediatags.min.js // @connect 45.127.129.8 // @connect 126.net // @downloadURL none // ==/UserScript== (function () { 'use strict'; const appendCookie = 'os=android;appver=9.1.10' const baseCDNURL = 'https://fastly.jsdelivr.net/gh/Cinvin/cdn@latest/artist/' //svip:超清母带,沉浸环绕声,vip:高清环绕声,Hi-Res,无损 const levelOptions = { jymaster: '超清母带', sky: '沉浸环绕声', jyeffect: '高清环绕声', hires: 'Hi-Res', lossless: '无损', exhigh: '极高', higher: '较高', standard: '标准' } const levelWeight = { jymaster: 8, sky: 7, jyeffect: 6, hires: 5, lossless: 4, exhigh: 3, higher: 2, standard: 1, none: 0 } const defaultOfDEFAULT_LEVEL = 'jymaster' const uploadChunkSize = 8 * 1024 * 1024 const player = unsafeWindow.top.player const songMark = { explicit: 1048576 } const extractLrcRegex = /^(?(?:\[.+?\])+)(?!\[)(?.+)$/gm const extractTimestampRegex = /\[(?\d+):(?\d+)(?:\.|:)*(?\d+)*\]/g function getcookie(key) { var cookies = document.cookie, text = "\\b" + key + "=", find = cookies.search(text); if (find < 0) { return "" } find += text.length - 2; var index = cookies.indexOf(";", find); if (index < 0) { index = cookies.length } return cookies.substring(find, index) || "" }; function weapiRequest(url, config) { let data = config.data || {} data.csrf_token = getcookie("__csrf"); const encRes = unsafeWindow.asrsea( JSON.stringify(data), "010001", "00e0b509f6259df8642dbc35662901477df22677ec152b5ff68ace615bb7b725152b3ab17a876aea8a5aa76d2e417629ec4ee341f56135fccf695280104e0312ecbda92557c93870114af6c9d05c4f7f0c3685b7a46bee255932575cce10b424d813cfe4875d3e82047b97ddef52741d546b8e289dc6935b3ece0462db0a22b8e7", "0CoJUm6Qyw8W8jud"); const details = { url: url.replace("api", "weapi") + `?csrf_token=${data.csrf_token}`, method: "POST", responseType: "json", headers: { "content-type": "application/x-www-form-urlencoded", }, cookie: config.cookie || appendCookie, data: `params=${encodeURIComponent(encRes.encText)}&encSecKey=${encodeURIComponent(encRes.encSecKey)}`, onload: res => { config.onload(res.response) }, onerror: config.onerror, } GM_xmlhttpRequest(details) } function showConfirmBox(msg) { Swal.fire({ title: '提示', text: msg, confirmButtonText: '确定', }) } function showTips(tip, type) { //type:1 √ 2:! unsafeWindow.g_showTipCard({ tip: tip, type: type }) } function fileSizeDesc(fileSize) { if (fileSize < 1024) { return fileSize + 'B' } else if (fileSize >= 1024 && fileSize < Math.pow(1024, 2)) { return (fileSize / 1024) .toFixed(1) .toString() + 'K' } else if (fileSize >= Math.pow(1024, 2) && fileSize < Math.pow(1024, 3)) { return (fileSize / Math.pow(1024, 2)) .toFixed(1) .toString() + 'M'; } else if (fileSize > Math.pow(1024, 3) && fileSize < Math.pow(1024, 4)) { return (fileSize / Math.pow(1024, 3)) .toFixed(2) .toString() + 'G'; } else if (fileSize > Math.pow(1024, 4)) { return (fileSize / Math.pow(1024, 4)) .toFixed(2) .toString() + 'T'; } }; function duringTimeDesc(dt) { let secondTotal = Math.floor(dt / 1000) let min = Math.floor(secondTotal / 60) let sec = secondTotal % 60 return min.toString().padStart(2, '0') + ':' + sec.toString().padStart(2, '0') }; function levelDesc(level) { return levelOptions[level] || level }; function sleep(millisec) { return new Promise(resolve => setTimeout(resolve, millisec)); }; function getArtistTextInSongDetail(song) { let artist = '' if (song.ar && song.ar[0].name && song.ar[0].name.length > 0) { artist = song.ar.map(ar => ar.name).join() } else if (song.pc && song.pc.ar && song.pc.ar.length > 0) { artist = song.pc.ar } return artist } function getAlbumTextInSongDetail(song) { let album = '' if (song.al && song.al.name && song.al.name.length > 0) { album = song.al.name } else if (song.pc && song.pc.alb && song.pc.alb.length > 0) { album = song.pc.alb } return album } function nameFileWithoutExt(title, artist, out) { if (out == 'title' || !artist || artist.length == 0) { return title } if (out == 'artist-title') { return `${artist} - ${title}` } if (out == 'title-artist') { return `${title} - ${artist}` } } function saveContentAsFile(content, fileName) { let data = new Blob([content], { type: 'type/plain' }) let fileurl = URL.createObjectURL(data) GM_download({ url: fileurl, name: fileName, onload: function () { URL.revokeObjectURL(data) }, onerror: function (e) { console.error(e) showTips(`下载失败,请尝试将 .${fileName.split('.').pop()} 格式加入 文件扩展名白名单`, 2) } }) } function createBigButton(desc, parent, appendWay) { let btn = document.createElement('a') btn.className = 'u-btn2 u-btn2-1' let btnDesc = document.createElement('i') btnDesc.innerHTML = desc btn.appendChild(btnDesc) btn.style.margin = '5px'; if (appendWay == 1) { parent.appendChild(btn) } else { parent.insertBefore(btn, parent.lastChild) } return btn } function combineLyric(lyricOri, lyricAdd) { let resLyric = { lyric: '', parsedLyric: lyricOri.parsedLyric.slice(0), } for (const parsedAddLyric of lyricAdd.parsedLyric) { resLyric.parsedLyric.splice(parsedLyricsBinarySearch(parsedAddLyric, resLyric.parsedLyric), 0, parsedAddLyric); } resLyric.lyric = resLyric.parsedLyric.map(lyric => lyric.line).join('\n') return resLyric } function parseLyric(lrc) { const parsedLyrics = []; for (const line of lrc.trim().matchAll(extractLrcRegex)) { const { lyricTimestamps, content } = line.groups; for (const timestamp of lyricTimestamps.matchAll(extractTimestampRegex)) { const { min, sec, ms } = timestamp.groups; const rawTime = timestamp[0]; const time = Number(min) * 60 + Number(sec) + Number(ms ?? 0) * 0.001; const parsedLyric = { rawTime, time, content: trimLyricContent(content), line: line[0] }; parsedLyrics.splice(parsedLyricsBinarySearch(parsedLyric, parsedLyrics), 0, parsedLyric); } } return parsedLyrics; } function parsedLyricsBinarySearch(lyric, lyrics) { let time = lyric.time; let low = 0; let high = lyrics.length - 1; while (low <= high) { const mid = Math.floor((low + high) / 2); const midTime = lyrics[mid].time; if (midTime === time) { return mid; } else if (midTime < time) { low = mid + 1; } else { high = mid - 1; } } return low; }; function trimLyricContent(content) { let t = content.trim(); return t.length < 1 ? content : t; } function handleLyric(lyricRes) { let LyricObj = { orilrc: { lyric: lyricRes.lrc.lyric, parsedLyric: parseLyric(lyricRes.lrc.lyric), }, romalrc: { lyric: lyricRes.romalrc.lyric, parsedLyric: parseLyric(lyricRes.romalrc.lyric), }, tlyriclrc: { lyric: lyricRes.tlyric.lyric, parsedLyric: parseLyric(lyricRes.tlyric.lyric), }, } if (LyricObj.orilrc.parsedLyric.length > 0 && LyricObj.tlyriclrc.parsedLyric.length > 0) { LyricObj.oritlrc = combineLyric(LyricObj.tlyriclrc, LyricObj.orilrc) } if (LyricObj.orilrc.parsedLyric.length > 0 && LyricObj.romalrc.parsedLyric.length > 0) { LyricObj.oriromalrc = combineLyric(LyricObj.orilrc, LyricObj.romalrc) } return LyricObj } //下载后上传 class ncmDownUpload { constructor(songs, showfinishBox = true, onSongDUSuccess = null, onSongDUFail = null, out = 'artist-title') { this.songs = songs this.currentIndex = 0 this.failSongs = [] this.out = out this.showfinishBox = showfinishBox this.onSongDUSuccess = onSongDUSuccess this.onSongDUFail = onSongDUFail }; startUpload() { this.currentIndex = 0 this.failSongs = [] if (this.songs.length > 0) { this.uploadSong(this.songs[0]) } } uploadSong(song) { try { weapiRequest(song.api.url, { data: song.api.data, onload: (content) => { showTips(`(1/6)${song.title} 获取文件信息完成`, 1) //console.log(content) let resData = content.data[0] || content.data if (resData.url != null) { song.fileFullName = nameFileWithoutExt(song.title, song.artist, this.out) + '.' + resData.type.toLowerCase() song.dlUrl = resData.url song.md5 = resData.md5 song.size = resData.size song.ext = resData.type.toLowerCase() song.bitrate = Math.floor(resData.br / 1000) //是否直接import let songCheckData = [{ md5: song.md5, songId: song.id, bitrate: song.bitrate, fileSize: song.size, }] weapiRequest("/api/cloud/upload/check/v2", { data: { uploadType: 0, songs: JSON.stringify(songCheckData), }, onload: (res1) => { console.log(song.title, '1.检查资源', res1) if (res1.code != 200 || res1.data.length < 1) { this.uploadSongFail(song) return } showTips(`(2/6)${song.title} 检查资源`, 1) song.cloudId = res1.data[0].songId if (res1.data[0].upload == 1) { this.uploadSongWay1Part1(song) } else if (res1.data[0].upload == 2) { this.uploadSongWay2Part1(song) } else { this.uploadSongWay3Part1(song) } }, onerror: (res) => { console.error(song.title, '1.检查资源', res) this.uploadSongFail(song) } }) } else { this.uploadSongFail(song) } }, onerror: (res) => { console.error(song.title, '0.获取URL', res) this.uploadSongFail(song) } }) } catch (e) { console.error(e); this.uploadSongFail(song) } } uploadSongWay1Part1(song) { let importSongData = [{ songId: song.cloudId, bitrate: song.bitrate, song: nameFileWithoutExt(song.title, song.artist, this.out), artist: song.artist, album: song.album, fileName: song.fileFullName }] //step2 导入歌曲 try { weapiRequest("/api/cloud/user/song/import", { data: { uploadType: 0, songs: JSON.stringify(importSongData), }, onload: (res) => { console.log(song.title, '2.导入文件', res) if (res.code != 200 || res.data.successSongs.length < 1) { console.error(song.title, '2.导入文件', res) this.uploadSongFail(song) return } showTips(`(2/6)${song.title} 2.导入文件完成`, 1) song.cloudSongId = res.data.successSongs[0].song.songId this.uploadSongMatch(song) }, onerror: (responses2) => { console.error(song.title, '2.导入歌曲', responses2) this.uploadSongFail(song) } }) } catch (e) { console.error(e); this.uploadSongFail(song) } } uploadSongWay2Part1(song) { try { weapiRequest("/api/nos/token/alloc", { data: { filename: song.fileFullName, length: song.size, ext: song.ext, type: 'audio', bucket: 'jd-musicrep-privatecloud-audio-public', local: false, nos_product: 3, md5: song.md5 }, onload: (res2) => { if (res2.code != 200) { console.error(song.title, '2.获取令牌', res2) this.uploadSongFail(song) return } song.resourceId = res2.result.resourceId showTips(`(3/6)${song.title} 获取令牌完成`, 1) console.log(song.title, '2.获取令牌', res2) showTips(`(3/6)${song.title} 开始上传文件`, 1) this.uploadSongPart2(song) }, onerror: (res) => { console.error(song.title, '2.获取令牌', res) this.uploadSongFail(song) } }) } catch (e) { console.error(e); this.uploadSongFail(song) } } uploadSongPart2(song) { //上传文件 showTips(`(3.1/6)${song.title} 开始下载文件`, 1) try { GM_xmlhttpRequest({ method: "GET", url: song.dlUrl, headers: { "Content-Type": "audio/mpeg" }, responseType: "blob", onload: (response) => { showTips(`(3.2/6)${song.title} 文件下载完成`, 1) let buffer = response.response //step2 上传令牌 weapiRequest("/api/nos/token/alloc", { data: { filename: song.fileFullName, length: song.size, ext: song.ext, type: 'audio', bucket: 'jd-musicrep-privatecloud-audio-public', local: false, nos_product: 3, md5: song.md5 }, onload: (tokenRes) => { song.token = tokenRes.result.token song.objectKey = tokenRes.result.objectKey console.log(song.title, '2.2.开始上传', tokenRes) showTips(`(3.3/6)${song.title} 开始上传文件`, 1) this.uploadFile(buffer, song, 0) }, onerror: (responses2) => { console.error(song.title, '2.1.获取令牌', responses2) this.uploadSongFail(song) } }) } }) } catch (e) { console.error(e); this.uploadSongFail(song) } } uploadFile(data, song, offset, context = null) { let complete = offset + uploadChunkSize > song.size let url = `http://45.127.129.8/jd-musicrep-privatecloud-audio-public/${encodeURIComponent(song.objectKey)}?offset=${offset}&complete=${String(complete)}&version=1.0` if (context) url += `&context=${context}` GM_xmlhttpRequest({ method: "POST", url: url, headers: { 'x-nos-token': song.token, 'Content-MD5': song.md5, 'Content-Type': 'audio/mpeg', }, data: data.slice(offset, offset + uploadChunkSize), onload: (response3) => { let res = JSON.parse(response3.response) if (complete) { console.log(song.title, '2.5.上传文件完成', res) showTips(`(4/6)${song.title} 上传文件完成`, 1) this.uploadSongPart3(song) } else { showTips(`(4/6)${song.title} 正在上传${fileSizeDesc(res.offset)}/${fileSizeDesc(song.size)}`, 1) this.uploadFile(data, song, res.offset, res.context) } }, onerror: (response3) => { console.error(song.title, '文件上传时失败', response3) this.uploadSongFail(song) }, }); } uploadSongWay3Part1(song) { try { weapiRequest("/api/nos/token/alloc", { data: { filename: song.fileFullName, length: song.size, ext: song.ext, type: 'audio', bucket: 'jd-musicrep-privatecloud-audio-public', local: false, nos_product: 3, md5: song.md5 }, onload: (res2) => { if (res2.code != 200) { console.error(song.title, '2.获取令牌', res2) this.uploadSongFail(song) return } song.resourceId = res2.result.resourceId showTips(`(3/6)${song.title} 获取令牌完成`, 1) console.log(song.title, '2.获取令牌', res2) this.uploadSongPart3(song) }, onerror: (res) => { console.error(song.title, '2.获取令牌', res) this.uploadSongFail(song) } }) } catch (e) { console.error(e); this.uploadSongFail(song) } } uploadSongPart3(song) { //step3 提交 try { console.log(song) weapiRequest("/api/upload/cloud/info/v2", { data: { md5: song.md5, songid: song.cloudId, filename: song.fileFullName, song: song.title, album: song.album, artist: song.artist, bitrate: String(song.bitrate), resourceId: song.resourceId, }, onload: (res3) => { if (res3.code != 200) { if (song.expireTime < Date.now() || (res3.msg && res3.msg.includes('rep create failed'))) { console.error(song.title, '3.提交文件', res3) this.uploadSongFail(song) } else { console.log(song.title, '3.正在转码', res3) showTips(`(5/6)${song.title} 正在转码...`, 1) sleep(1000).then(() => { this.uploadSongPart3(song) }) } return } console.log(song.title, '3.提交文件', res3) showTips(`(5/6)${song.title} 提交文件完成`, 1) //step4 发布 weapiRequest("/api/cloud/pub/v2", { data: { songid: res3.songId, }, onload: (res4) => { if (res4.code != 200 && res4.code != 201) { console.error(song.title, '4.发布资源', res4) this.uploadSongFail(song) return } console.log(song.title, '4.发布资源', res4) showTips(`(5/6)${song.title} 提交文件完成`, 1) song.cloudSongId = res4.privateCloud.songId this.uploadSongMatch(song) }, onerror: (res) => { console.error(song.title, '4.发布资源', res) this.uploadSongFail(song) } }) }, onerror: (res) => { console.error(song.title, '3.提交文件', res) this.uploadSongFail(song) } }); } catch (e) { console.error(e); this.uploadSongFail(song) } } uploadSongMatch(song) { //step5 关联 if (song.cloudSongId != song.id) { weapiRequest("/api/cloud/user/song/match", { data: { songId: song.cloudSongId, adjustSongId: song.id, }, onload: (res5) => { if (res5.code != 200) { console.error(song.title, '5.匹配歌曲', res5) this.uploadSongFail(song) return } console.log(song.title, '5.匹配歌曲', res5) console.log(song.title, '完成') //完成 showTips(`(6/6)${song.title} 上传完成`, 1) this.uploadSongSuccess(song) }, onerror: (res) => { console.error(song.title, '5.匹配歌曲', res) this.uploadSongFail(song) } }) } else { console.log(song.title, '完成') //完成 showTips(`(6/6)${song.title} 上传完成`, 1) this.uploadSongSuccess(song) } } uploadSongFail(song) { showTips(`${song.title} 上传失败`, 2) this.failSongs.push(song) if (this.onSongDUFail) this.onSongDUFail(song) this.uploadNextSong() } uploadSongSuccess(song) { if (this.onSongDUSuccess) this.onSongDUSuccess(song) this.uploadNextSong() } uploadNextSong() { this.currentIndex += 1; if (this.currentIndex < this.songs.length) { this.uploadSong(this.songs[this.currentIndex]) } else { let msg = this.failSongs == 0 ? `${this.songs[0].title}上传完成` : `${this.songs[0].title}上传失败` if (this.songs.length > 1) msg = this.failSongs == 0 ? '全部上传完成' : `上传完毕,存在${this.failSongs.length}首上传失败的歌曲.它们为:${this.failSongs.map(song => song.title).join()}` if (this.showfinishBox) { showConfirmBox(msg) } } } } //歌曲页 if (location.href.includes('song')) { let cvrwrap = document.querySelector(".cvrwrap") if (cvrwrap) { let songId = new URLSearchParams(location.search).get('id') //SongDetailAppend class SongDetailAppend { constructor(songId, maindDiv) { this.songId = songId this.maindDiv = maindDiv }; start() { weapiRequest("/api/batch", { data: { '/api/v3/song/detail': JSON.stringify({ c: JSON.stringify([{ 'id': this.songId }]) }), '/api/song/music/detail/get': JSON.stringify({ 'songId': this.songId, 'immerseType': 'ste' }), '/api/song/red/count': JSON.stringify({ 'songId': this.songId }), '/api/song/lyric/v1': JSON.stringify({ id: this.songId, cp: false, tv: 0, lv: 0, rv: 0, kv: 0, yv: 0, ytv: 0, yrv: 0, }), '/api/song/play/about/block/page': JSON.stringify({ 'songId': this.songId }), }, onload: (res) => { //example songid:1914447186 console.log(res) this.title = res["/api/v3/song/detail"].songs[0].name this.album = getAlbumTextInSongDetail(res["/api/v3/song/detail"].songs[0]) this.artist = getArtistTextInSongDetail(res["/api/v3/song/detail"].songs[0]) this.filename = nameFileWithoutExt(this.title, this.artist, 'artist-title') if (res["/api/v3/song/detail"].privileges[0].plLevel != 'none') { this.downLoadBlock = this.createBlock('下载歌曲') let plLevel = res["/api/v3/song/detail"].privileges[0].plLevel let dlLevel = res["/api/v3/song/detail"].privileges[0].dlLevel let songPlWeight = levelWeight[plLevel] || 0 let songDlWeight = levelWeight[dlLevel] || 0 let songDetail = res["/api/song/music/detail/get"].data if (res["/api/v3/song/detail"].privileges[0].cs) { this.createDLButton(`云盘文件(${levelDesc(plLevel)})`, 'standard', 'pl') } else { this.upLoadBlock = this.createBlock('转存云盘') if (songDetail.l && songPlWeight >= 1) { let desc = `标准(${Math.round(songDetail.l.br / 1000)}k/${fileSizeDesc(songDetail.l.size)})`; let level = 'standard'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.m && songPlWeight >= 2) { let desc = `较高(${Math.round(songDetail.m.br / 1000)}k/${fileSizeDesc(songDetail.m.size)})`; let level = 'higher'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.h && songPlWeight >= 3) { let desc = `极高(${Math.round(songDetail.h.br / 1000)}k/${fileSizeDesc(songDetail.h.size)})`; let level = 'exhigh'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.sq && songPlWeight >= 4) { let desc = `无损(${Math.round(songDetail.sq.br / 1000)}k/${fileSizeDesc(songDetail.sq.size)})`; let level = 'lossless'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.hr && songPlWeight >= 4) { let desc = `Hi-Res(${Math.round(songDetail.hr.br / 1000)}k/${songDetail.hr.sr / 1000}kHz/${fileSizeDesc(songDetail.hr.size)})`; let level = 'hires'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.je && songPlWeight >= 4) { let desc = `高清环绕声(${Math.round(songDetail.je.br / 1000)}k/${songDetail.je.sr / 1000}kHz/${fileSizeDesc(songDetail.je.size)})`; let level = 'jyeffect'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.sk && songPlWeight >= 7) { let desc = `沉浸环绕声(${Math.round(songDetail.sk.br / 1000)}k/${songDetail.sk.sr / 1000}kHz/${fileSizeDesc(songDetail.sk.size)})`; let level = 'sky'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDetail.jm && songPlWeight >= 7) { let desc = `超清母带(${Math.round(songDetail.jm.br / 1000)}k/${songDetail.jm.sr / 1000}kHz/${fileSizeDesc(songDetail.jm.size)})`; let level = 'jymaster'; this.createDLButton(desc, level, 'pl'); this.createULButton(desc, level, 'pl') } if (songDlWeight > songPlWeight && res["/api/v3/song/detail"].privileges[0].fee == 0) { if (songDetail.m && songDlWeight >= 2 && songPlWeight < 2) { let desc = `较高(${Math.round(songDetail.m.br / 1000)}k/${fileSizeDesc(songDetail.m.size)})`; let level = 'higher'; this.createDLButton(desc, level, 'dl'); this.createULButton(desc, level, 'dl') } if (songDetail.h && songDlWeight >= 3 && songPlWeight < 3) { let desc = `极高(${Math.round(songDetail.h.br / 1000)}k/${fileSizeDesc(songDetail.h.size)})`; let level = 'exhigh'; this.createDLButton(desc, level, 'dl'); this.createULButton(desc, level, 'dl') } if (songDetail.sq && songDlWeight >= 4 && songPlWeight < 4) { let desc = `无损(${Math.round(songDetail.sq.br / 1000)}k/${fileSizeDesc(songDetail.sq.size)})`; let level = 'lossless'; this.createDLButton(desc, level, 'dl'); this.createULButton(desc, level, 'dl') } if (songDetail.hr && songDlWeight >= 5 && songPlWeight < 5) { let desc = `Hi-Res(${Math.round(songDetail.hr.br / 1000)}k/${songDetail.hr.sr / 1000}kHz/${fileSizeDesc(songDetail.hr.size)})`; let level = 'hires'; this.createDLButton(desc, level, 'dl'); this.createULButton(desc, level, 'dl') } } } } //lyric this.lyricObj = handleLyric(res["/api/song/lyric/v1"]) if (this.lyricObj.orilrc.parsedLyric.length > 0) { this.lyricBlock = this.createBlock('下载歌词') if (this.lyricObj.oritlrc) { let btn = this.createButton('原歌词+翻译') btn.addEventListener('click', () => { this.downloadLyric('oritlrc') }) this.lyricBlock.appendChild(btn) } if (this.lyricObj.oriromalrc) { let btn = this.createButton('罗马音+原歌词') btn.addEventListener('click', () => { this.downloadLyric('oriromalrc') }) this.lyricBlock.appendChild(btn) } let btn = this.createButton('原歌词') btn.addEventListener('click', () => { this.downloadLyric('orilrc') }) this.lyricBlock.appendChild(btn) } if (res["/api/v3/song/detail"].songs[0].al.picUrl) { let CoverBlock = this.createBlock('') let btn = this.createButton('专辑封面原图') btn.href = res["/api/v3/song/detail"].songs[0].al.picUrl btn.target = '_blank' CoverBlock.appendChild(btn) } if (res["/api/song/red/count"].data.count > 0) { this.createBlock(`红心数量 ${res["/api/song/red/count"].data.count}`) } if (res["/api/v3/song/detail"].songs[0].originCoverType > 0) { this.createBlock(`原唱翻唱类型 ${res["/api/v3/song/detail"].songs[0].originCoverType == 1 ? "原唱" : "翻唱"}`) } //脏标 if ((res["/api/v3/song/detail"].songs[0].mark & songMark.explicit) == songMark.explicit) { this.createBlock('🅴:内容含有不健康因素') } //wiki for (let block of res["/api/song/play/about/block/page"].data.blocks) { if (block.code == 'SONG_PLAY_ABOUT_MUSIC_MEMORY' && block.creatives.length > 0) { let memoryBlock = this.createBlock('回忆坐标') let info = block.creatives[0].resources let firstTimeP = document.createElement('p'); firstTimeP.innerHTML = `第一次听:${info[0].resourceExt.musicFirstListenDto.date}` firstTimeP.style.margin = '5px'; memoryBlock.appendChild(firstTimeP) let recordP = document.createElement('p'); recordP.innerHTML = `累计播放:${info[1].resourceExt.musicTotalPlayDto.playCount}次 ${info[1].resourceExt.musicTotalPlayDto.duration}分钟 ${info[1].resourceExt.musicTotalPlayDto.text}` recordP.style.margin = '5px'; memoryBlock.appendChild(recordP) } if (block.code == 'SONG_PLAY_ABOUT_SONG_BASIC' && block.creatives.length > 0) { for (let creative of block.creatives) { if (creative.creativeType == 'sheet' && creative.resources.length == 0) continue let wikiItemBlock = this.createBlock() if (creative.uiElement) { if (creative.uiElement.mainTitle) { wikiItemBlock.innerHTML = creative.uiElement.mainTitle.title } if (creative.uiElement.descriptions) { let descriptionDiv = document.createElement('div') for (let description of creative.uiElement.descriptions) { let descriptionP = this.createText(description.description) descriptionDiv.appendChild(descriptionP) } wikiItemBlock.appendChild(descriptionDiv) } if (creative.uiElement.textLinks) { for (let textLink of creative.uiElement.textLinks) { let textLinkP = this.createText(textLink.text) wikiItemBlock.appendChild(textLinkP) } } } if (creative.resources) { for (let resource of creative.resources) { let resourceDiv = document.createElement('div'); resourceDiv.className = "des s-fc3" if (resource.uiElement.mainTitle) { let IsLink = resource.action?.clickAction?.action == 1 && resource.action?.clickAction?.targetUrl.startsWith('https://') let domType = IsLink ? 'a' : 'span' let mainTitleItem = IsLink ? this.createButton(resource.uiElement.mainTitle.title) : this.createText(resource.uiElement.mainTitle.title) if (IsLink) { mainTitleItem.target = '_blank' mainTitleItem.href = resource.action?.clickAction?.targetUrl } wikiItemBlock.appendChild(mainTitleItem) } if (resource.uiElement.subTitles) { let subTitleP = this.createText(resource.uiElement.subTitles.map(t => t.title).join(' ')) subTitleP.innerHTML = resource.uiElement.subTitles.map(t => t.title).join(' ') wikiItemBlock.appendChild(subTitleP) } if (resource.uiElement.descriptions) { for (let description of resource.uiElement.descriptions) { let descriptionP = this.createText(description.description) wikiItemBlock.appendChild(descriptionP) } } if (resource.uiElement.images) { for (let image of resource.uiElement.images) { let imageA = this.createButton(image.title) imageA.target = '_blank' imageA.href = image.imageUrl || image.imageWithoutTextUrl wikiItemBlock.appendChild(imageA) } } if (resource.uiElement.textLinks) { for (let textLink of resource.uiElement.textLinks) { if (textLink.text) { let textLinkP = this.createText(textLink.text) wikiItemBlock.appendChild(textLinkP) } } } } } } } } } }) } createBlock(innerHTML) { let blockDiv = document.createElement('div'); blockDiv.className = "out s-fc3" let blockP = document.createElement('p'); blockP.innerHTML = innerHTML blockDiv.appendChild(blockP) this.maindDiv.appendChild(blockDiv) return blockP } createButton(desc) { let btn = document.createElement('a'); btn.text = desc; btn.className = "s-fc7" btn.style.margin = '5px' return btn } createText(desc) { let btn = document.createElement('span'); btn.innerHTML = desc btn.style.margin = '5px' return btn } createDLButton(desc, level, channel) { let btn = this.createButton(desc) btn.addEventListener('click', () => { this.dwonloadSong(channel, level, btn) }) this.downLoadBlock.appendChild(btn) } createULButton(desc, level, channel) { if (!unsafeWindow.GUser.userId) return let apiUrl = '/api/song/enhance/player/url/v1' if (channel == 'dl') apiUrl = '/api/song/enhance/download/url/v1' let data = { ids: JSON.stringify([songId]), level: level, encodeType: 'mp3' } if (channel == 'dl') data = { id: songId, level: level, encodeType: 'mp3' } let api = { url: apiUrl, data: data } let songItem = { api: api, id: this.songId, title: this.title, artist: this.artist, album: this.album } let btn = this.createButton(desc) btn.addEventListener('click', () => { let ULobj = new ncmDownUpload([songItem]) ULobj.startUpload() }) this.upLoadBlock.appendChild(btn) } dwonloadSong(channel, level, dlbtn) { let api = '/api/song/enhance/player/url/v1' if (channel == 'dl') api = '/api/song/enhance/download/url/v1' let data = { ids: JSON.stringify([songId]), level: level, encodeType: 'mp3' } if (channel == 'dl') data = { id: songId, level: level, encodeType: 'mp3' } weapiRequest(api, { data: data, onload: (content) => { let resData = content.data[0] || content.data if (resData.url != null) { //console.log(content) let fileFullName = this.filename + '.' + resData.type.toLowerCase() let url = resData.url let btntext = dlbtn.text GM_download({ url: url, name: fileFullName, onprogress: function (e) { dlbtn.text = btntext + ` 正在下载(${fileSizeDesc(e.loaded)})` }, onload: function () { dlbtn.text = btntext }, onerror: function (e) { console.error(e) dlbtn.text = btntext + ' 下载失败' } }); } else { showTips('下载失败', 2) } } }) } downloadLyric(lrcKey) { saveContentAsFile(this.lyricObj[lrcKey].lyric, this.filename + '.lrc') } } let songDetailAppend = new SongDetailAppend(songId, cvrwrap) songDetailAppend.start() } } //个人主页 if (location.href.includes('user')) { let urlUserId = new URLSearchParams(location.search).get('id') let editArea = document.querySelector('#head-box > dd > div.name.f-cb > div > div.edit') if (editArea && urlUserId == unsafeWindow.GUser.userId) { //听歌量打卡 let btnListen = createBigButton('听歌量打卡', editArea, 2) btnListen.addEventListener('click', listenDaily) function listenDaily() { let begin = Math.floor(new Date().getTime() / 1000) let logs = [] for (let i = begin; i < begin + 320; i++) { logs.push({ action: 'play', json: { download: 0, end: 'playend', id: i, sourceId: '', time: 300, type: 'song', wifi: 0, source: 'list' } }) } weapiRequest('/api/feedback/weblog', { cookie: 'os=pc;appver=2.9.7', data: { logs: JSON.stringify(logs) }, onload: (res) => { //console.log(res1) if (res.code == 200) { showConfirmBox('今日听歌量+300首完成') } else { showConfirmBox('听歌量打卡失败。' + res) } } }) } //歌曲快传 let btnUpload = createBigButton('快速上传加载中', editArea, 2) let btnUploadDesc = btnUpload.firstChild var toplist = [] var selectOptions = { "热门": {}, "华语男歌手": {}, "华语女歌手": {}, "华语组合": {}, "欧美男歌手": {}, "欧美女歌手": {}, "欧美组合": {}, "日本男歌手": {}, "日本女歌手": {}, "日本组合": {}, "韩国男歌手": {}, "韩国女歌手": {}, "韩国组合": {}, } var optionMap = { 0: "热门", 1: "华语男歌手", 2: "华语女歌手", 3: "华语组合", 4: "欧美男歌手", 5: "欧美女歌手", 6: "欧美组合", 7: "日本男歌手", 8: "日本女歌手", 9: "日本组合", 10: "韩国男歌手", 11: "韩国女歌手", 12: "韩国组合" } var artistmap = {} fetch(`${baseCDNURL}top.json`) .then(r => r.json()) .then(r => { toplist = r; toplist.forEach(artist => { let desc = `${artist.name}(${artist.count}首/${artist.sizeDesc})`; let id = artist.id selectOptions[optionMap[artist.categroy]][artist.id] = `${artist.name}(${artist.count}首/${artist.sizeDesc})` artistmap[artist.id] = artist }) //console.log(selectOptions) btnUpload.addEventListener('click', ShowCloudUploadPopUp) btnUploadDesc.innerHTML = '云盘快速上传' }) function ShowCloudUploadPopUp() { Swal.fire({ title: '快速上传', input: 'select', inputOptions: selectOptions, inputPlaceholder: '选择歌手', confirmButtonText: '下一步', showCloseButton: true, footer: 'Github', inputValidator: (value) => { if (!value) { return '请选择歌手' } }, }) .then(result => { if (result.isConfirmed) { fetchCDNConfig(result.value) } }) } function fetchCDNConfig(artistId) { showTips(`正在获取资源配置...`, 1) fetch(`${baseCDNURL}${artistId}.json`) .then(r => r.json()) .then(r => { let uploader = new Uploader(r) uploader.start() }) .catch(`获取资源配置失败`) } class Uploader { constructor(config, showAll = false) { this.songs = [] this.config = config this.filter = { text: '', noCopyright: true, vip: true, pay: true, lossless: false, all: showAll, songIndexs: [] } this.page = { current: 1, max: 1, limitCount: 50 } this.batchUpload = { threadMax: 2, threadCount: 2, working: false, finnishThread: 0, songIndexs: [] } }; start() { this.showPopup() } showPopup() { Swal.fire({ showCloseButton: true, showConfirmButton: false, width: 800, html: `
操作歌曲标题歌手时长文件信息备注
`, footer: '
', didOpen: () => { let container = Swal.getHtmlContainer() let footer = Swal.getFooter() let tbody = container.querySelector('tbody') this.popupObj = { container: container, tbody: tbody, footer: footer } //this.filter={text:'',noCopyright:true,vip:true,pay:true,lossless:false,songIds:[]} let filterInput = container.querySelector('#text-filter') filterInput.addEventListener('change', () => { let filtertext = filterInput.value.trim() if (this.filter.text != filtertext) { this.filter.text = filtertext this.applyFilter() } }) let copyrightInput = container.querySelector('#cb-copyright') copyrightInput.addEventListener('change', () => { this.filter.noCopyright = copyrightInput.checked this.applyFilter() }) let vipInput = container.querySelector('#cb-vip') vipInput.addEventListener('change', () => { this.filter.vip = vipInput.checked this.applyFilter() }) let payInput = container.querySelector('#cb-pay') payInput.addEventListener('change', () => { this.filter.pay = payInput.checked this.applyFilter() }) let losslessInput = container.querySelector('#cb-lossless') losslessInput.addEventListener('change', () => { this.filter.lossless = losslessInput.checked this.applyFilter() }) let allInput = container.querySelector('#cb-all') allInput.addEventListener('change', () => { this.filter.all = allInput.checked this.applyFilter() }) let uploader = this this.btnUploadBatch = container.querySelector('#btn-upload-batch') this.btnUploadBatch.addEventListener('click', () => { if (this.batchUpload.working) { return } this.batchUpload.songIndexs = [] this.filter.songIndexs.forEach(idx => { if (!uploader.songs[idx].uploaded) { uploader.batchUpload.songIndexs.push(idx) } }) if (this.batchUpload.songIndexs.length == 0) { showTips('没有需要上传的歌曲', 1) return } this.batchUpload.working = true this.batchUpload.finnishThread = 0 this.batchUpload.threadCount = Math.min(this.batchUpload.songIndexs.length, this.batchUpload.threadMax) for (let i = 0; i < this.batchUpload.threadCount; i++) { this.uploadSong(this.batchUpload.songIndexs[i]) } }) this.fetchSongInfo() }, }) } fetchSongInfo() { //console.log(songList) let ids = this.config.data.map(item => { return { 'id': item.id } }) //获取需上传的song this.popupObj.tbody.innerHTML = '正在获取歌曲信息...' this.fetchSongInfoSub(ids, 0) } fetchSongInfoSub(ids, startIndex) { if (startIndex >= ids.length) { if (this.songs.length == 0) { this.popupObj.tbody.innerHTML = '没有可以上传的歌曲' return } //排序 this.songs.sort((a, b) => { if (a.albumid != b.albumid) { return b.albumid - a.albumid } return a.id - b.id }) this.createTableRow() this.applyFilter() return } this.popupObj.tbody.innerHTML = `正在获取第${startIndex + 1}到${Math.min(ids.length, startIndex + 1000)}首歌曲信息...` let uploader = this weapiRequest("/api/v3/song/detail", { data: { c: JSON.stringify(ids.slice(startIndex, startIndex + 1000)) }, onload: function (content) { //console.log(content) let songslen = content.songs.length let privilegelen = content.privileges.length for (let i = 0; i < privilegelen; i++) { if (!content.privileges[i].cs) { let config = uploader.config.data.find(item => { return item.id == content.privileges[i].id }) let item = { id: content.privileges[i].id, name: '未知', album: '未知', albumid: 0, artists: '未知', tns: '', //翻译 dt: duringTimeDesc(0), filename: '未知.' + config.ext, ext: config.ext, md5: config.md5, size: config.size, bitrate: config.bitrate, picUrl: 'http://p4.music.126.net/UeTuwE7pvjBpypWLudqukA==/3132508627578625.jpg', isNoCopyright: content.privileges[i].st < 0, isVIP: false, isPay: false, uploaded: false, needMatch: config.name == undefined, } for (let j = 0; j < songslen; j++) { if (content.songs[j].id == content.privileges[i].id) { item.name = content.songs[j].name item.album = getAlbumTextInSongDetail(content.songs[j]) item.albumid = content.songs[j].al.id || 0 item.artists = getArtistTextInSongDetail(content.songs[j]) item.tns = content.songs[j].tns ? content.songs[j].tns.join() : '' //翻译 item.dt = duringTimeDesc(content.songs[j].dt || 0) item.filename = nameFileWithoutExt(item.name, item.artists, 'artist-title') + '.' + config.ext item.picUrl = (content.songs[j].al && content.songs[j].al.picUrl) ? content.songs[j].al.picUrl : 'http://p4.music.126.net/UeTuwE7pvjBpypWLudqukA==/3132508627578625.jpg' item.isVIP = content.songs[j].fee == 1 item.isPay = content.songs[j].fee == 4 break } } if (config.name) { item.name = config.name item.album = config.al item.artists = config.ar item.filename = nameFileWithoutExt(item.name, item.artists, 'artist-title') + '.' + config.ext } uploader.songs.push(item) } } uploader.fetchSongInfoSub(ids, startIndex + 1000) } }) } createTableRow() { for (let i = 0; i < this.songs.length; i++) { let song = this.songs[i] let tablerow = document.createElement('tr') tablerow.innerHTML = `${song.name}${song.artists}${song.dt}${fileSizeDesc(song.size)} ${song.ext.toUpperCase()}` let songTitle = tablerow.querySelector('.song-remark') if (song.isNoCopyright) { songTitle.innerHTML = '无版权' } else if (song.isVIP) { songTitle.innerHTML = 'VIP' } else if (song.isPay) { songTitle.innerHTML = '数字专辑' } let btn = tablerow.querySelector('button') btn.addEventListener('click', () => { if (this.batchUpload.working) { return } this.uploadSong(i) }) song.tablerow = tablerow } } applyFilter() { this.filter.songIndexs = [] let filterText = this.filter.text let isNoCopyright = this.filter.noCopyright let isVIP = this.filter.vip let isPay = this.filter.pay let isLossless = this.filter.lossless let isALL = this.filter.all for (let i = 0; i < this.songs.length; i++) { let song = this.songs[i] if (filterText.length > 0 && !song.name.includes(filterText) && !song.album.includes(filterText) && !song.artists.includes(filterText) && !song.tns.includes(filterText)) { continue } if (isALL) { this.filter.songIndexs.push(i) } else if (isNoCopyright && song.isNoCopyright) { this.filter.songIndexs.push(i) } else if (isVIP && song.isVIP) { this.filter.songIndexs.push(i) } else if (isPay && song.isPay) { this.filter.songIndexs.push(i) } else if (isLossless && song.ext == 'flac') { this.filter.songIndexs.push(i) } } this.page.current = 1 this.page.max = Math.ceil(this.filter.songIndexs.length / this.page.limitCount) this.renderData() this.renderFilterInfo() } renderData() { if (this.filter.songIndexs.length == 0) { this.popupObj.tbody.innerHTML = '空空如也' this.popupObj.footer.innerHTML = '' return } //table this.popupObj.tbody.innerHTML = '' let songBegin = (this.page.current - 1) * this.page.limitCount let songEnd = Math.min(this.filter.songIndexs.length, songBegin + this.page.limitCount) for (let i = songBegin; i < songEnd; i++) { this.popupObj.tbody.appendChild(this.songs[this.filter.songIndexs[i]].tablerow) } //page let pageIndexs = [1] let floor = Math.max(2, this.page.current - 2); let ceil = Math.min(this.page.max - 1, this.page.current + 2); for (let i = floor; i <= ceil; i++) { pageIndexs.push(i) } if (this.page.max > 1) { pageIndexs.push(this.page.max) } let uploader = this this.popupObj.footer.innerHTML = '' pageIndexs.forEach(pageIndex => { let pageBtn = document.createElement('button') pageBtn.setAttribute("type", "button") pageBtn.className = "swal2-styled" pageBtn.innerHTML = pageIndex if (pageIndex != uploader.page.current) { pageBtn.addEventListener('click', () => { uploader.page.current = pageIndex uploader.renderData() }) } else { pageBtn.style.background = 'white' } uploader.popupObj.footer.appendChild(pageBtn) }) } renderFilterInfo() { //this.btnUploadBatch //this.filter.songs let sizeTotal = 0 let countCanUpload = 0 this.filter.songIndexs.forEach(idx => { let song = this.songs[idx] if (!song.uploaded) { countCanUpload += 1 sizeTotal += song.size } }) this.btnUploadBatch.innerHTML = '全部上传' if (countCanUpload > 0) { this.btnUploadBatch.innerHTML += ` (${countCanUpload}首 ${fileSizeDesc(sizeTotal)})` } } uploadSong(songIndex) { let song = this.songs[songIndex] let uploader = this try { let songCheckData = [{ md5: song.md5, songId: song.id, bitrate: song.bitrate, fileSize: song.size, }] weapiRequest("/api/cloud/upload/check/v2", { data: { uploadType: 0, songs: JSON.stringify(songCheckData), }, onload: (res1) => { if (res1.code != 200) { console.error(song.name, '1.检查资源', res1) uploader.onUploadFail(songIndex) return } if (res1.data.length < 1) { if (song.id > 0) { //被**的歌曲要id设为0 uploader.songs[songIndex].id = 0 uploader.uploadSong(songIndex) } else { console.error(song.name, '1.检查资源', res1) uploader.onUploadFail(songIndex) } return } console.log(song.name, '1.检查资源', res1) song.cloudId = res1.data[0].songId showTips(`(2/6)${song.name} 检查资源`, 1) if (res1.data[0].upload == 1) { uploader.uploadSongWay1Part1(songIndex) } else { uploader.uploadSongWay2Part1(songIndex) } }, onerror: function (res) { console.error(song.name, '1.检查资源', res) uploader.onUploadFail(songIndex) } }) } catch (e) { console.error(e); uploader.onUploadFail(songIndex) } } uploadSongWay1Part1(songIndex) { let song = this.songs[songIndex] let uploader = this let importSongData = [{ songId: song.cloudId, bitrate: song.bitrate, song: song.needMatch ? nameFileWithoutExt(song.name, song.artists, 'artist-title') : song.name, artist: song.artists, album: song.album, fileName: song.filename, }] //step2 导入歌曲 try { weapiRequest("/api/cloud/user/song/import", { data: { uploadType: 0, songs: JSON.stringify(importSongData), }, onload: (res) => { if (res.code != 200 || res.data.successSongs.length < 1) { console.error(song.name, '2.导入文件', res) uploader.onUploadFail(songIndex) return } console.log(song.name, '2.导入文件', res) song.cloudSongId = res.data.successSongs[0].song.songId uploader.uploadSongMatch(songIndex) }, onerror: (responses2) => { console.error(song.name, '2.导入歌曲', responses2) uploader.onUploadFail(songIndex) } }) } catch (e) { console.error(e); uploader.onUploadFail(songIndex) } } uploadSongWay2Part1(songIndex) { let song = this.songs[songIndex] let uploader = this try { weapiRequest("/api/nos/token/alloc", { data: { filename: song.filename, length: song.size, ext: song.ext, type: 'audio', bucket: 'jd-musicrep-privatecloud-audio-public', local: false, nos_product: 3, md5: song.md5 }, onload: (tokenRes) => { song.token = tokenRes.result.token song.objectKey = tokenRes.result.objectKey song.resourceId = tokenRes.result.resourceId song.expireTime = Date.now() + 60000 console.log(song.name, '2.2.开始上传', tokenRes) uploader.uploadSongWay2Part2(songIndex) }, onerror: (responses2) => { console.error(song.name, '2.获取令牌', responses2) uploader.onUploadFail(songIndex) } }) } catch (e) { console.error(e); uploader.onUploadFail(songIndex) } } uploadSongWay2Part2(songIndex) { let song = this.songs[songIndex] let uploader = this weapiRequest("/api/upload/cloud/info/v2", { data: { md5: song.md5, songid: song.cloudId, filename: song.filename, song: song.name, album: song.album, artist: song.artists, bitrate: String(song.bitrate || 128), resourceId: song.resourceId, }, onload: (res3) => { if (res3.code != 200) { if (song.expireTime < Date.now() || (res3.msg && res3.msg.includes('rep create failed'))) { console.error(song.name, '3.提交文件', res3) uploader.onUploadFail(songIndex) } else { console.log(song.name, '3.正在转码', res3) sleep(1000).then(() => { uploader.uploadSongWay2Part2(songIndex) }) } return } console.log(song.name, '3.提交文件', res3) //step4 发布 weapiRequest("/api/cloud/pub/v2", { data: { songid: res3.songId, }, onload: (res4) => { if (res4.code != 200 && res4.code != 201) { console.error(song.name, '4.发布资源', res4) uploader.onUploadFail(songIndex) return } console.log(song.name, '4.发布资源', res4) song.cloudSongId = res4.privateCloud.songId //step5 关联 uploader.uploadSongMatch(songIndex) }, onerror: function (res) { console.error(song.name, '4.发布资源', res) uploader.onUploadFail(songIndex) } }) }, onerror: function (res) { console.error(song.name, '3.提交文件', res) uploader.onUploadFail(songIndex) } }); } uploadSongMatch(songIndex) { let song = this.songs[songIndex] let uploader = this if (song.cloudSongId != song.id && song.id > 0) { weapiRequest("/api/cloud/user/song/match", { data: { songId: song.cloudSongId, adjustSongId: song.id, }, onload: (res5) => { if (res5.code != 200) { console.error(song.name, '5.匹配歌曲', res5) uploader.onUploadFail(songIndex) return } console.log(song.name, '5.匹配歌曲', res5) console.log(song.name, '完成') //完成 uploader.onUploadSucess(songIndex) }, onerror: function (res) { console.error(song.name, '5.匹配歌曲', res) uploader.onUploadFail(songIndex) } }) } else { console.log(song.name, '完成') //完成 uploader.onUploadSucess(songIndex) } } onUploadFail(songIndex) { let song = this.songs[songIndex] showTips(`${song.name} - ${song.artists} - ${song.album} 上传失败`, 2) this.onUploadFinnsh(songIndex) } onUploadSucess(songIndex) { let song = this.songs[songIndex] showTips(`${song.name} - ${song.artists} - ${song.album} 上传成功`, 1) song.uploaded = true let btnUpload = song.tablerow.querySelector('button') btnUpload.innerHTML = '已上传' btnUpload.disabled = 'disabled' this.onUploadFinnsh(songIndex) } onUploadFinnsh(songIndex) { if (this.batchUpload.working) { let batchSongIdx = this.batchUpload.songIndexs.indexOf(songIndex) if (batchSongIdx + this.batchUpload.threadCount < this.batchUpload.songIndexs.length) { this.uploadSong(this.batchUpload.songIndexs[batchSongIdx + this.batchUpload.threadCount]) } else { this.batchUpload.finnishThread += 1 if (this.batchUpload.finnishThread == this.batchUpload.threadCount) { this.batchUpload.working = false this.renderFilterInfo() showTips('上传完成', 1) } } } else { this.renderFilterInfo() } } } //匹配纠正 let btnMatch = createBigButton('云盘匹配纠正', editArea, 2) btnMatch.addEventListener('click', () => { let matcher = new Matcher() matcher.start() }) class Matcher { start() { this.cloudCountLimit = 50 this.currentPage = 1 this.filter = { text: '', notMatch: false, songs: [], filterInput: null, notMatchCb: null } this.controls = { tbody: null, pageArea: null, cloudDesc: null } this.openCloudList() } openCloudList() { Swal.fire({ showCloseButton: true, showConfirmButton: false, width: 800, html: ` `, footer: `

${this.controls.cloudDesc ? this.controls.cloudDesc.innerHTML : ''}
`, didOpen: () => { let cloudListContainer = Swal.getHtmlContainer() let cloudListFooter = Swal.getFooter() cloudListFooter.style.display = 'block' cloudListFooter.style.textAlign = 'center' let songtb = document.createElement('table') songtb.border = 1 songtb.frame = 'hsides' songtb.rules = 'rows' songtb.innerHTML = `操作歌曲标题歌手时长文件信息上传日期 ` let tbody = songtb.querySelector('tbody') this.controls.tbody = tbody this.controls.pageArea = cloudListFooter.querySelector('#page-area') this.controls.cloudDesc = cloudListFooter.querySelector('#cloud-desc') let filterInput = cloudListContainer.querySelector('#text-filter') let notMatchCb = cloudListContainer.querySelector('#cb-notmatch') this.filter.filterInput = filterInput this.filter.notMatchCb = notMatchCb let matcher = this filterInput.addEventListener('change', () => { if (matcher.filter.text == filterInput.value.trim()) { return } matcher.filter.text = filterInput.value.trim() this.onCloudInfoFilterChange() }) notMatchCb.addEventListener('change', () => { matcher.filter.notMatch = notMatchCb.checked this.onCloudInfoFilterChange() }) cloudListContainer.appendChild(songtb) if (this.filter.text == '' && !this.filter.notMatch) { this.fetchCloudInfoForMatchTable((this.currentPage - 1) * this.cloudCountLimit) } else { this.sepreateFilterCloudListPage(this.currentPage) } }, }) } fetchCloudInfoForMatchTable(offset) { this.controls.tbody.innerHTML = '正在获取...' let matcher = this weapiRequest('/api/v1/cloud/get', { data: { limit: this.cloudCountLimit, offset: offset, }, onload: (res) => { //console.log(res) matcher.currentPage = (offset / this.cloudCountLimit) + 1 let maxPage = Math.ceil(res.count / this.cloudCountLimit) this.controls.cloudDesc.innerHTML = `云盘容量 ${fileSizeDesc(res.size)}/${fileSizeDesc(res.maxSize)} 共${res.count}首歌曲` let pageIndexs = [1] let floor = Math.max(2, matcher.currentPage - 2); let ceil = Math.min(maxPage - 1, matcher.currentPage + 2); for (let i = floor; i <= ceil; i++) { pageIndexs.push(i) } if (maxPage > 1) { pageIndexs.push(maxPage) } matcher.controls.pageArea.innerHTML = '' pageIndexs.forEach(pageIndex => { let pageBtn = document.createElement('button') pageBtn.setAttribute("type", "button") pageBtn.className = "swal2-styled" pageBtn.innerHTML = pageIndex if (pageIndex != matcher.currentPage) { pageBtn.addEventListener('click', () => { matcher.fetchCloudInfoForMatchTable(matcher.cloudCountLimit * (pageIndex - 1)) }) } else { pageBtn.style.background = 'white' } matcher.controls.pageArea.appendChild(pageBtn) }) this.fillCloudListTable(res.data) } }) } fillCloudListTable(songs) { let matcher = this matcher.controls.tbody.innerHTML = '' if (songs.length == 0) { matcher.controls.tbody.innerHTML = '空空如也' } songs.forEach(function (song) { let album = song.album let picUrl = 'http://p4.music.126.net/UeTuwE7pvjBpypWLudqukA==/3132508627578625.jpg' if (song.simpleSong.al && song.simpleSong.al.picUrl) { picUrl = song.simpleSong.al.picUrl } if (song.simpleSong.al && song.simpleSong.al.name && song.simpleSong.al.name.length > 0) { album = song.simpleSong.al.name } let artist = song.artist if (song.simpleSong.ar) { let artist2 = '' let arcount = 0 song.simpleSong.ar.forEach(ar => { if (ar.name) { if (ar.id > 0) artist2 += `${ar.name},` else artist2 += ar.name + ',' arcount += 1 } }) if (arcount > 0) { artist = artist2.substring(0, artist2.length - 1) } } let dateObj = new Date(song.addTime) let addTime = `${dateObj.getFullYear()}-${dateObj.getMonth() + 1}-${dateObj.getDate()}` let tablerow = document.createElement('tr') tablerow.innerHTML = `${song.simpleSong.name}${artist}${duringTimeDesc(song.simpleSong.dt)}${fileSizeDesc(song.fileSize)} ${levelDesc(song.simpleSong.privilege.plLevel)}${addTime}` if (song.simpleSong.al && song.simpleSong.al.id > 0) { let albumLink = tablerow.querySelector('.album-link') albumLink.href = 'https://music.163.com/album?id=' + song.simpleSong.al.id albumLink.target = "_blank" } let btn = tablerow.querySelector('button') btn.addEventListener('click', () => { matcher.openMatchPopup(song) }) matcher.controls.tbody.appendChild(tablerow) }) } onCloudInfoFilterChange() { this.filter.songs = [] if (this.filter.text == '' && !this.filter.notMatch) { this.fetchCloudInfoForMatchTable(0) return } this.filter.filterInput.setAttribute("disabled", 1) this.filter.notMatchCb.setAttribute("disabled", 1) this.cloudInfoFilterFetchData(0) } cloudInfoFilterFetchData(offset) { let matcher = this if (offset == 0) { this.filter.songs = [] } weapiRequest('/api/v1/cloud/get', { data: { limit: 1000, offset: offset, }, onload: (res) => { matcher.controls.tbody.innerHTML = `正在搜索第${offset + 1}到${Math.min(offset + 1000, res.count)}云盘歌曲` res.data.forEach(song => { if (matcher.filter.text.length > 0) { let matchFlag = false if (song.album.includes(matcher.filter.text) || song.artist.includes(matcher.filter.text) || song.simpleSong.name.includes(matcher.filter.text) || (song.simpleSong.al && song.simpleSong.al.id > 0 && song.simpleSong.al.name && song.simpleSong.al.name.includes(matcher.filter.text))) { matchFlag = true } if (!matchFlag && song.simpleSong.ar) { song.simpleSong.ar.forEach(ar => { if (ar.name && ar.name.includes(matcher.filter.text)) { matchFlag = true } }) if (!matchFlag) { return } } } if (matcher.filter.notMatch && song.simpleSong.cd) { return } matcher.filter.songs.push(song) }) if (res.hasMore) { //if(offset<2001){//testing res = {} matcher.cloudInfoFilterFetchData(offset + 1000) } else { matcher.sepreateFilterCloudListPage(1) matcher.filter.filterInput.removeAttribute("disabled") matcher.filter.notMatchCb.removeAttribute("disabled") } } }) } sepreateFilterCloudListPage(currentPage) { this.currentPage = currentPage let matcher = this let count = this.filter.songs.length let maxPage = Math.ceil(count / this.cloudCountLimit) this.controls.pageArea.innerHTML = '' let pageIndexs = [1] let floor = Math.max(2, currentPage - 2); let ceil = Math.min(maxPage - 1, currentPage + 2); for (let i = floor; i <= ceil; i++) { pageIndexs.push(i) } if (maxPage > 1) { pageIndexs.push(maxPage) } matcher.controls.pageArea.innerHTML = '' pageIndexs.forEach(pageIndex => { let pageBtn = document.createElement('button') pageBtn.setAttribute("type", "button") pageBtn.className = "swal2-styled" pageBtn.innerHTML = pageIndex if (pageIndex != currentPage) { pageBtn.addEventListener('click', () => { matcher.sepreateFilterCloudListPage(pageIndex) }) } else { pageBtn.style.background = 'white' } matcher.controls.pageArea.appendChild(pageBtn) }) let songindex = (currentPage - 1) * this.cloudCountLimit matcher.fillCloudListTable(matcher.filter.songs.slice(songindex, songindex + this.cloudCountLimit)) } openMatchPopup(song) { let matcher = this Swal.fire({ showCloseButton: true, title: `歌曲 ${song.simpleSong.name} 匹配纠正`, input: 'number', inputLabel: '目标歌曲ID', footer: 'ID为0时解除匹配 歌曲页面网址里的数字就是ID', inputValidator: (value) => { if (!value) { return '内容为空' } }, didOpen: () => { let titleDOM = Swal.getTitle() weapiRequest("/api/song/enhance/player/url/v1", { data: { immerseType: 'ste', ids: JSON.stringify([song.simpleSong.id]), level: 'standard', encodeType: 'mp3' }, onload: (content) => { titleDOM.innerHTML += ' 文件时长' + duringTimeDesc(content.data[0].time) } }) } }) .then(result => { if (result.isConfirmed) { let fromId = song.simpleSong.id let toId = result.value weapiRequest("/api/cloud/user/song/match", { data: { songId: fromId, adjustSongId: toId, }, onload: (res) => { //console.log(res) if (res.code != 200) { showTips(res.message || res.msg || '匹配失败', 2) } else { let msg = '解除匹配成功' if (toId > 0) { msg = '匹配成功' if (res.matchData) { msg = `${res.matchData.songName} 成功匹配到 ${res.matchData.simpleSong.name} ` } } showTips(msg, 1) if (matcher.filter.songs.length > 0 && res.matchData) { for (let i = 0; i < matcher.filter.songs.length; i++) { if (matcher.filter.songs[i].simpleSong.id == fromId) { //matchData里无privilege res.matchData.simpleSong.privilege = matcher.filter.songs[i].simpleSong.privilege matcher.filter.songs[i] = res.matchData break } } } } matcher.openCloudList() }, }) } else { matcher.openCloudList() } }) } } //音质升级 let btnUpgrade = createBigButton('云盘音质提升', editArea, 2) btnUpgrade.addEventListener('click', ShowCloudUpgradePopUp) function ShowCloudUpgradePopUp() { Swal.fire({ title: '云盘音质提升', input: 'select', inputOptions: { lossless: '无损', hires: 'Hi-Res' }, inputPlaceholder: '选择目标音质', confirmButtonText: '下一步', showCloseButton: true, footer: '寻找网易云音源比云盘音质好的歌曲,然后进行替换Github', inputValidator: (value) => { if (!value) { return '请选择目标音质' } }, }) .then(result => { if (result.isConfirmed) { checkVIPBeforeUpgrade(result.value) } }) } function checkVIPBeforeUpgrade(level) { weapiRequest(`/api/v1/user/detail/${urlUserId}`, { onload: (res) => { if (res.profile.vipType <= 10) { showConfirmBox('当前不是会员,无法获取无损以上音源,领取个会员礼品卡再来吧。') } else { let upgrade = new Upgrader(level) upgrade.start() } } }) } class Upgrader { constructor(level) { this.targetLevel = level this.targetWeight = levelWeight[level] this.songs = [] this.page = { current: 1, max: 1, limitCount: 50 } this.filter = { text: '', songIndexs: [] } this.batchUpgrade = { threadMax: 1, threadCount: 1, working: false, finnishThread: 0, songIndexs: [] } }; start() { this.showPopup() } showPopup() { Swal.fire({ showCloseButton: true, showConfirmButton: false, width: 800, html: `
操作歌曲标题歌手云盘音源目标音源
`, footer: '
', didOpen: () => { let container = Swal.getHtmlContainer() let tbody = container.querySelector('tbody') let footer = Swal.getFooter() this.popupObj = { container: container, tbody: tbody, footer: footer } let filterInput = container.querySelector('#text-filter') filterInput.addEventListener('change', () => { let filtertext = filterInput.value.trim() if (this.filter.text != filtertext) { this.filter.text = filtertext this.applyFilter() } }) let upgrader = this this.btnUpgradeBatch = container.querySelector('#btn-upgrade-batch') this.btnUpgradeBatch.addEventListener('click', () => { if (this.batchUpgrade.working) { return } this.batchUpgrade.songIndexs = [] this.filter.songIndexs.forEach(idx => { if (!upgrader.songs[idx].upgraded) { upgrader.batchUpgrade.songIndexs.push(idx) } }) if (this.batchUpgrade.songIndexs.length == 0) { showTips('没有需要提升的歌曲', 1) return } this.batchUpgrade.working = true this.batchUpgrade.finnishThread = 0 this.batchUpgrade.threadCount = Math.min(this.batchUpgrade.songIndexs.length, this.batchUpgrade.threadMax) for (let i = 0; i < this.batchUpgrade.threadCount; i++) { this.upgradeSong(this.batchUpgrade.songIndexs[i]) } }) this.fetchSongInfo() }, }) } fetchSongInfo() { //获取需上传的song this.popupObj.tbody.innerHTML = '正在查找云盘歌曲...' this.fetchCloudSongInfoSub(0, []) } fetchCloudSongInfoSub(offset, songIds) { let upgrader = this weapiRequest('/api/v1/cloud/get', { data: { limit: 1000, offset: offset, }, onload: (res) => { upgrader.popupObj.tbody.innerHTML = `正在搜索第${offset + 1}到${Math.min(offset + 1000, res.count)}云盘歌曲` res.data.forEach(song => { if (song.simpleSong.privilege.toast) return if (song.simpleSong.privilege.fee == 4) return if (song.simpleSong.privilege.playMaxBrLevel != "lossless") return let cloudWeight = levelWeight[song.simpleSong.privilege.plLevel] || 0 let ncmMaxWeight = levelWeight[song.simpleSong.privilege.playMaxBrLevel] if (cloudWeight >= this.targetWeight) return songIds.push({ 'id': song.simpleSong.id }) upgrader.popupObj.tbody.innerHTML = `正在搜索第${offset + 1}到${Math.min(offset + 1000, res.count)}云盘歌曲 找到${songIds.length}首可能有提升的歌曲` }) if (res.hasMore) { //if(offset<2000){//testing res = {} upgrader.fetchCloudSongInfoSub(offset + 1000, songIds) } else { upgrader.filterTargetLevelSongSub(0, songIds) } } }) } filterTargetLevelSongSub(offset, songIds) { let upgrader = this upgrader.popupObj.tbody.innerHTML = `正在确认${songIds.length}首潜在歌曲是否有目标音质` if (offset >= songIds.length) { upgrader.createTableRow() upgrader.applyFilter() return } weapiRequest("/api/v3/song/detail", { data: { c: JSON.stringify(songIds.slice(offset, offset + 1000)) }, onload: function (content) { let songlen = content.songs.length let privilegelen = content.privileges.length for (let i = 0; i < songlen; i++) { for (let j = 0; j < privilegelen; j++) { if (content.songs[i].id == content.privileges[j].id) { let songItem = { id: content.songs[i].id, name: content.songs[i].name, album: getAlbumTextInSongDetail(content.songs[i]), albumid: content.songs[i].al.id || 0, artists: getArtistTextInSongDetail(content.songs[i]), tns: content.songs[i].tns ? content.songs[i].tns.join() : '', //翻译 dt: duringTimeDesc(content.songs[i].dt || 0), picUrl: (content.songs[i].al && content.songs[i].al.picUrl) ? content.songs[i].al.picUrl : 'http://p4.music.126.net/UeTuwE7pvjBpypWLudqukA==/3132508627578625.jpg', upgraded: false, } let cloudBr = content.songs[i].pc.br if (upgrader.targetLevel == 'lossless' && content.songs[i].sq) { songItem.fileinfo = { originalLevel: content.privileges[j].plLevel, originalBr: content.songs[i].pc.br, tagetBr: Math.round(content.songs[i].sq.br / 1000) } upgrader.songs.push(songItem) } else if (upgrader.targetLevel == 'hires' && content.songs[i].hr) { songItem.fileinfo = { originalLevel: content.privileges[j].plLevel, originalBr: content.songs[i].pc.br, tagetBr: Math.round(content.songs[i].hr.br / 1000) } upgrader.songs.push(songItem) } break } } } upgrader.filterTargetLevelSongSub(offset + 1000, songIds) } }) } createTableRow() { let tagetLevelDesc = levelDesc(this.targetLevel) for (let i = 0; i < this.songs.length; i++) { let song = this.songs[i] let tablerow = document.createElement('tr') tablerow.innerHTML = `${song.name}${song.artists}${levelDesc(song.fileinfo.originalLevel)} ${song.fileinfo.originalBr}k${tagetLevelDesc} ${song.fileinfo.tagetBr}k` let btn = tablerow.querySelector('button') btn.addEventListener('click', () => { if (this.batchUpgrade.working) { return } this.upgradeSong(i) }) song.tablerow = tablerow } } applyFilter() { this.filter.songIndexs = [] let filterText = this.filter.text for (let i = 0; i < this.songs.length; i++) { let song = this.songs[i] if (filterText.length > 0 && !song.name.includes(filterText) && !song.album.includes(filterText) && !song.artists.includes(filterText) && !song.tns.includes(filterText)) { continue } this.filter.songIndexs.push(i) } this.page.current = 1 this.page.max = Math.ceil(this.filter.songIndexs.length / this.page.limitCount) this.renderData() this.renderFilterInfo() } renderData() { if (this.filter.songIndexs.length == 0) { this.popupObj.tbody.innerHTML = '内容为空' this.popupObj.footer.innerHTML = '' return } //table this.popupObj.tbody.innerHTML = '' let songBegin = (this.page.current - 1) * this.page.limitCount let songEnd = Math.min(this.filter.songIndexs.length, songBegin + this.page.limitCount) for (let i = songBegin; i < songEnd; i++) { this.popupObj.tbody.appendChild(this.songs[this.filter.songIndexs[i]].tablerow) } //page let pageIndexs = [1] let floor = Math.max(2, this.page.current - 2); let ceil = Math.min(this.page.max - 1, this.page.current + 2); for (let i = floor; i <= ceil; i++) { pageIndexs.push(i) } if (this.page.max > 1) { pageIndexs.push(this.page.max) } let upgrader = this this.popupObj.footer.innerHTML = '' pageIndexs.forEach(pageIndex => { let pageBtn = document.createElement('button') pageBtn.setAttribute("type", "button") pageBtn.className = "swal2-styled" pageBtn.innerHTML = pageIndex if (pageIndex != upgrader.page.current) { pageBtn.addEventListener('click', () => { upgrader.page.current = pageIndex upgrader.renderData() }) } else { pageBtn.style.background = 'white' } upgrader.popupObj.footer.appendChild(pageBtn) }) } renderFilterInfo() { let sizeTotal = 0 let countCanUpgrade = 0 this.filter.songIndexs.forEach(idx => { let song = this.songs[idx] if (!song.upgraded) { countCanUpgrade += 1 sizeTotal += song.size } }) this.btnUpgradeBatch.innerHTML = '全部提升' if (countCanUpgrade > 0) { this.btnUpgradeBatch.innerHTML += ` (${countCanUpgrade}首)` } } upgradeSong(songIndex) { let song = this.songs[songIndex] let upgrade = this try { weapiRequest("/api/cloud/user/song/match", { data: { songId: song.id, adjustSongId: 0, }, onload: (res) => { console.log(res) if (res.code == 200) { showTips(`${song.name}解绑成功`, 1) song.originalId = res.matchData.songId let songItem = { api: { url: '/api/song/enhance/player/url/v1', data: { ids: JSON.stringify([song.id]), level: upgrade.targetLevel, encodeType: 'mp3' } }, id: song.id, title: song.name, artist: song.artists, album: song.album, songIndex: songIndex, Upgrader: this } let ULobj = new ncmDownUpload([songItem], false, this.onUploadSuccess, this.onUploadFail) ULobj.startUpload() } else { showTips(`${song.name}解绑失败`, 2) upgrade.onUpgradeFail(songIndex) } }, }) } catch (e) { console.error(e); upgrade.onUpgradeFail(songIndex) } } onUploadFail(ULsong) { let song = ULsong.Upgrader.songs[ULsong.songIndex] try { weapiRequest("/api/cloud/user/song/match", { data: { songId: song.originalId, adjustSongId: song.id, }, onload: (res) => { console.log(res) if (res.code != 200) { showTips(`${song.name} 重新关联失败`, 2) } ULsong.Upgrader.onUpgradeFail(ULsong.songIndex) }, }) } catch (e) { console.error(e); ULsong.Upgrader.onUpgradeFail(ULsong.songIndex) } } onUploadSuccess(ULsong) { let song = ULsong.Upgrader.songs[ULsong.songIndex] try { weapiRequest("/api/cloud/del", { data: { songIds: [song.originalId], }, onload: (responses) => { ULsong.Upgrader.onUpgradeSucess(ULsong.songIndex) }, }) } catch (e) { console.error(e); ULsong.Upgrader.onUpgradeFail(ULsong.songIndex) } } onUpgradeFail(songIndex) { let song = this.songs[songIndex] showTips(`${song.name} 音质提升失败`, 2) this.onUpgradeFinnsh(songIndex) } onUpgradeSucess(songIndex) { let song = this.songs[songIndex] showTips(`${song.name} 音质提升成功`, 1) song.upgraded = true let btnUpgrade = song.tablerow.querySelector('button') btnUpgrade.innerHTML = '已提升' btnUpgrade.disabled = 'disabled' this.onUpgradeFinnsh(songIndex) } onUpgradeFinnsh(songIndex) { if (this.batchUpgrade.working) { let batchSongIdx = this.batchUpgrade.songIndexs.indexOf(songIndex) if (batchSongIdx + this.batchUpgrade.threadCount < this.batchUpgrade.songIndexs.length) { this.upgradeSong(this.batchUpgrade.songIndexs[batchSongIdx + this.batchUpgrade.threadCount]) } else { this.batchUpgrade.finnishThread += 1 if (this.batchUpgrade.finnishThread == this.batchUpgrade.threadCount) { this.batchUpgrade.working = false this.renderFilterInfo() showTips('歌曲提升完成', 1) } } } else { this.renderFilterInfo() } } } //本地上传 let btnLocalUpload = createBigButton('云盘本地上传', editArea, 2) btnLocalUpload.addEventListener('click', ShowLocalUploadPopUp) function ShowLocalUploadPopUp() { Swal.fire({ title: '云盘本地上传', html: `
`, confirmButtonText: '上传', showCloseButton: true, preConfirm: (level) => { let files = document.getElementById('song-file').files if (files.length == 0) return Swal.showValidationMessage('请选择文件') return { files: files, needFillInfo: document.getElementById('need-fill-info-radio').checked, } }, }) .then(result => { if (result.isConfirmed) { new LocalUpload().start(result.value) } }) } class LocalUpload { start(config) { this.files = config.files this.needFillInfo = config.needFillInfo this.task = [] this.currentIndex = 0 this.failIndexs = [] for (let i = 0; i < config.files.length; i++) { let file = config.files[i] let fileName = file.name let song = { id: -2, songFile: file, fileFullName: fileName, title: fileName.slice(0, fileName.lastIndexOf('.')), artist: '未知', album: '未知', size: file.size, ext: fileName.slice(fileName.lastIndexOf('.') + 1), bitrate: 128 } this.task.push(song) } showTips(`开始获取文件中的标签信息`, 1) this.readFileTags(0) } readFileTags(songIndex) { if (songIndex >= this.task.length) { if (this.needFillInfo) { this.showFillSongInforBox() } else { this.localUploadPart1(0) } return } let fileData = this.task[songIndex].songFile new jsmediatags.Reader(fileData) .read({ onSuccess: (res) => { if (res.tags.title) this.task[songIndex].title = res.tags.title if (res.tags.artist) this.task[songIndex].artist = res.tags.artist if (res.tags.album) this.task[songIndex].album = res.tags.album this.readFileTags(songIndex + 1) }, onError: (error) => { this.readFileTags(songIndex + 1) } }); } showFillSongInforBox() { Swal.fire({ html: `
操作歌曲标题歌手专辑
`, confirmButtonText: '上传', allowOutsideClick: false, allowEscapeKey: false, showCloseButton: false, didOpen: () => { let container = Swal.getHtmlContainer() let tbody = container.querySelector('tbody') for (let i = 0; i < this.task.length; i++) { let tablerow = document.createElement('tr') tablerow.innerHTML = `${this.task[i].title}${this.task[i].artist}${this.task[i].album}` let btnEdit = tablerow.querySelector('.my-edit') btnEdit.addEventListener('click', () => { this.showEditInforBox(i) }) tbody.appendChild(tablerow) } }, }) .then(result => { if (result.isConfirmed) { this.localUploadPart1(0) } }) } showEditInforBox(songIndex) { Swal.fire({ title: this.task[songIndex].fileFullName, html: `
`, allowOutsideClick: false, allowEscapeKey: false, showCloseButton: false, confirmButtonText: '确定', preConfirm: () => { let songTitle = document.getElementById('text-title').value.trim() if (songTitle.length == 0) return Swal.showValidationMessage('歌名不能为空') return { title: songTitle, artist: document.getElementById('text-artist').value.trim(), album: document.getElementById('text-album').value.trim(), } }, }) .then((result) => { if (result.isConfirmed) { this.task[songIndex].title = result.value.title this.task[songIndex].artist = result.value.artist this.task[songIndex].album = result.value.album this.showFillSongInforBox() } }) } localUploadPart1(songindex) { let self = this let song = self.task[songindex] let reader = new FileReader() let chunkSize = 1024 * 1024 let loaded = 0 let md5sum = unsafeWindow.CryptoJS.algo.MD5.create() showTips(`(1/5)${song.title} 正在获取文件MD5值`, 1) reader.onload = function (e) { md5sum.update(unsafeWindow.CryptoJS.enc.Latin1.parse(reader.result)); loaded += e.loaded; if (loaded < song.size) { readBlob(loaded); } else { showTips(`(1/5)${song.title} 已计算文件MD5值`, 1) song.md5 = md5sum.finalize().toString() try { weapiRequest("/api/cloud/upload/check", { data: { songId: 0, md5: song.md5, length: song.size, ext: song.ext, version: 1, bitrate: song.bitrate, }, onload: (res1) => { console.log(song.title, '1.检查资源', res1) if (res1.code != 200) { console.error(song.title, '1.检查资源', res1) self.uploadFail() return } song.cloudId = res1.songId song.needUpload = res1.needUpload weapiRequest("/api/nos/token/alloc", { data: { filename: song.title, length: song.size, ext: song.ext, type: 'audio', bucket: 'jd-musicrep-privatecloud-audio-public', local: false, nos_product: 3, md5: song.md5 }, onload: (res2) => { if (res2.code != 200) { console.error(song.title, '2.获取令牌', res2) self.uploadFail() return } song.resourceId = res2.result.resourceId song.token = res2.result.token song.objectKey = res2.result.objectKey showTips(`(3/5)${song.title} 开始上传文件`, 1) console.log(song.title, '2.获取令牌', res2) if (res1.needUpload) { self.localUploadFile(songindex, 0) } else { song.expireTime = Date.now() + 60000 self.localUploadPart2(songindex) } }, onerror: (res) => { console.error(song.title, '2.获取令牌', res) self.uploadFail() } }); }, onerror: (res) => { console.error(song.title, '1.检查资源', res) self.uploadFail() } }) } catch (e) { console.error(e); self.uploadFail() } } } readBlob(0); function readBlob(offset) { let blob = song.songFile.slice(offset, offset + chunkSize); reader.readAsBinaryString(blob); } } localUploadFile(songindex, offset, context = null) { let self = this let song = self.task[songindex] try { let complete = offset + uploadChunkSize > song.size let url = `http://45.127.129.8/jd-musicrep-privatecloud-audio-public/${encodeURIComponent(song.objectKey)}?offset=${offset}&complete=${String(complete)}&version=1.0` if (context) url += `&context=${context}` GM_xmlhttpRequest({ method: "POST", url: url, headers: { 'x-nos-token': song.token, 'Content-MD5': song.md5, 'Content-Type': 'audio/mpeg', }, data: song.songFile.slice(offset, offset + uploadChunkSize), onload: (response3) => { let res = JSON.parse(response3.response) if (complete) { console.log(song.title, '2.5.上传文件完成', res) showTips(`(3.5/5)${song.title} 上传文件完成`, 1) song.expireTime = Date.now() + 60000 self.localUploadPart2(songindex) } else { showTips(`(3.4/5)${song.title} 正在上传${fileSizeDesc(res.offset)}/${fileSizeDesc(song.size)}`, 1) self.localUploadFile(songindex, res.offset, res.context) } }, onerror: (response3) => { console.error(song.title, '文件上传时失败', response3) self.uploadFail() }, }); } catch (e) { console.error(e); self.uploadFail() } } localUploadPart2(songindex) { let self = this let song = self.task[songindex] try { weapiRequest("/api/upload/cloud/info/v2", { data: { md5: song.md5, songid: song.cloudId, filename: song.fileFullName, song: song.title, album: song.album, artist: song.artist, bitrate: String(song.bitrate), resourceId: song.resourceId, }, onload: (res3) => { if (res3.code != 200) { if (song.expireTime < Date.now() || (res3.msg && res3.msg.includes('rep create failed'))) { console.error(song.title, '3.提交文件', res3) self.uploadFail() } else { console.log(song.title, '3.正在转码', res3) showTips(`(4/5)${song.title} 正在转码...`, 1) sleep(1000).then(() => { self.localUploadPart2(songindex) }) } return } console.log(song.title, '3.提交文件', res3) showTips(`(4/5)${song.title} 提交文件完成`, 1) //step4 发布 weapiRequest("/api/cloud/pub/v2", { data: { songid: res3.songId, }, onload: (res4) => { if (res4.code != 200 && res4.code != 201) { console.error(song.title, '4.发布资源', res4) self.uploadFail() return } //完成 showTips(`(5/5)${song.title} 上传完成`, 1) self.uploadSuccess() }, onerror: (res) => { console.error(song.title, '4.发布资源', res) self.uploadFail() } }) }, onerror: (res) => { console.error(song.title, '3.提交文件', res) self.uploadFail() } }); } catch (e) { console.error(e); self.uploadFail() } } uploadFail() { this.failIndexs.push(this.currentIndex) showTips(`${this.task[this.currentIndex].title}上传失败`, 2) this.uploadNext() } uploadSuccess() { this.uploadNext() } uploadNext() { this.currentIndex += 1 if (this.currentIndex >= this.task.length) { this.uploadFinnsh() } else { this.localUploadPart1(this.currentIndex) } } uploadFinnsh() { let msg = '上传完成' if (this.failIndexs.length > 0) { msg += ',以下文件上传失败:' msg += this.failIndexs.map(idx => this.task[idx].fileFullName).join() } showConfirmBox(msg) } } //限免VIP歌曲 let btnVIPfreeA = createBigButton('限免VIP歌曲A', editArea, 2) btnVIPfreeA.addEventListener('click', VIPfreeA) function VIPfreeA() { weapiRequest('/api/homepage/block/page', { data: { cursor: JSON.stringify({ offset: 0, blockCodeOrderList: ['HOMPAGE_BLOCK_VIP_RCMD'] }), refresh: true, extInfo: JSON.stringify({ refreshType: 1, abInfo: { 'hp-new-homepageV3.1': 't3' }, netstate: 1 }), }, onload: (res) => { //console.log(res) let songList = res.data.blocks[0].resourceIdList.map(item => { return { 'id': Number(item) } }) openVIPDownLoadPopup(songList, 'APP发现页「免费听VIP歌曲」的内容', 23) } }) } let btnVIPfreeB = createBigButton('限免VIP歌曲B', editArea, 2) btnVIPfreeB.addEventListener('click', VIPfreeB) function VIPfreeB() { weapiRequest('/api/v6/playlist/detail', { data: { id: 8402996200, n: 100000, s: 8, }, onload: (res) => { //console.log(res) let songList = res.playlist.trackIds.map(item => { return { 'id': Number(item.id) } }) openVIPDownLoadPopup(songList, '歌单「会员雷达」的内容', 22) } }) } function openVIPDownLoadPopup(songIds, footer, trialMode) { Swal.fire({ title: '限免VIP歌曲', showCloseButton: true, showConfirmButton: false, width: 800, html: `
操作歌曲标题歌手时长大小
`, footer: footer + ',只有标准(128k)音质Github', didOpen: () => { let container = Swal.getHtmlContainer() let footer = Swal.getFooter() let tbody = container.querySelector('tbody') weapiRequest("/api/v3/song/detail", { data: { c: JSON.stringify(songIds) }, onload: function (content) { //console.log(content) let songlen = content.songs.length let privilegelen = content.privileges.length for (let i = 0; i < songlen; i++) { for (let j = 0; j < privilegelen; j++) { if (content.songs[i].id == content.privileges[j].id) { //生成表格 if (content.privileges[j].cs) { //移除云盘已存在歌曲 break } let songArtist = content.songs[i].ar ? content.songs[i].ar.map(ar => `${ar.name}`).join() : '' let songTitle = content.songs[i].name let filename = nameFileWithoutExt(songTitle, songArtist, 'artist-title') let tablerow = document.createElement('tr') tablerow.innerHTML = `${content.songs[i].name}${songArtist}${duringTimeDesc(content.songs[i].dt || 0)}${fileSizeDesc(content.songs[i].l.size)}` let btnDL = tablerow.querySelector('.mydl') btnDL.addEventListener('click', () => { TrialDownLoad(content.songs[i].id, trialMode, filename) }) let btnUL = tablerow.querySelector('.myul') btnUL.addEventListener('click', () => { let songItem = { api: { url: '/api/song/enhance/player/url/v1', data: { ids: JSON.stringify([content.songs[i].id]), trialMode: trialMode, level: 'exhigh', encodeType: 'mp3' } }, id: content.songs[i].id, title: content.songs[i].name, artist: getArtistTextInSongDetail(content.songs[i]), album: getAlbumTextInSongDetail(content.songs[i]) } let ULobj = new ncmDownUpload([songItem], false) ULobj.startUpload() }) tbody.appendChild(tablerow) break } } } }, }) }, }) } function TrialDownLoad(songId, trialMode, filename) { weapiRequest("/api/song/enhance/player/url/v1", { data: { immerseType: 'ste', trialMode: trialMode, ids: JSON.stringify([songId]), level: 'exhigh', encodeType: 'mp3' }, onload: (content) => { //console.log(content) if (content.data[0].url != null) { GM_download({ url: content.data[0].url, name: filename + '.' + content.data[0].type.toLowerCase(), onload: function () { // }, onerror: function (e) { console.error(e) showTips('下载失败', 2) } }) } else { showTips('下载失败', 2) } } }) } //云盘导出 let btnExport = createBigButton('云盘导出', editArea, 2) btnExport.addEventListener('click', openExportPopup) function openExportPopup() { Swal.fire({ title: '云盘导出', showCloseButton: true, html: `
`, footer: '过滤条件取交集', confirmButtonText: '导出', preConfirm: () => { return [ document.getElementById('text-artist') .value.trim(), document.getElementById('text-album') .value.trim(), document.getElementById('text-song') .value.trim(), document.getElementById('text-playlistid') .value ] }, }) .then((result) => { if (result.isConfirmed) { exportCloud(result.value) } }) } function exportCloud(filter) { showTips('开始导出', 1) if (filter[3]) { exportCloudByPlaylist(filter) } else { exportCloudSub(filter, { data: [] }, 0) } } function exportCloudSub(filter, config, offset) { weapiRequest('/api/v1/cloud/get', { data: { limit: 1000, offset: offset, }, onload: (res) => { showTips(`正在获取第${offset + 1}到${Math.min(offset + 1000, res.count)}首云盘歌曲信息`, 1) let matchSongs = [] res.data.forEach(song => { if (song.simpleSong.al && song.simpleSong.al.id > 0) { //已关联歌曲 if (filter[0].length > 0) { let flag = false for (let i = 0; i < song.simpleSong.ar.length; i++) { if (song.simpleSong.ar[i].name == filter[0]) { flag = true break } } if (!flag) { return } } if (filter[1].length > 0 && filter[1] != getAlbumTextInSongDetail(song.simpleSong)) { return } if (filter[2].length > 0 && filter[2] != song.simpleSong.name) { return } let songItem = { 'id': song.songId, 'size': song.fileSize, 'ext': song.fileName.split('.') .pop() .toLowerCase(), 'bitrate': song.bitrate, 'md5': null, } matchSongs.push(songItem) } else { //未关联歌曲 if (filter[0].length > 0 && song.artist != filter[0]) { return } if (filter[1].length > 0 && song.album != filter[1]) { return } if (filter[2].length > 0 && song.songName != filter[2]) { return } let songItem = { 'id': song.songId, 'size': song.fileSize, 'ext': song.fileName.split('.') .pop() .toLowerCase(), 'bitrate': song.bitrate, 'md5': null, 'name': song.songName, 'al': song.album, 'ar': song.artist, } matchSongs.push(songItem) } }) let ids = matchSongs.map(song => song.id) if (ids.length > 0) { weapiRequest("/api/song/enhance/player/url/v1", { data: { ids: JSON.stringify(ids), level: 'hires', encodeType: 'mp3', }, onload: (res2) => { //console.log(res2) if (res2.code != 200) { //重试 exportCloudSub(filter, config, offset) return } matchSongs.forEach(song => { let songId = song.id for (let i = 0; i < res2.data.length; i++) { if (res2.data[i].id == songId) { song.md5 = res2.data[i].md5 config.data.push(song) break } } }) if (res.hasMore) { exportCloudSub(filter, config, offset + 1000) } else { configToFile(config) } } }) } else { if (res.hasMore) { exportCloudSub(filter, config, offset + 1000) } else { configToFile(config) } } } }) } function exportCloudByPlaylist(filter) { weapiRequest('/api/v6/playlist/detail', { data: { id: filter[3], n: 100000, s: 8, }, onload: (res) => { //console.log(res) let trackIds = res.playlist.trackIds.map(item => { return item.id }) exportCloudByPlaylistSub(filter, trackIds, { data: [] }, 0) } }) } function exportCloudByPlaylistSub(filter, trackIds, config, offset) { let limit = 100 if (trackIds.length <= offset) { configToFile(config) return } showTips(`正在获取第${offset + 1}到${Math.min(offset + limit, trackIds.length)}首云盘歌曲信息`, 1) weapiRequest("/api/v1/cloud/get/byids", { data: { songIds: JSON.stringify(trackIds.slice(offset, offset + limit)) }, onload: function (res) { //console.log(res) let matchSongs = [] res.data.forEach(song => { if (song.simpleSong.al && song.simpleSong.al.id > 0) { //已关联歌曲 if (filter[0].length > 0) { let flag = false for (let i = 0; i < song.simpleSong.ar.length; i++) { if (song.simpleSong.ar[i].name == filter[0]) { flag = true break } } if (!flag) { return } } if (filter[1].length > 0 && filter[1] != getAlbumTextInSongDetail(song.simpleSong)) { return } if (filter[2].length > 0 && filter[2] != song.simpleSong.name) { return } let songItem = { 'id': song.songId, 'size': song.fileSize, 'ext': song.fileName.split('.') .pop() .toLowerCase(), 'bitrate': song.bitrate, 'md5': null, } matchSongs.push(songItem) } else { //未关联歌曲 if (filter[0].length > 0 && song.artist != filter[0]) { return } if (filter[1].length > 0 && song.album != filter[1]) { return } if (filter[2].length > 0 && song.songName != filter[2]) { return } let songItem = { 'id': song.songId, 'size': song.fileSize, 'ext': song.fileName.split('.') .pop() .toLowerCase(), 'bitrate': song.bitrate, 'md5': null, 'name': song.songName, 'al': song.album, 'ar': song.artist, } matchSongs.push(songItem) } }) let ids = matchSongs.map(song => song.id) if (ids.length > 0) { weapiRequest("/api/song/enhance/player/url/v1", { data: { ids: JSON.stringify(ids), level: 'hires', encodeType: 'mp3', }, onload: (res2) => { //console.log(res2) if (res2.code != 200) { //重试 exportCloudByPlaylistSub(filter, trackIds, config, offset) return } matchSongs.forEach(song => { let songId = song.id for (let i = 0; i < res2.data.length; i++) { if (res2.data[i].id == songId) { song.md5 = res2.data[i].md5 config.data.push(song) break } } }) exportCloudByPlaylistSub(filter, trackIds, config, offset + limit) } }) } else { exportCloudByPlaylistSub(filter, trackIds, config, offset + limit) } } }) } function configToFile(config) { let content = JSON.stringify(config) let temp = document.createElement('a'); let data = new Blob([content], { type: 'type/plain' }) let fileurl = URL.createObjectURL(data) temp.href = fileurl temp.download = '网易云云盘信息.json' temp.click() URL.revokeObjectURL(data); showTips(`导出云盘信息完成,共${config.data.length}首歌曲`, 1) } //云盘导入 let btnImport = createBigButton('云盘导入', editArea, 2) btnImport.addEventListener('click', openImportPopup) function openImportPopup() { Swal.fire({ title: '云盘导入', input: 'file', inputAttributes: { 'accept': 'application/json', 'aria-label': '选择文件' }, confirmButtonText: '导入' }) .then((result) => { if (result.isConfirmed) { importCloud(result.value) } }) } function importCloud(file) { let reader = new FileReader(); reader.readAsText(file); reader.onload = (e) => { let uploader = new Uploader(JSON.parse(e.target.result), true) uploader.start() } } } } //歌单页&&专辑页 if (location.href.includes('playlist') || location.href.includes('album')) { let pageType = location.href.includes('playlist') ? 'playlist' : 'album' let operationArea = document.querySelector('#content-operation') let listId = new URLSearchParams(location.search).get('id') if (operationArea && unsafeWindow.GUser.userId) { //批量下载 let btnBatchDownload = createBigButton('批量下载', operationArea, 1) btnBatchDownload.addEventListener('click', () => { ShowBatchDLPopUp(pageType) }) function ShowBatchDLPopUp(pageType) { Swal.fire({ title: '批量下载', customClass: { input: 'f-rdi', }, html: `
`, confirmButtonText: '开始下载', showCloseButton: true, footer: '请将 TamperMonkey 插件设置中的 下载模式 设置为 浏览器 API 并将 /\.(mp3|flac|lrc)$/ 添加进 文件扩展名白名单 以保证能正常下载。Github', focusConfirm: false, preConfirm: (level) => { let container = Swal.getHtmlContainer() return { free: container.querySelector('#cb-fee0').checked, VIP: container.querySelector('#cb-fee1').checked, pay: container.querySelector('#cb-fee4').checked, lowFree: container.querySelector('#cb-fee8').checked, skipCloud: container.querySelector('#cb-skipcloud').checked, downloadLyric: container.querySelector('#cb-dlLyric').checked, level: container.querySelector('#level-select').value, out: container.querySelector('#out-select').value, folder: container.querySelector('#folder-select').value, threadCount: Number(container.querySelector('#thread-count-select').value), listType: pageType, action: 'batchDownload' } } }).then(res => { fetchSongList(res.value) }) } //批量转存云盘 let btnBatchUpload = createBigButton('批量转存云盘', operationArea, 1) btnBatchUpload.addEventListener('click', () => { ShowBatchDLULPopUp(pageType) }) function ShowBatchDLULPopUp(pageType) { Swal.fire({ title: '批量转存云盘', html: `
免费歌曲
`, confirmButtonText: '开始转存', showCloseButton: true, footer: 'Github', focusConfirm: false, preConfirm: (level) => { return [ document.getElementById('cb-fee0').checked, document.getElementById('cb-fee1').checked, document.getElementById('cb-fee4').checked, document.getElementById('cb-fee8').checked, document.getElementById('level-select').value, document.getElementById('out-select').value, ] } }).then(res => { if (res.value[0].length == 0) return let config = { ree: res.value[0], VIP: res.value[1], pay: res.value[2], lowFree: res.value[3], skipCloud: true, level: res.value[4], out: res.value[5], listType: pageType, action: 'batchUpload' } fetchSongList(config) }) } function fetchSongList(config) { if (config.listType == 'playlist') { weapiRequest("/api/v6/playlist/detail", { data: { id: listId, n: 100000, s: 8, }, onload: (content) => { //console.log(content) let songList = [] let tracklen = content.playlist.tracks.length let privilegelen = content.privileges.length for (let i = 0; i < tracklen; i++) { for (let j = 0; j < privilegelen; j++) { if (content.playlist.tracks[i].id == content.privileges[j].id) { if (content.privileges[j].st < 0 || content.privileges[j].plLevel == 'none') break if (content.privileges[j].cs && config.skipCloud) break if (content.privileges[j].fee == 0 && !config.free) break if (content.privileges[j].fee == 1 && !config.VIP) break if (content.privileges[j].fee == 4 && !config.pay) break if (content.privileges[j].fee == 8 && !config.lowFree) break let api = { url: '/api/song/enhance/player/url/v1', data: { ids: JSON.stringify([content.playlist.tracks[i].id]), level: config.level, encodeType: 'mp3' } } if (content.privileges[j].fee == 0 && (levelWeight[content.privileges[j].plLevel] || 99) < (levelWeight[content.privileges[j].dlLevel] || -1)) api = { url: '/api/song/enhance/download/url/v1', data: { id: content.playlist.tracks[i].id, level: config.level, encodeType: 'mp3' } } let songItem = { api: api, id: content.playlist.tracks[i].id, title: content.playlist.tracks[i].name, artist: getArtistTextInSongDetail(content.playlist.tracks[i]), album: getAlbumTextInSongDetail(content.playlist.tracks[i]) } songList.push(songItem) break } } } if (content.playlist.trackCount > content.playlist.tracks.length) { showTips(`大歌单,开始分批获取${content.playlist.trackCount}首歌信息`, 1) let trackIds = content.playlist.trackIds.map(item => { return { 'id': item.id } }) fetchplaylistSongInfo(trackIds, 0, [], config) } else { if (config.action == 'batchUpload') { startUploadSongs(songList, config) } else if (config.action == 'batchDownload') { startDownloadSongs(songList, config) } } } }) } else if (config.listType == 'album') { weapiRequest(`/api/v1/album/${listId}`, { onload: (content) => { //console.log(content) let songList = [] for (let i = 0; i < content.songs.length; i++) { if (content.songs[i].privilege.st < 0 || content.songs[i].privilege.plLevel == 'none') continue if (content.songs[i].privilege.cs && config.skipCloud) continue if (content.songs[i].privilege.fee == 0 && !config.free) continue if (content.songs[i].privilege.fee == 1 && !config.VIP) continue if (content.songs[i].privilege.fee == 4 && !config.pay) continue if (content.songs[i].privilege.fee == 8 && !config.lowFree) continue let api = { url: '/api/song/enhance/player/url/v1', data: { ids: JSON.stringify([content.songs[i].id]), level: config.level, encodeType: 'mp3' } } if (content.songs[i].privilege.fee == 0 && (levelWeight[content.songs[i].privilege.plLevel] || 99) < (levelWeight[content.songs[i].privilege.dlLevel] || -1)) api = { url: '/api/song/enhance/download/url/v1', data: { id: content.songs[i].id, level: config.level, encodeType: 'mp3' } } let songItem = { api: api, id: content.songs[i].id, title: content.songs[i].name, artist: getArtistTextInSongDetail(content.songs[i]), album: getAlbumTextInSongDetail(content.songs[i]) } songList.push(songItem) } if (config.action == 'batchUpload') { startUploadSongs(songList, config) } else if (config.action == 'batchDownload') { startDownloadSongs(songList, config) } } }) } } function fetchplaylistSongInfo(trackIds, startIndex, songList, config) { if (startIndex >= trackIds.length) { if (config.action == 'batchUpload') { startUploadSongs(songList, config) } else if (config.action == 'batchDownload') { startDownloadSongs(songList, config) } return } weapiRequest("/api/v3/song/detail", { data: { c: JSON.stringify(trackIds.slice(startIndex, startIndex + 1000)) }, onload: function (content) { let songlen = content.songs.length let privilegelen = content.privileges.length for (let i = 0; i < songlen; i++) { for (let j = 0; j < privilegelen; j++) { if (content.songs[i].id == content.privileges[j].id) { if (content.privileges[j].st < 0 || content.privileges[j].plLevel == 'none') break if (content.privileges[j].cs && config.skipCloud) break if (content.privileges[j].fee == 0 && !config.free) break if (content.privileges[j].fee == 1 && !config.VIP) break if (content.privileges[j].fee == 4 && !config.pay) break if (content.privileges[j].fee == 8 && !config.lowFree) break let api = { url: '/api/song/enhance/player/url/v1', data: { ids: JSON.stringify([content.songs[i].id]), level: config.level, encodeType: 'mp3' } } if (content.privileges[j].fee == 0 && (levelWeight[content.privileges[j].plLevel] || 99) < (levelWeight[content.privileges[j].dlLevel] || -1)) api = { url: '/api/song/enhance/download/url/v1', data: { id: content.songs[i].id, level: config.level, encodeType: 'mp3' } } let songItem = { api: api, id: content.songs[i].id, title: content.songs[i].name, artist: getArtistTextInSongDetail(content.songs[i]), album: getAlbumTextInSongDetail(content.songs[i]) } songList.push(songItem) break } } } fetchplaylistSongInfo(trackIds, startIndex + content.songs.length, songList, config) } }) } function startDownloadSongs(songList, config) { if (songList.length == 0) { showConfirmBox('没有可下载的歌曲') return } Swal.fire({ title: '批量下载', allowOutsideClick: false, allowEscapeKey: false, showCloseButton: false, showConfirmButton: false, width: 800, html: `
歌曲标题歌手专辑音质大小进度
`, footer: '
', didOpen: () => { let container = Swal.getHtmlContainer() let tbodyDOM = container.querySelector('tbody') let threadList = [] for (let i = 0; i < config.threadCount; i++) { let trDOM = document.createElement('tr') tbodyDOM.appendChild(trDOM) threadList.push({ tableRowDOM: trDOM, working: true }) } config.finnshCount = 0 config.errorSongTitles = [] config.taskCount = songList.length config.threadList = threadList for (let i = 0; i < config.threadCount; i++) { downloadSongSub(i, songList, config) } }, }) } function downloadSongSub(threadIndex, songList, config) { let song = songList.shift() let tableRowDOM = config.threadList[threadIndex].tableRowDOM if (song == undefined) { config.threadList[threadIndex].working = false let allFinnsh = true for (let i = 0; i < config.threadCount; i++) { if (config.threadList[i].working) { allFinnsh = false break } } if (allFinnsh) { let finnshText = '下载完成' if (config.errorSongTitles.length > 0) { finnshText = `下载完成。以下${config.errorSongTitles.length}首歌曲下载失败: ${config.errorSongTitles.join()}` } Swal.update({ allowOutsideClick: true, allowEscapeKey: true, showCloseButton: true, showConfirmButton: true, html: finnshText, }) } return } tableRowDOM.innerHTML = `${song.title}${song.artist}${song.album}` let levelText = tableRowDOM.querySelector('.my-level') let sizeText = tableRowDOM.querySelector('.my-size') let prText = tableRowDOM.querySelector('.my-pr') try { weapiRequest(song.api.url, { data: song.api.data, onload: (content) => { let resData = content.data[0] || content.data if (resData.url != null) { let fileName = nameFileWithoutExt(song.title, song.artist, config.out) let fileFullName = fileName + '.' + resData.type.toLowerCase() let folder = '' if (config.folder != 'none' && song.artist.length > 0) { folder = song.artist + '/' } if (config.folder == 'artist-album' && song.album.length > 0) { folder += song.album + '/' } fileFullName = folder + fileFullName let dlUrl = resData.url levelText.innerHTML = levelDesc(resData.level) sizeText.innerHTML = fileSizeDesc(resData.size) GM_download({ url: dlUrl, name: fileFullName, onprogress: function (e) { prText.innerHTML = `${fileSizeDesc(e.loaded)}` }, onload: function () { config.finnshCount += 1 Swal.getFooter().innerHTML = `已完成: ${config.finnshCount} 总共: ${config.taskCount}` prText.innerHTML = `完成` if (config.downloadLyric) { downloadSongLyric(song.id, folder + fileName) } downloadSongSub(threadIndex, songList, config) }, onerror: function (e) { if (song.retry) { prText.innerHTML = `下载出错` config.errorSongTitles.push(song.title) } else { prText.innerHTML = `下载出错\t稍后重试` song.retry = true songList.push(song) } console.error(e) downloadSongSub(threadIndex, songList, config) } }); } else { showTips(`${song.title}\t无法下载`, 2) prText.innerHTML = `无法下载` config.errorSongTitles.push(song.title) downloadSongSub(threadIndex, songList, config) } }, onerror: (res) => { console.error(res) if (song.retry) { prText.innerHTML = `下载出错` config.errorSongTitles.push(song.title) } else { prText.innerHTML = `下载出错\t稍后重试` song.retry = true songList.push(song) } downloadSongSub(threadIndex, songList, config) } }) } catch (e) { console.error(e); if (song.retry) { prText.innerHTML = `下载出错` config.errorSongTitles.push(song.title) } else { prText.innerHTML = `下载出错\t稍后重试` song.retry = true songList.push(song) } downloadSongSub(threadIndex, songList, config) } } function startUploadSongs(songList, config) { if (songList.length == 0) { showConfirmBox('没有可上传的歌曲') return } showTips(`开始下载上传${songList.length}首歌曲`, 1) let ULobj = new ncmDownUpload(songList, true, null, null, config.out) ULobj.startUpload() } function downloadSongLyric(songId, fileName) { weapiRequest('/api/song/lyric/v1', { data: { id: songId, cp: false, tv: 0, lv: 0, rv: 0, kv: 0, yv: 0, ytv: 0, yrv: 0, }, onload: (content) => { const LyricObj = handleLyric(content) if (LyricObj.orilrc.parsedLyric.length == 0) return const LyricItem = LyricObj.oritlrc || LyricObj.orilrc saveContentAsFile(LyricItem.lyric, fileName + '.lrc') } }) } //歌单排序 if (pageType == 'playlist') { let creatorhomeURL = document.head.querySelector("[property~='music:creator'][content]")?.content let creatorId = new URLSearchParams(new URL(creatorhomeURL).search).get('id') if (creatorId == unsafeWindow.GUser.userId) { //歌单排序 let btnPlaylistSort = createBigButton('歌单排序', operationArea, 1) btnPlaylistSort.addEventListener('click', () => { ShowPLSortPopUp() }) function ShowPLSortPopUp() { Swal.fire({ title: '歌单内歌曲排序', input: 'select', inputOptions: ['发行时间降序', '发行时间升序', '红心数量降序', '红心数量升序', '评论数量降序', '评论数量升序'], inputPlaceholder: '选择排序方式', confirmButtonText: '开始排序', showCloseButton: true, focusConfirm: false, inputValidator: (way) => { if (!way) { return '请选择排序方式' } }, }).then(res => { if (!res.isConfirmed) return if (res.value == 0) { PlaylistTimeSort(listId, true) } else if (res.value == 1) { PlaylistTimeSort(listId, false) } else if (res.value == 2) { PlaylistCountSort(listId, true, 'Red') } else if (res.value == 3) { PlaylistCountSort(listId, false, 'Red') } else if (res.value == 4) { PlaylistCountSort(listId, true, 'Comment') } else if (res.value == 5) { PlaylistCountSort(listId, false, 'Comment') } }) } function PlaylistTimeSort(playlistId, descending) { showTips(`正在获取歌单内歌曲信息`, 1) weapiRequest("/api/v6/playlist/detail", { data: { id: playlistId, n: 100000, s: 8, }, onload: (content) => { let songList = [] let tracklen = content.playlist.tracks.length for (let i = 0; i < tracklen; i++) { let songItem = { id: content.playlist.tracks[i].id, publishTime: content.playlist.tracks[i].publishTime, albumId: content.playlist.tracks[i].al.id, cd: content.playlist.tracks[i].cd ? Number(content.playlist.tracks[i].cd.split(' ')[0]) : 0, no: content.playlist.tracks[i].no } songList.push(songItem) } if (content.playlist.trackCount > content.playlist.tracks.length) { showTips(`大歌单,开始分批获取${content.playlist.trackCount}首歌信息`, 1) let trackIds = content.playlist.trackIds.map(item => { return { 'id': item.id } }) PlaylistTimeSortFetchAll(playlistId, descending, trackIds, 0, songList) } else { PlaylistTimeSortFetchAllPublishTime(playlistId, descending, 0, songList, {}) } } }) } function PlaylistTimeSortFetchAll(playlistId, descending, trackIds, startIndex, songList) { if (startIndex >= trackIds.length) { PlaylistTimeSortFetchAllPublishTime(playlistId, descending, 0, songList, {}) return } weapiRequest("/api/v3/song/detail", { data: { c: JSON.stringify(trackIds.slice(startIndex, startIndex + 1000)) }, onload: function (content) { let songlen = content.songs.length for (let i = 0; i < songlen; i++) { let songItem = { id: content.songs[i].id, publishTime: content.songs[i].publishTime, albumId: content.songs[i].al.id, cd: content.songs[i].cd ? Number(content.songs[i].cd.split(' ')[0]) : 0, no: content.songs[i].no } songList.push(songItem) } PlaylistTimeSortFetchAll(playlistId, descending, trackIds, startIndex + content.songs.length, songList) } }) } function PlaylistTimeSortFetchAllPublishTime(playlistId, descending, index, songList, aldict) { if (index >= songList.length) { PlaylistTimeSortSongs(playlistId, descending, songList) return } if (index == 0) showTips('开始获取歌曲专辑发行时间') if (index % 10 == 9) showTips(`正在获取歌曲专辑发行时间(${index + 1}/${songList.length})`) let albumId = songList[index].albumId if (albumId <= 0) { PlaylistTimeSortFetchAllPublishTime(playlistId, descending, index + 1, songList, aldict) return } if (aldict[albumId]) { songList[index].publishTime = aldict[albumId] PlaylistTimeSortFetchAllPublishTime(playlistId, descending, index + 1, songList, aldict) return } weapiRequest(`/api/v1/album/${albumId}`, { onload: function (content) { let publishTime = content.album.publishTime aldict[albumId] = publishTime songList[index].publishTime = publishTime PlaylistTimeSortFetchAllPublishTime(playlistId, descending, index + 1, songList, aldict) } }) } function PlaylistTimeSortSongs(playlistId, descending, songList) { songList.sort((a, b) => { if (a.publishTime != b.publishTime) { if (descending) { return b.publishTime - a.publishTime } else { return a.publishTime - b.publishTime } } else if (a.albumId != b.albumId) { if (descending) { return b.albumId - a.albumId } else { return a.albumId - b.albumId } } else if (a.cd != b.cd) { return a.cd - b.cd } else if (a.no != b.no) { return a.no - b.no } return a.id - b.id }) let trackIds = songList.map(song => song.id) weapiRequest("/api/playlist/manipulate/tracks", { data: { pid: playlistId, trackIds: JSON.stringify(trackIds), op: 'update', }, onload: function (content) { //console.log(content) if (content.code == 200) { showConfirmBox('排序完成') } else { showConfirmBox('排序失败,' + content) } } }) } function PlaylistCountSort(playlistId, descending, way) { showTips(`正在获取歌单内歌曲信息`, 1) weapiRequest("/api/v6/playlist/detail", { data: { id: playlistId, n: 100000, s: 8, }, onload: (content) => { let songList = content.playlist.trackIds.map(item => { return { 'id': item.id, 'count': 0, } }) let trackIds = content.playlist.trackIds.map(item => { return item.id }) if (way == 'Red') { PlaylistCountSortFetchRedCount(playlistId, songList, 0, descending) } else if (way == 'Comment') { PlaylistCountSortFetchCommentCount(playlistId, songList, trackIds, 0, descending) } } }) } function PlaylistCountSortFetchRedCount(playlistId, songList, index, descending) { if (index >= songList.length) { PlaylistCountSortSongs(playlistId, descending, songList) return } if (index == 0) showTips('开始获取歌曲红心数量') if (index % 10 == 9) showTips(`正在获取歌曲红心数量(${index + 1}/${songList.length})`) weapiRequest("/api/song/red/count", { data: { songId: songList[index].id, }, onload: function (content) { songList[index].count = content.data.count PlaylistCountSortFetchRedCount(playlistId, songList, index + 1, descending) }, }); } function PlaylistCountSortFetchCommentCount(playlistId, songList, trackIds, index, descending) { if (index >= songList.length) { PlaylistCountSortSongs(playlistId, descending, songList) return } if (index == 0) showTips('开始获取歌曲评论数量') else showTips(`正在获取歌曲评论数量(${index + 1}/${songList.length})`) weapiRequest("/api/resource/commentInfo/list", { data: { resourceType: "4", resourceIds: JSON.stringify(trackIds.slice(index, index + 1000)), }, onload: function (content) { content.data.forEach(item => { let songId = item.resourceId for (let i = 0; i < songList.length; i++) { if (songList[i].id == songId) { songList[i].count = item.commentCount break } } }) PlaylistCountSortFetchCommentCount(playlistId, songList, trackIds, index + 1000, descending) }, }); } function PlaylistCountSortSongs(playlistId, descending, songList) { songList.sort((a, b) => { if (a.count != b.count) { if (descending) { return b.count - a.count } else { return a.count - b.count } } return a.id - b.id }) let trackIds = songList.map(song => song.id) weapiRequest("/api/playlist/manipulate/tracks", { data: { pid: playlistId, trackIds: JSON.stringify(trackIds), op: 'update', }, onload: function (content) { if (content.code == 200) { showConfirmBox('排序完成') } else { showConfirmBox('排序失败') } } }) } } } //显示脏标 else if (pageType == 'album') { let topblk = document.querySelector('.topblk') if (topblk) { weapiRequest(`/api/v1/album/${listId}`, { onload: (content) => { console.log(content) topblk.innerHTML += `

专辑类型:${content.album.type} ${content.album.subType}

` if ((content.album.mark & songMark.explicit) == songMark.explicit) { topblk.innerHTML += `

🅴:内容含有不健康因素

` } if (content.album.blurPicUrl) { topblk.innerHTML += `

专辑封面原图

` } } }) } } } } //高音质播放 GM_registerMenuCommand(`优先试听音质`, setLevel) function setLevel() { Swal.fire({ title: '优先试听音质', input: 'select', inputOptions: levelOptions, inputValue: GM_getValue('DEFAULT_LEVEL', defaultOfDEFAULT_LEVEL), confirmButtonText: '确定', showCloseButton: true, footer: 'Github', }) .then(result => { if (result.isConfirmed) { GM_setValue('DEFAULT_LEVEL', result.value) } }) } if (unsafeWindow.self == unsafeWindow.top) { unsafeWindow.myhook = ah.proxy({ onResponse: (response, handler) => { if (response.config.url.includes('/weapi/song/enhance/player/url/v1')) { let content = JSON.parse(response.response) let songId = content.data[0].id let targetLevel = GM_getValue('DEFAULT_LEVEL', defaultOfDEFAULT_LEVEL) if (content.data[0].type.toLowerCase() !== "mp3" && content.data[0].type.toLowerCase() !== "m4a") { content.data[0].type = 'mp3' } if (content.data[0].url) { if (content.data[0].level == 'standard') { if (targetLevel != 'standard') { let apiData = { '/api/song/enhance/player/url/v1': JSON.stringify({ ids: JSON.stringify([songId]), level: targetLevel, encodeType: 'mp3' }), } if (content.data[0].fee == 0) { apiData['/api/song/enhance/download/url/v1'] = JSON.stringify({ id: songId, level: levelWeight[targetLevel] > levelWeight.hires ? 'hires' : targetLevel, encodeType: 'mp3' }) } weapiRequest("/api/batch", { data: apiData, onload: (res) => { let songUrl = res['/api/song/enhance/player/url/v1'].data[0].url let songLevel = res["/api/song/enhance/player/url/v1"].data[0].level if (res['/api/song/enhance/download/url/v1']) { let songDLLevel = res["/api/song/enhance/download/url/v1"].data.level if (res["/api/song/enhance/download/url/v1"].data.url && (levelWeight[songDLLevel] || -1) > (levelWeight[songLevel] || 99)) { songUrl = res["/api/song/enhance/download/url/v1"].data.url songLevel = songDLLevel } } if (songLevel != 'standard') { content.data[0].url = songUrl player.tipPlay(levelDesc(songLevel) + '音质') } response.response = JSON.stringify(content) handler.next(response) }, onerror: (res) => { console.error('/api/batch', apiData, res) response.response = JSON.stringify(content) handler.next(response) } }) } else { response.response = JSON.stringify(content) handler.next(response) } } else { player.tipPlay(levelDesc(content.data[0].level) + '音质(云盘文件)') response.response = JSON.stringify(content) handler.next(response) } } else { response.response = JSON.stringify(content) handler.next(response) } } else { handler.next(response) } } }, unsafeWindow) } })();