您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
(感谢闲鱼买家ZEP的有偿定制) 用纯文字列表的方式展示B站收藏夹结果,方便按各列排序。
// ==UserScript== // @name BiliBili-TextFavList // @name:zh-CN BiliBili-文字收藏夹列表 // @namespace https://github.com/Mehver // @version 1.0 // @description (Thanks to ZEP's paid customization) Display Bilibili favourite list results in a text list, which is convenient for sorting by each column. // @description:zh-CN (感谢闲鱼买家ZEP的有偿定制) 用纯文字列表的方式展示B站收藏夹结果,方便按各列排序。 // @sponsor ZEP // @author https://github.com/Mehver // @icon  // @match http*://space.bilibili.com/*/favlist* // @exclude http*://space.bilibili.com/*/favlist*ctype* // @license MPL-2.0 // @license^ Mozilla Public License 2.0 // @charset UTF-8 // @homepageURL https://github.com/SynRGB/BiliBili-TextFavList // @contributionURL https://github.com/SynRGB/BiliBili-TextFavList // @copyright Copyright © 2022-PRESENT, Mehver (https://github.com/Mehver) // @grant GM_addStyle // @grant GM_getResourceText // @grant GM_getValue // @grant GM_setValue // @grant GM_registerMenuCommand // @resource DataTablesCSS https://cdn.datatables.net/1.11.3/css/jquery.dataTables.min.css // ==/UserScript== let table_font_size = GM_getValue('table_font_size', 16); let network_delay = GM_getValue('network_delay', 400); GM_registerMenuCommand('设置表格字体大小', async () => { let newFontSize = prompt('请输入新的字体大小(单位px):', table_font_size); if (newFontSize) { table_font_size = newFontSize; await GM_setValue('table_font_size', table_font_size); alert('字体大小已更新!请刷新页面以查看更改。'); } }); GM_registerMenuCommand('设置脚本加载时延', async () => { let newNetworkDelay = prompt('脚本依赖外部组件,对于不同的网络环境和电脑性能,\n组件加载速度也不同。如果出现显示错误、白屏等BUG,\n可以适度调高这个数字确保外部引入的组件完成加载。\n默认400(单位ms):', network_delay); if (newNetworkDelay) { network_delay = newNetworkDelay; await GM_setValue('network_delay', network_delay); alert('脚本加载时延已更新!请刷新页面以查看更改。'); } }); ////////////////////////////////////// //////////// DataTables ////////////// let cssTxt = GM_getResourceText("DataTablesCSS"); GM_addStyle(cssTxt); let head = document.head || document.getElementsByTagName('head')[0]; let link = document.createElement('link'); link.type = 'text/css'; link.rel = 'stylesheet'; link.href = 'https://cdn.datatables.net/1.11.3/css/jquery.dataTables.min.css'; head.appendChild(link); (function() { let jQueryScript = document.createElement("script"); jQueryScript.src = "https://code.jquery.com/jquery-3.6.0.min.js"; jQueryScript.onload = () => { let dtScript = document.createElement("script"); dtScript.src = "https://cdn.datatables.net/1.11.3/js/jquery.dataTables.min.js"; // 加载完成后首次运行 dtScript.onload = () => { main(); }; document.body.appendChild(dtScript); }; document.body.appendChild(jQueryScript); })(); //////////// DataTables ////////////// ////////////////////////////////////// ////////////////////////////////////// /////////////// 触发器 //////////////// // 延时避免在 dtScript 和 jQueryScript 加载完成前就运行 setTimeout(function() { setTimeout(function() { main(); }, network_delay / 2); let observerElement = new window.MutationObserver(function() { setTimeout(function() { main(); }, network_delay / 3); }); // 收藏夹切换触发 let targetForObserver1 = document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.favList-info'); if (targetForObserver1) { observerElement.observe(targetForObserver1, { childList: true, subtree: true, characterData: true }); } // 手动选择分区或收藏时间 let targetElement2 = document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-header.fav-header-info > div > div'); if (targetElement2) { targetElement2.addEventListener('click', function() { setTimeout(function() { main(); }, network_delay / 2); }); } // 翻页触发 let targetElement3 = document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-content.section > ul.be-pager'); if (targetElement3) { targetElement3.addEventListener('click', function() { setTimeout(function() { main(); }, network_delay / 2); }); } // 搜索触发 let targetElement4 = document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-header.fav-header-info > div > div > div.filter-item.search'); if (targetElement4) { targetElement4.addEventListener('click', function() { setTimeout(function() { main(); }, network_delay * 2); }); } // 订阅等其他类型的收藏夹,点击后应避免脚本生效 let excludeElement1 = document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-sidenav > div:nth-child(2)'); if (excludeElement1) { excludeElement1.addEventListener('click', function() { setTimeout(function() { revert_main(); }, network_delay / 2); }); } let excludeElement2 = document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-sidenav > div:nth-child(3)'); if (excludeElement2) { excludeElement2.addEventListener('click', function() { setTimeout(function() { revert_main(); }, network_delay / 2); }); } }, network_delay * 2); /////////////// 触发器 //////////////// ////////////////////////////////////// ///////////////////////////////////// /////////////// main //////////////// function revert_main() { try { // 隐藏列表 try { document.querySelector('#biliResultsTable').style.display = 'none'; } catch (e) {} // 显示收藏夹 try { document.querySelector("#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-content.section > ul.fav-video-list.clearfix.content").style.display = 'block'; } catch (e) {} } catch (e) { console.log(e); } } function main() { try { // 隐藏”批量操作“按钮 try { document.querySelector('#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-header.fav-header-info > div > div > div.filter-item.do-batch').style.display = 'none'; } catch (e) {} // 隐藏收藏夹 try { document.querySelector("#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-content.section > ul.fav-video-list.clearfix.content").style.display = 'none'; } catch (e) {} // Create table with thead for DataTables let table = document.createElement('table'); table.id = "biliResultsTable"; let thead = document.createElement('thead'); let tbody = document.createElement('tbody'); let header = ["收藏于", "时长", "标题"]; let trHead = document.createElement('tr'); header.forEach(text => { let th = document.createElement('th'); th.textContent = text; trHead.appendChild(th); }); thead.appendChild(trHead); setTimeout(function() { let videoCards = document.querySelectorAll('li'); videoCards.forEach(videoCard => { let title; let length; let pubdate; try { title = videoCard.querySelector('a.title')?.textContent.trim(); length = videoCard.querySelector('span.length')?.textContent.trim(); pubdate = videoCard.querySelector('div.meta.pubdate')?.textContent.trim().replace('收藏于: ', ''); // let link_video = videoCard.querySelectorAll('a')[0].getAttribute('href'); } catch (e) {} let tr = document.createElement('tr'); // 确保没有为空的数据 if ( (title !== undefined) && (length !== undefined) && (pubdate !== undefined) // (link_video !== undefined) ) { [pubdate, length, title].forEach(text => { let td = document.createElement('td'); td.textContent = text; tr.appendChild(td); }); // let tdTitle = tr.querySelector('td:nth-child(3)'); // tdTitle.innerHTML = `<a href="${link_video}" target="_blank">${title}</a>`; // b230815.02 时长加粗 tr.querySelector('td:nth-child(2)').style.fontWeight = 'bold'; // b230815.02 标题用 `#00AEEC` 颜色 tr.querySelector('td:nth-child(3)').style.color = '#00AEEC'; tbody.appendChild(tr); } }); table.appendChild(thead); table.appendChild(tbody); // 回调获取异步数据,适用于异步加载 if (tbody.childElementCount === 0) { main(); return; } let tables = document.querySelectorAll('#biliResultsTable'); if (tables.length === 0) { let targetDiv_bott = document.querySelector("#page-fav > div.col-full.clearfix.master > div.fav-main.section > div.fav-content.section > ul.be-pager"); targetDiv_bott.parentNode.insertBefore(table, targetDiv_bott); } else { tables[0].parentNode.replaceChild(table, tables[0]); } // DataTables 的自定义排序算法 $.fn.dataTable.ext.type.order['length-sort-pre'] = function (d) {return convertLength(d);}; $.fn.dataTable.ext.type.order['pubdate-sort-pre'] = function (d) {return convertPubDate(d);}; // Initialize DataTables $(table).DataTable({ "paging": false, "searching": false, "info": false, "columnDefs": [ { "type": "length-sort", "targets": 1 }, { "type": "pubdate-sort", "targets": 0 } ] }); // b230815.02 去掉底边横线 GM_addStyle("table.dataTable.no-footer { border-bottom: 0px none !important; }"); // b230815.02 去掉表头横线 (因CSS复杂,所以创建白色色块覆盖) GM_addStyle(".dataTable thead th { border-bottom: 0px none !important; }"); // b230815.02 调大字号 GM_addStyle(`.dataTable { font-size: ${table_font_size}px !important; }`); // 左侧留30px空白 GM_addStyle(`.dataTable { margin-left: 30px !important; }`); }, 100); } catch (e) { console.log(e); } } /////////////// main //////////////// ///////////////////////////////////// /////////////////////////////////// /////// DataTable 的排序算法 //////// function convertLength(duration) { let parts = duration.split(':').map(part => parseInt(part, 10)); if (parts.length === 3) { return parts[0] * 3600 + parts[1] * 60 + parts[2]; } else if (parts.length === 2) { return parts[0] * 60 + parts[1]; } else { return NaN; } } function convertPubDate(date) { const now = new Date(); if (date.includes('刚刚')) { return now.getTime(); } if (date.includes('小时前')) { const hoursAgo = parseFloat(date.replace('小时前', '')); return now - hoursAgo * 3600 * 1000; // Convert hours to milliseconds } if (date === "昨天") { return now - 24 * 3600 * 1000; // 24 hours in milliseconds } if (date.includes('-')) { const parts = date.split('-').map(part => { return part.padStart(2, '0'); }); // If only month and day are given, use the current year. if (parts.length === 2) { parts.unshift(now.getFullYear().toString()); } // Create a new Date object and return its time value in milliseconds return new Date(parts.join('-')).getTime(); } } /////// DataTable 的排序算法 //////// /////////////////////////////////// console.log("JS script BiliBili-TextFavList (BiliBili-文字收藏夹列表) loaded. See more details at https://github.com/SynRGB/BiliBili-TextFavList");