// ==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");