网易云音乐显示完整歌单

解除歌单歌曲展示数量限制 & 播放列表 1000 首上限

  1. // ==UserScript==
  2. // @name 网易云音乐显示完整歌单
  3. // @namespace https://github.com/nondanee
  4. // @version 1.4.13
  5. // @description 解除歌单歌曲展示数量限制 & 播放列表 1000 首上限
  6. // @author nondanee
  7. // @match *://music.163.com/*
  8. // @icon https://s1.music.126.net/style/favicon.ico
  9. // @grant none
  10. // @run-at document-start
  11. // ==/UserScript==
  12.  
  13. (() => {
  14. if (window.top === window.self) {
  15. const observe = () => {
  16. try {
  17. const callback = () => document.contentFrame.dispatchEvent(new Event('songchange'))
  18. const observer = new MutationObserver(callback)
  19. observer.observe(document.querySelector('.m-playbar .words'), { childList: true })
  20. } catch (_) {}
  21. }
  22. window.addEventListener('load', observe, false)
  23. return
  24. }
  25.  
  26. const locate = (object, pattern) => {
  27. for (const key in object) {
  28. const value = object[key]
  29. if (!Object.prototype.hasOwnProperty.call(object, key) || !value) continue
  30. switch (typeof value) {
  31. case 'function': {
  32. if (String(value).match(pattern)) return [key]
  33. break
  34. }
  35. case 'object': {
  36. const path = locate(value, pattern)
  37. if (path) return [key].concat(path)
  38. break
  39. }
  40. }
  41. }
  42. }
  43.  
  44. const findMethod = (object, pattern) => {
  45. const path = locate(object, pattern)
  46. if (!path) throw new Error('MethodNotFound')
  47. let poiner = object
  48. const last = path.pop()
  49. path.forEach(key => poiner = poiner[key])
  50. const origin = poiner[last]
  51. return {
  52. origin,
  53. override: (value) => {
  54. value.toString = () => origin.toString()
  55. poiner[last] = value
  56. }
  57. }
  58. }
  59.  
  60. const cloneEvent = (event) => {
  61. const copy = new event.constructor(event.type, event)
  62. // copy.target = event.target // 有问题
  63. Object.defineProperty(copy, 'target', { value: event.target })
  64. return copy
  65. }
  66.  
  67. const normalize = song => {
  68. song = { ...song, ...song.privilege }
  69. return {
  70. ...song,
  71. album: song.al,
  72. alias: song.alia || song.ala || [],
  73. artists: song.ar || [],
  74. commentThreadId: `R_SO_4_${song.id}`,
  75. copyrightId: song.cp,
  76. duration: song.dt,
  77. mvid: song.mv,
  78. position: song.no,
  79. ringtone: song.rt,
  80. status: song.st,
  81. pstatus: song.pst,
  82. version: song.v,
  83. songType: song.t,
  84. score: song.pop,
  85. transNames: song.tns || [],
  86. privilege: song.privilege,
  87. lyrics: song.lyrics
  88. }
  89. }
  90.  
  91. const zFill = (string = '', length = 2) => {
  92. string = String(string)
  93. while (string.length < length) string = '0' + string
  94. return string
  95. }
  96.  
  97. const formatDuration = duration => {
  98. const oneSecond = 1e3
  99. const oneMinute = 60 * oneSecond
  100. const result = []
  101.  
  102. Array(oneMinute, oneSecond)
  103. .reduce((remain, unit) => {
  104. const value = Math.floor(remain / unit)
  105. result.push(value)
  106. return remain - value * unit
  107. }, duration || 0)
  108.  
  109. return result
  110. .map(value => zFill(value, 2))
  111. .join(':')
  112. }
  113.  
  114. const TYPE = {
  115. SONG: '18',
  116. PLAYLIST: '13',
  117. }
  118.  
  119. const CACHE = window.COMPLETE_PLAYLIST_CACHE = {
  120. [TYPE.SONG]: {},
  121. [TYPE.PLAYLIST]: {}
  122. }
  123.  
  124. const interceptRequest = () => {
  125. if (window.getPlaylistDetail) return
  126.  
  127. const request = findMethod(window.nej, '\\.replace\\("api","weapi')
  128.  
  129. const Fetch = (url, options) => (
  130. new Promise((resolve, reject) =>
  131. request.origin(url, {
  132. ...options,
  133. cookie: true,
  134. method: 'GET',
  135. onerror: reject,
  136. onload: resolve,
  137. type: 'json'
  138. })
  139. )
  140. )
  141.  
  142. window.getPlaylistDetail = async (url, options) => {
  143. // const search = new URLSearchParams(options.data)
  144. // search.set('n', 0)
  145. // options.data = search.toString()
  146.  
  147. const data = await Fetch(url, options)
  148. const slice = 1000
  149.  
  150. const trackIds = (data.playlist || {}).trackIds || []
  151. const tracks = (data.playlist || {}).tracks || []
  152.  
  153. if (!trackIds.length || trackIds.length === tracks.length) return data
  154.  
  155. const missingTrackIds = trackIds.slice(tracks.length)
  156. const round = Math.ceil(missingTrackIds.length / slice)
  157.  
  158. const result = await Promise.all(
  159. Array(round).fill().map((_, index) => {
  160. const part = missingTrackIds.slice(index * slice).slice(0, slice).map(({ id }) => ({ id }))
  161. return Fetch('/api/v3/song/detail', { data: `c=${JSON.stringify(part)}` })
  162. })
  163. )
  164.  
  165. const songMap = {}
  166. const privilegeMap = {}
  167.  
  168. result.forEach(({ songs, privileges }) => {
  169. songs.forEach(_ => songMap[_.id] = _)
  170. privileges.forEach(_ => privilegeMap[_.id] = _)
  171. })
  172.  
  173. const missingTracks = missingTrackIds
  174. .map(({ id }) => ({ ...songMap[id], privilege: privilegeMap[id] }))
  175.  
  176. const missPrivileges = missingTracks
  177. .map(({ id }) => privilegeMap[id])
  178.  
  179. data.playlist.tracks = tracks.concat(missingTracks)
  180. data.privileges = (data.privileges || []).concat(missPrivileges)
  181.  
  182. CACHE[TYPE.PLAYLIST][data.playlist.id] = data.playlist.tracks
  183. .map(song => CACHE[TYPE.SONG][song.id] = normalize(song))
  184.  
  185. return data
  186. }
  187.  
  188. const overrideRequest = async (url, options) => {
  189. if (/\/playlist\/detail/.test(url)) {
  190. const { onload, onerror } = options
  191. return window.getPlaylistDetail(url, options).then(onload).catch(onerror)
  192. }
  193. return request.origin(url, options)
  194. }
  195.  
  196. request.override(overrideRequest)
  197. }
  198.  
  199. const handleSongChange = () => {
  200. try {
  201. const { track } = window.top.player.getPlaying()
  202. const { id, source, program } = track
  203. if (program) return
  204.  
  205. const base = 'span.ply'
  206. const attrs = `[data-res-id="${id}"][data-res-type="${TYPE.SONG}"]`
  207.  
  208. // player.addTo() 相同 id 不同 source 会被过滤
  209. // const { fid, fdata } = source
  210. // if (String(fid) !== TYPE.PLAYLIST) return
  211. // const attrs = `[data-res-id="${id}"][data-res-from="${fid}"][data-res-data="${fdata}"]`
  212.  
  213. document.querySelectorAll(base).forEach(node => {
  214. node.classList.remove('ply-z-slt')
  215. })
  216.  
  217. document.querySelectorAll(base + attrs).forEach(node => {
  218. node.classList.add('ply-z-slt')
  219. })
  220. } catch (_) {}
  221. }
  222.  
  223. const escapeHTML = string => (
  224. string.replace(
  225. /[&<>'"]/g,
  226. word =>
  227. ({
  228. '&': '&amp;',
  229. '<': '&lt;',
  230. '>': '&gt;',
  231. "'": '&#39;',
  232. '"': '&quot;',
  233. })[word] || word
  234. )
  235. )
  236.  
  237. const bindEvent = () => {
  238. const ACTIONS = new Set(['play', 'addto'])
  239.  
  240. const onClick = (event) => {
  241. const {
  242. resAction,
  243. resId,
  244. resType,
  245. resData,
  246. } = event.target.dataset
  247.  
  248. const data = (CACHE[resType] || {})[resId]
  249. if (!data) return
  250.  
  251. event.stopPropagation()
  252.  
  253. if (!ACTIONS.has(resAction)) {
  254. // 没有 privilege 冒泡后会报错
  255. document.body.dispatchEvent(cloneEvent(event))
  256. return
  257. }
  258.  
  259. const playlistId = Number(resType === TYPE.PLAYLIST ? resId : resData)
  260.  
  261. const list = (Array.isArray(data) ? data : [data])
  262. .map(song => ({
  263. ...song,
  264. source: {
  265. fdata: playlistId,
  266. fid: TYPE.PLAYLIST,
  267. link: `/playlist?id=${playlistId}&_hash=songlist-${song.id}`,
  268. title: '歌单',
  269. },
  270. }))
  271.  
  272. window.top.player.addTo(
  273. list,
  274. resAction === 'play' && resType === TYPE.PLAYLIST,
  275. resAction === 'play'
  276. )
  277. }
  278.  
  279. const body = document.querySelector('table tbody')
  280. const play = document.querySelector('#content-operation .u-btni-addply')
  281. const add = document.querySelector('#content-operation .u-btni-add')
  282.  
  283. if (play) play.addEventListener('click', onClick)
  284. if (add) add.addEventListener('click', onClick)
  285. if (body) body.addEventListener('click', onClick)
  286. }
  287.  
  288. const completePlaylist = async (id) => {
  289. const render = (song, index, playlist) => {
  290. const { album, artists, status, duration } = song
  291. const deletable = playlist.creator.userId === window.GUser.userId
  292. const durationText = formatDuration(duration)
  293. const artistText = artists.map(({ name }) => escapeHTML(name)).join('/')
  294. const annotation = escapeHTML(song.transNames[0] || song.alias[0] || '')
  295. const albumName = escapeHTML(album.name)
  296. const songName = escapeHTML(song.name)
  297.  
  298. return `
  299. <tr id="${song.id}${Date.now()}" class="${index % 2 ? '' : 'even'} ${status ? 'js-dis' : ''}">
  300. <td class="left">
  301. <div class="hd "><span data-res-id="${song.id}" data-res-type="18" data-res-action="play" data-res-from="13" data-res-data="${playlist.id}" class="ply ">&nbsp;</span><span class="num">${index + 1}</span></div>
  302. </td>
  303. <td>
  304. <div class="f-cb">
  305. <div class="tt">
  306. <div class="ttc">
  307. <span class="txt">
  308. <a href="#/song?id=${song.id}"><b title="${songName}${annotation ? ` - (${annotation})` : ''}">${songName}</b></a>
  309. ${annotation ? `<span title="${annotation}" class="s-fc8">${annotation ? ` - (${annotation})` : ''}</span>` : ''}
  310. ${song.mvid ? `<a href="#/mv?id=${song.mvid}" title="播放mv" class="mv">MV</a>` : ''}
  311. </span>
  312. </div>
  313. </div>
  314. </div>
  315. </td>
  316. <td class=" s-fc3">
  317. <span class="u-dur candel">${durationText}</span>
  318. <div class="opt hshow">
  319. <a class="u-icn u-icn-81 icn-add" href="javascript:;" title="添加到播放列表" hidefocus="true" data-res-type="18" data-res-id="${song.id}" data-res-action="addto" data-res-from="13" data-res-data="${playlist.id}"></a>
  320. <span data-res-id="${song.id}" data-res-type="18" data-res-action="fav" class="icn icn-fav" title="收藏"></span>
  321. <span data-res-id="${song.id}" data-res-type="18" data-res-action="share" data-res-name="${albumName}" data-res-author="${artistText}" data-res-pic="${album.picUrl}" class="icn icn-share" title="分享">分享</span>
  322. <span data-res-id="${song.id}" data-res-type="18" data-res-action="download" class="icn icn-dl" title="下载"></span>
  323. ${deletable ? `<span data-res-id="${song.id}" data-res-type="18" data-res-from="13" data-res-data="${playlist.id}" data-res-action="delete" class="icn icn-del" title="删除">删除</span>` : ''}
  324. </div>
  325. </td>
  326. <td>
  327. <div class="text" title="${artistText}">
  328. <span title="${artistText}">
  329. ${artists.map(({ id, name }) => `<a href="#/artist?id=${id}" hidefocus="true">${escapeHTML(name)}</a>`).join('/')}
  330. </span>
  331. </div>
  332. </td>
  333. <td>
  334. <div class="text">
  335. <a href="#/album?id=${album.id}" title="${albumName}">${albumName}</a>
  336. </div>
  337. </td>
  338. </tr>
  339. `
  340. }
  341.  
  342. const seeMore = document.querySelector('.m-playlist-see-more')
  343. if (seeMore) seeMore.innerHTML = '<div class="text">更多内容加载中...</div>'
  344.  
  345. const data = await window.getPlaylistDetail(
  346. '/api/v6/playlist/detail/',
  347. { data: `id=${id}&offset=0&total=true&limit=1000&n=1000` }
  348. )
  349. const { playlist } = data
  350. const content = playlist.tracks
  351. .map((song, index) => render(normalize(song), index, playlist))
  352. .join('')
  353.  
  354. const body = document.querySelector('table tbody')
  355. if (body) body.innerHTML = content
  356. bindEvent()
  357. handleSongChange()
  358.  
  359. if (seeMore) seeMore.parentNode.removeChild(seeMore)
  360. }
  361.  
  362. const handleRoute = () => {
  363. interceptRequest()
  364. const { href, search } = location
  365. if (/\/my\//.test(href)) return
  366.  
  367. const id = new URLSearchParams(search).get('id')
  368. if (/playlist[/?]/.test(href) && id) completePlaylist(id)
  369. }
  370.  
  371. window.addEventListener('songchange', handleSongChange)
  372. window.addEventListener('load', handleRoute, false)
  373. window.addEventListener('hashchange', handleRoute, false)
  374. })()

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址