// ==UserScript==
// @name bilibili favlist hidden video detection
// @name:zh-CN 哔哩哔哩(B站|Bilibili)收藏夹Fix(隐藏视频检测)
// @name:zh-TW 嗶哩嗶哩(B站|Bilibili)收藏夾Fix(隱藏影片檢測)
// @namespace http://tampermonkey.net/
// @version 3
// @description detect videos in favlist that only visiable to upper
// @description:zh-CN 检测收藏夹中被up主设置为仅自己可见的视频
// @description:zh-TW 檢測收藏夾中被上傳者設定為僅自己可見的影片
// @author YTB0710
// @match https://space.bilibili.com/*
// @connect api.bilibili.com
// @grant GM_xmlhttpRequest
// @grant GM_cookie
// @grant GM_openInTab
// @grant GM_setValue
// @grant GM_getValue
// ==/UserScript==
(function () {
'use strict';
const textDictionary = {
'UPDATES': {
'zh-CN': '更新内容: 优化脚本部分逻辑',
'zh-TW': '更新內容: 優化腳本部分邏輯'
},
'ENTER_AV_OR_BV_HERE': {
'zh-CN': '在此输入av号或bv号',
'zh-TW': '在此輸入av號或bv號'
},
'DETECT_HIDDEN_VIDEO_WITH_PROMPT': {
'zh-CN': '检测隐藏视频(先刷新页面)',
'zh-TW': '檢測隱藏影片(先重新載入頁面)'
},
'GET_VIDEO_INFO_WITH_PROMPT': {
'zh-CN': '查询视频信息(输入bv号)',
'zh-TW': '查詢影片資訊(輸入bv號)'
},
'REMOVE_VIDEO_WITH_PROMPT': {
'zh-CN': '取消收藏(输入av号)',
'zh-TW': '取消收藏(輸入av號)'
},
'ADD_VIDEO_WITH_PROMPT': {
'zh-CN': '添加收藏(输入av号)',
'zh-TW': '新增收藏(輸入av號)'
},
'DETECT_HIDDEN_VIDEO': {
'zh-CN': '检测隐藏视频',
'zh-TW': '檢測隱藏影片'
},
'GET_VIDEO_INFO': {
'zh-CN': '查询视频信息',
'zh-TW': '查詢影片資訊'
},
'REMOVE_VIDEO': {
'zh-CN': '取消收藏',
'zh-TW': '取消收藏'
},
'ADD_VIDEO': {
'zh-CN': '添加收藏',
'zh-TW': '新增收藏'
},
'AV': {
'zh-CN': 'av号',
'zh-TW': 'av號'
},
'BV': {
'zh-CN': 'bv号',
'zh-TW': 'bv號'
},
'ENTER_AV': {
'zh-CN': '请输入av号',
'zh-TW': '請輸入av號'
},
'ENTER_BV': {
'zh-CN': '请输入bv号',
'zh-TW': '請輸入bv號'
},
'NO_HIDDEN_VIDEO_ON_THIS_PAGE': {
'zh-CN': '本页没有隐藏的视频',
'zh-TW': '本頁沒有隱藏的影片'
},
'POSITION_ON_THIS_PAGE_WITH_PROMPT': {
'zh-CN': '在本页的位置(从1开始)',
'zh-TW': '在本頁的位置(從1開始)'
},
'POSITION_ON_THIS_PAGE': {
'zh-CN': '在本页的位置',
'zh-TW': '在本頁的位置'
},
'RESPONSE_CONTENT': {
'zh-CN': 'b站接口响应内容',
'zh-TW': 'b站介面回應內容'
},
'REFRESH_PROMPT': {
'zh-CN': '如果出现问题, 请刷新页面后重试',
'zh-TW': '如果出現問題, 請重新載入頁面後再試'
},
'ERROR_COOKIE': {
'zh-CN': '无法读取cookie, 请更新tampermonkey, 前往控制台查看错误信息',
'zh-TW': '無法讀取cookie, 請更新tampermonkey, 前往控制台查看錯誤資訊'
}
};
const preferredLanguage = getPreferredLanguage();
const currentVersion = 3;
const avRegex = /^[1-9]\d*$/;
const bvRegex = /^BV[A-Za-z0-9]{10}$/;
const videosPerPage = 20;
function getPreferredLanguage() {
const languages = navigator.languages || [navigator.language];
for (const lang of languages) {
if (lang === 'zh-CN') {
return 'zh-CN';
}
if (lang === 'zh-TW') {
return 'zh-TW';
}
if (lang === 'zh-HK') {
return 'zh-TW';
}
}
return 'zh-CN';
}
function getText(key) {
return textDictionary[key][preferredLanguage];
}
const checkInterval = setInterval(function () {
const favSidenav = document.querySelector('.fav-sidenav');
if (!favSidenav) {
return;
}
clearInterval(checkInterval);
let version = GM_getValue('version', currentVersion - 1);
let usageCount = GM_getValue('usageCount', 0);
const displayUpdate = version < currentVersion ? true : false;
const displayPrompt = usageCount < 10 ? true : false;
if (displayUpdate) {
GM_setValue('version', version + 0.5);
}
if (displayPrompt) {
GM_setValue('usageCount', usageCount + 1);
}
const watchLaterLink = favSidenav.querySelector('a.watch-later');
watchLaterLink.style.borderBottom = '1px solid #eee';
const controlsContainer = document.createElement('div');
controlsContainer.classList.add('fix-settings-container');
controlsContainer.style.borderTop = '1px solid #e4e9f0';
controlsContainer.style.padding = '2px';
favSidenav.appendChild(controlsContainer);
const inputTextContainer = document.createElement('div');
inputTextContainer.classList.add('fix-inputText-container');
inputTextContainer.style.padding = '2px';
controlsContainer.appendChild(inputTextContainer);
const inputText = document.createElement('input');
inputText.type = 'text';
inputText.classList.add('fix-inputText');
if (displayPrompt) {
inputText.placeholder = getText('ENTER_AV_OR_BV_HERE');
}
inputTextContainer.appendChild(inputText);
const buttonAContainer = document.createElement('div');
buttonAContainer.classList.add('fix-button-container');
buttonAContainer.style.padding = '2px';
controlsContainer.appendChild(buttonAContainer);
const buttonA = document.createElement('button');
buttonA.type = 'button';
if (displayPrompt) {
buttonA.innerText = getText('DETECT_HIDDEN_VIDEO_WITH_PROMPT');
} else {
buttonA.innerText = getText('DETECT_HIDDEN_VIDEO');
}
buttonA.classList.add('fix-action-button');
buttonA.addEventListener('click', function () {
removeInfo();
addInfo(getText('REFRESH_PROMPT'), 10);
const favNum = parseInt(document.querySelector('.fav-item.cur span.num').innerText, 10);
const pager = document.querySelector('.be-pager-next');
let currentPageExpectedNum;
if (pager && !pager.classList.contains('be-pager-disabled')) {
currentPageExpectedNum = videosPerPage;
} else {
currentPageExpectedNum = favNum % videosPerPage;
}
const lis = document.querySelectorAll('ul.fav-video-list li.small-item');
if (lis.length === currentPageExpectedNum) {
addInfo(getText('NO_HIDDEN_VIDEO_ON_THIS_PAGE'), 12);
return;
}
const fid = document.querySelector('.fav-item.cur').getAttribute('fid');
GM_xmlhttpRequest({
method: 'GET',
url: `https://api.bilibili.com/x/v3/fav/resource/ids?media_id=${fid}`,
responseType: 'json',
onload: function (response) {
const datas = response.response.data;
let currentPage = 1;
if (favNum > 20) {
currentPage = parseInt(document.querySelector('.be-pager-item-active').innerText, 10);
}
const startIndex = (currentPage - 1) * videosPerPage;
const currentPageExpectedDatas = datas.slice(startIndex, startIndex + videosPerPage);
const currentPageActualBVs = Array.from(lis).map(li => li.getAttribute('data-aid'));
const hiddenDatas = currentPageExpectedDatas.filter(data => !currentPageActualBVs.includes(data.bvid));
hiddenDatas.forEach(hiddenData => {
if (displayPrompt) {
addInfo(`${getText('POSITION_ON_THIS_PAGE_WITH_PROMPT')}: ${currentPageExpectedDatas.findIndex(data => data.bvid === hiddenData.bvid) + 1}`, 12);
} else {
addInfo(`${getText('POSITION_ON_THIS_PAGE')}: ${currentPageExpectedDatas.findIndex(data => data.bvid === hiddenData.bvid) + 1}`, 12);
}
addInfo(`${getText('AV')}: ${hiddenData.id}`, 12);
addInfo(`${getText('BV')}: ${hiddenData.bvid}`, 12);
});
}
});
});
buttonAContainer.appendChild(buttonA);
const buttonBContainer = document.createElement('div');
buttonBContainer.classList.add('fix-button-container');
buttonBContainer.style.padding = '2px';
controlsContainer.appendChild(buttonBContainer);
const buttonB = document.createElement('button');
buttonB.type = 'button';
if (displayPrompt) {
buttonB.innerText = getText('GET_VIDEO_INFO_WITH_PROMPT');
} else {
buttonB.innerText = getText('GET_VIDEO_INFO');
}
buttonB.classList.add('fix-action-button');
buttonB.addEventListener('click', function () {
removeInfo();
const bv = document.querySelector('div.fix-inputText-container input').value;
if (!bvRegex.test(bv)) {
addInfo(getText('ENTER_BV'), 12);
return;
}
GM_openInTab(`https://www.biliplus.com/video/${bv}`, { active: true, insert: false, setParent: true });
GM_openInTab(`https://xbeibeix.com/video/${bv}`, { insert: false, setParent: true });
GM_openInTab(`https://www.jijidown.com/video/${bv}`, { insert: false, setParent: true });
});
buttonBContainer.appendChild(buttonB);
const buttonCContainer = document.createElement('div');
buttonCContainer.classList.add('fix-button-container');
buttonCContainer.style.padding = '2px';
controlsContainer.appendChild(buttonCContainer);
const buttonC = document.createElement('button');
buttonC.type = 'button';
if (displayPrompt) {
buttonC.innerText = getText('REMOVE_VIDEO_WITH_PROMPT');
} else {
buttonC.innerText = getText('REMOVE_VIDEO');
}
buttonC.classList.add('fix-action-button');
buttonC.addEventListener('click', function () {
removeInfo();
GM_cookie.list({ name: 'bili_jct' }, function (cookies, error) {
if (!error) {
const av = document.querySelector('div.fix-inputText-container input').value;
if (!avRegex.test(av)) {
addInfo(getText('ENTER_AV'), 12);
return;
}
const id = document.querySelector('.fav-item.cur').getAttribute('fid');
const csrf = cookies[0].value;
const data = `resources=${av}%3A2&media_id=${id}&platform=web&csrf=${csrf}`;
GM_xmlhttpRequest({
method: 'POST',
url: 'https://api.bilibili.com/x/v3/fav/resource/batch-del',
data: data,
headers: {
'Content-Length': `${data.length}`,
'Content-Type': 'application/x-www-form-urlencoded'
},
onload: function (response) {
const json = response.response;
addInfo(`${getText('RESPONSE_CONTENT')}:`, 12);
addInfo(json, 10);
}
});
} else {
console.error(error);
addInfo(getText('ERROR_COOKIE'), 12);
}
});
});
buttonCContainer.appendChild(buttonC);
const buttonDContainer = document.createElement('div');
buttonDContainer.classList.add('fix-button-container');
buttonDContainer.style.padding = '2px';
controlsContainer.appendChild(buttonDContainer);
const buttonD = document.createElement('button');
buttonD.type = 'button';
if (displayPrompt) {
buttonD.innerText = getText('ADD_VIDEO_WITH_PROMPT');
} else {
buttonD.innerText = getText('ADD_VIDEO');
}
buttonD.classList.add('fix-action-button');
buttonD.addEventListener('click', function () {
removeInfo();
GM_cookie.list({ name: 'bili_jct' }, function (cookies, error) {
if (!error) {
const av = document.querySelector('div.fix-inputText-container input').value;
if (!avRegex.test(av)) {
addInfo(getText('ENTER_AV'), 12);
return;
}
const id = document.querySelector('.fav-item.cur').getAttribute('fid');
const csrf = cookies[0].value;
const data = `rid=${av}&type=2&add_media_ids=${id}&csrf=${csrf}`;
GM_xmlhttpRequest({
method: 'POST',
url: 'https://api.bilibili.com/x/v3/fav/resource/deal',
data: data,
headers: {
'Content-Length': `${data.length}`,
'Content-Type': 'application/x-www-form-urlencoded'
},
onload: function (response) {
const json = response.response;
addInfo(`${getText('RESPONSE_CONTENT')}:`, 12);
addInfo(json, 10);
}
});
} else {
console.error(error);
addInfo(getText('ERROR_COOKIE'), 12);
}
});
});
buttonDContainer.appendChild(buttonD);
const textContainer = document.createElement('div');
textContainer.classList.add('fix-text-container');
textContainer.style.padding = '2px';
controlsContainer.appendChild(textContainer);
if (displayUpdate) {
addInfo(getText('UPDATES', 12));
}
function addInfo(info, px) {
const p = document.createElement('p');
p.innerText = info;
p.style.fontSize = `${px}px`;
textContainer.appendChild(p);
p.scrollIntoView({ behavior: 'instant', block: 'nearest' });
}
function removeInfo() {
while (textContainer.firstChild) {
textContainer.removeChild(textContainer.firstChild);
}
}
}, 300);
})();