Greasy Fork is available in English.
Download pictures from bilibili timeline and 720P videos.
// ==UserScript== // @name Bilibili Download Pictures // @name:zh-CN 下载Bilibili动态页面图片 // @version 0.9.9 // @description Download pictures from bilibili timeline and 720P videos. // @description:zh-CN 下载“Bilibili动态”时间线页面的图片,也可下载视频(720P单文件) // @author OWENDSWANG // @icon https://avatars.githubusercontent.com/u/9076865?s=40&v=4 // @license MIT // @homepage https://greasyfork.org/scripts/421885 // @supportURL https://github.com/owendswang/Download-Pictures-from-Bilibili-Timeline/issues // @match https://t.bilibili.com/* // @match https://space.bilibili.com/*/dynamic* // @match https://www.bilibili.com/opus/* // @match https://www.bilibili.com/video/* // @match https://www.bilibili.com/v/topic/detail/?* // @connect bilibili.com // @connect bilivideo.com // @connect bilivideo.cn // @connect hdslb.com // @connect biliimg.com // @grant GM_download // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @grant GM_cookie // @grant GM_registerMenuCommand // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js // @namespace https://greasyfork.org/users/738244 // ==/UserScript== (function() { 'use strict'; // Your code here... const settingVersion = 1; const downloadIcon = 'url(\'\')'; let notLoaded = true; let cardsTotal = 0; let skeletonsTotal = 0; let downloadQueueCard = document.createElement('div'); downloadQueueCard.style.position = 'fixed'; downloadQueueCard.style.bottom = '0.5rem'; downloadQueueCard.style.left = '0.5rem'; downloadQueueCard.style.maxHeight = '50vh'; downloadQueueCard.style.overflowY = 'auto'; downloadQueueCard.style.overflowX = 'hidden'; downloadQueueCard.style.zIndex = '10'; let downloadQueueTitle = document.createElement('div'); downloadQueueTitle.textContent = '下载队列'; downloadQueueTitle.style.fontSize = '0.8rem'; downloadQueueTitle.style.color = 'gray'; downloadQueueTitle.style.display = 'none'; downloadQueueCard.appendChild(downloadQueueTitle); document.body.appendChild(downloadQueueCard); let progressBar = document.createElement('div'); progressBar.style.height = '1.4rem'; progressBar.style.width = '17rem'; // progressBar.style.background = 'linear-gradient(to right, red 100%, transparent 100%)'; progressBar.style.borderStyle = 'solid'; progressBar.style.borderWidth = '0.1rem'; progressBar.style.borderColor = 'grey'; progressBar.style.borderRadius = '0.5rem'; progressBar.style.boxSizing = 'content-box'; progressBar.style.marginTop = '0.5rem'; progressBar.style.marginRight = '1rem'; progressBar.style.position = 'relative'; let progressText = document.createElement('div'); // progressText.textContent = 'test.test'; progressText.style.mixBlendMode = 'screen'; progressText.style.width = '100%'; progressText.style.textAlign = 'center'; progressText.style.color = 'orange'; progressText.style.fontSize = '0.7rem'; progressText.style.lineHeight = '1.4rem'; progressText.style.overflow = 'hidden'; progressBar.appendChild(progressText); let progressCloseBtn = document.createElement('button'); progressCloseBtn.style.border = 'unset'; progressCloseBtn.style.background = 'unset'; progressCloseBtn.style.color = 'orange'; progressCloseBtn.style.position = 'absolute'; progressCloseBtn.style.right = '0.3rem'; progressCloseBtn.style.top = '0.2rem'; progressCloseBtn.style.fontSize = '1rem'; progressCloseBtn.style.lineHeight = '1rem'; progressCloseBtn.style.cursor = 'pointer'; progressCloseBtn.textContent = '×'; progressCloseBtn.title = '取消'; progressCloseBtn.onmouseover = function(e){ this.style.color = 'red'; } progressCloseBtn.onmouseout = function(e){ this.style.color = 'orange'; } progressBar.appendChild(progressCloseBtn); // downloadQueueCard.appendChild(progressBar); function oXMLHttpRequest(url, type) { return new Promise(function(resolve, reject) { let oReq = new XMLHttpRequest(); oReq.open("GET", url); oReq.withCredentials = true; oReq.responseType = type; oReq.onload = (e) => { // console.log(e); // console.log(oReq.response); resolve(oReq.response); }; oReq.onerror = (e) => { console.log(e); alert('请求失败!'); resolve(null); }; oReq.onabort = (e) => { console.log(e); alert('请求被中断!'); resolve(null); }; oReq.ontimeout = (e) => { console.log(e); alert('请求超时!'); resolve(null); }; oReq.send(null); }); } /*function saveAs(blob, name) { const link = document.createElement("a"); link.style.display = "none"; link.href = URL.createObjectURL(blob); link.download = name; link.target = '_blank'; document.body.appendChild(link); // console.log(link); link.click(); const timeout = setTimeout(() => { URL.revokeObjectURL(link.href); link.parentNode.removeChild(link); }, 1000); }*/ function downloadError(e, url, name, progress) { // console.log(e, url); /*GM_notification({ title: 'Download error', text: 'Error: ' + e.error + '\nUrl: ' + url, silent: true, timeout: 3, });*/ progress.style.background = 'red'; progress.firstChild.textContent = (name.length > 10 ? (name.substring(0,10) + '...') : name) + ' [' + (e.error || 'Unknown') + ']'; progress.firstChild.style.color = 'yellow'; progress.firstChild.style.mixBlendMode = 'unset'; let progressRetryBtn = document.createElement('button'); progressRetryBtn.style.border = 'unset'; progressRetryBtn.style.background = 'unset'; progressRetryBtn.style.color = 'yellow'; progressRetryBtn.style.position = 'absolute'; progressRetryBtn.style.right = '1.2rem'; progressRetryBtn.style.top = '0.05rem'; progressRetryBtn.style.fontSize = '1rem'; progressRetryBtn.style.lineHeight = '1rem'; progressRetryBtn.style.cursor = 'pointer'; progressRetryBtn.style.letterSpacing = '-0.2rem'; progressRetryBtn.textContent = '⤤⤦'; progressRetryBtn.title = '重试'; progressRetryBtn.onmouseover = function(e){ this.style.color = 'white'; } progressRetryBtn.onmouseout = function(e){ this.style.color = 'yellow'; } progressRetryBtn.onclick = function(e) { this.parentNode.remove(); downloadWrapper(url, name); } progress.insertBefore(progressRetryBtn, progress.lastChild); progress.lastChild.title = '关闭'; progress.lastChild.style.color = 'yellow'; progress.lastChild.onmouseover = function(e){ this.style.color = 'white'; }; progress.lastChild.onmouseout = function(e){ this.style.color = 'yellow'; }; progress.lastChild.onclick = function(e) { this.parentNode.remove(); if(progress.parent.childElementCount == 1) progress.parent.firstChild.style.display = 'none'; }; // setTimeout(() => { progress.remove(); if(downloadQueueCard.childElementCount == 1) downloadQueueTitle.style.display = 'none'; }, 1000); } function downloadWrapper(url, name) { // console.log('downloadWrapper: ', url, name); downloadQueueTitle.style.display = 'block'; let progress = downloadQueueCard.appendChild(progressBar.cloneNode(true)); progress.firstChild.textContent = (name.length > 10 ? (name.substring(0,10) + '...') : name) + ' [0%]'; return new Promise(function(resolve, reject) { const download = GM_xmlhttpRequest({ method: 'GET', url, responseType: 'blob', headers: { Referer: location.protocol + '//' + location.hostname, Origin: location.protocol + '//' + location.hostname, }, onprogress: (e) => { // e = { int done, finalUrl, bool lengthComputable, int loaded, int position, int readyState, response, str responseHeaders, responseText, responseXML, int status, statusText, int total, int totalSize } const percent = e.done / e.total * 100; progress.style.background = 'linear-gradient(to right, green ' + percent + '%, transparent ' + percent + '%)'; progress.firstChild.textContent = (name.length > 10 ? (name.substring(0,10) + '...') : name) + ' [' + percent.toFixed(0) + '%]'; }, onload: function({ response }) { // console.log(response); const timeout = setTimeout(() => { progress.remove(); if(downloadQueueCard.childElementCount == 1) downloadQueueTitle.style.display = 'none'; }, 1000); progress.lastChild.onclick = function(e) { clearTimeout(timeout); this.parentNode.remove(); if(downloadQueueCard.childElementCount == 1) downloadQueueTitle.style.display = 'none'; }; saveAs(response, name); resolve(null); }, onabort: function(e) { console.log(e); resolve(null); }, onerror: function(e) { downloadError(e, url, name, progress); console.log(e); resolve(null); }, ontimeout: function(e) { downloadError(e, url, name, progress); console.log(e); resolve(null); }, }); progress.lastChild.onclick = function(e) { download.abort(); this.parentNode.remove(); if(downloadQueueCard.childElementCount == 1) downloadQueueTitle.style.display = 'none'; }; }); } /*function getCookie(name) { return new Promise(function(resolve, reject) { GM_cookie.list({ name: name }, function(cookies, error) { if (!error) { // console.log(cookies); resolve(cookies[0].value); } else { console.error(error); } }); }); } function getAllCookies() { return new Promise(function(resolve, reject) { GM_cookie.list({}, function(cookies, error) { if (!error) { // console.log(cookies); const cookiesStr = cookies.map((ele) => { return ele.name + '=' + ele.value }).join('; '); // console.log(cookiesStr); resolve(cookiesStr); } else { console.error(error); resolve(null); } }); }); } function download2Blob(url) { // console.log('download2Blob: ', url); return new Promise(function(resolve, reject) { GM_xmlhttpRequest({ method: 'GET', url, responseType: 'blob', headers: { Referer: location.protocol + '//' + location.hostname, Origin: location.protocol + '//' + location.hostname, }, onload: function({ response }) { // console.log(response); // saveAs(response, name); resolve(response); }, onabort: function(e) { console.log(e); alert('请求被中断!'); resolve(null); }, onerror: function(e) { console.log(e); alert('请求失败!'); resolve(null); }, ontimeout: function(e) { console.log(e); alert('下载超时!'); resolve(null); }, }); }); }*/ function getPicName(nameSetting, originalName, index, data) { const card = JSON.parse(data.card.card); let setName = nameSetting; setName = setName.replace('{original}', originalName.split('.')[0]); setName = setName.replace('{ext}', originalName.split('.')[1]); const userName = card.user?.name || data.card.desc.user_profile.info.uname; const userId = card.user?.uid || data.card.desc.user_profile.info.uid; const dynamicId = data.card.desc.dynamic_id; const content = card.item?.description || card.title; setName = setName.replace('{username}', userName); setName = setName.replace('{userid}', userId); setName = setName.replace('{dynamicid}', dynamicId); setName = setName.replace('{index}', index); setName = setName.replace('{content}', content.substring(0, 25)); let YYYY, MM, DD, HH, mm, ss; const postAt = new Date((card.item?.upload_time || data.card.desc.timestamp) * 1000); YYYY = postAt.getFullYear().toString(); MM = (postAt.getMonth() + 1).toString().padStart(2, '0'); DD = postAt.getDate().toString().padStart(2, '0'); HH = postAt.getHours().toString().padStart(2, '0'); mm = postAt.getMinutes().toString().padStart(2, '0'); ss = postAt.getSeconds().toString().padStart(2, '0'); setName = setName.replace('{YYYY}', YYYY); setName = setName.replace('{MM}', MM); setName = setName.replace('{DD}', DD); setName = setName.replace('{HH}', HH); setName = setName.replace('{mm}', mm); setName = setName.replace('{ss}', ss); /*if (retweetPostId && GM_getValue('retweetMode', false)) { setName = setName.replace('{re.mblogid}', retweetPostId); setName = setName.replace('{re.username}', retweetUserName); setName = setName.replace('{re.userid}', retweetUserId); setName = setName.replace('{re.uid}', retweetPostUid); setName = setName.replace('{re.content}', retweetContent.substring(0, 25)); let reYYYY, reMM, reDD, reHH, remm, ress; const retweetPostAt = new Date(retweetPostTime); if (retweetPostTime) { reYYYY = retweetPostAt.getFullYear().toString(); reMM = (retweetPostAt.getMonth() + 1).toString().padStart(2, '0'); reDD = retweetPostAt.getDate().toString().padStart(2, '0'); reHH = retweetPostAt.getHours().toString().padStart(2, '0'); remm = retweetPostAt.getMinutes().toString().padStart(2, '0'); ress = retweetPostAt.getSeconds().toString().padStart(2, '0'); } setName = setName.replace('{re.YYYY}', reYYYY); setName = setName.replace('{re.MM}', reMM); setName = setName.replace('{re.DD}', reDD); setName = setName.replace('{re.HH}', reHH); setName = setName.replace('{re.mm}', remm); setName = setName.replace('{re.ss}', ress); }*/ return setName.replace(/[<|>|*|"|\/|\\|\||:|?|\n]/g, '_'); } function getVidName(nameSetting, originalName, data) { let setName = nameSetting; setName = setName.replace('{original}', originalName.split('.')[0]); setName = setName.replace('{ext}', originalName.split('.')[1]); const bvid = data.bvid; const aid = data.aid; const cid = data.cid; const title = data.title; const content = data.desc; const userName = data.owner.name; const userId = data.owner.mid; setName = setName.replace('{bvid}', bvid); setName = setName.replace('{aid}', aid); setName = setName.replace('{cid}', cid); setName = setName.replace('{title}', title); setName = setName.replace('{content}', content.substring(0, 25)); setName = setName.replace('{username}', userName); setName = setName.replace('{userid}', userId); let YYYY, MM, DD, HH, mm, ss; const postAt = new Date(data.ctime * 1000); YYYY = postAt.getFullYear().toString(); MM = (postAt.getMonth() + 1).toString().padStart(2, '0'); DD = postAt.getDate().toString().padStart(2, '0'); HH = postAt.getHours().toString().padStart(2, '0'); mm = postAt.getMinutes().toString().padStart(2, '0'); ss = postAt.getSeconds().toString().padStart(2, '0'); setName = setName.replace('{YYYY}', YYYY); setName = setName.replace('{MM}', MM); setName = setName.replace('{DD}', DD); setName = setName.replace('{HH}', HH); setName = setName.replace('{mm}', mm); setName = setName.replace('{ss}', ss); return setName.replace(/[<|>|*|"|\/|\\|\||:|?|\n]/g, '_'); } function handleImageDynamic(data) { const card = JSON.parse(data.card.card); const pictures = card.item.pictures; // console.log(pictures); for (const [ index, picture ] of pictures.entries()) { // console.log(picture); const pictureUrl = picture.img_src; const originalName = pictureUrl.split('/')[pictureUrl.split('/').length - 1]; const pictureName = getPicName(GM_getValue('dlPicName', '{original}.{ext}'), originalName, index + 1, data); /*GM_download({ url: pictureUrl, name: pictureName, onerror: function(e) { console.log(e); alert('下载失败!'); }, ontimeout: function(e) { console.log(e); alert('下载超时!'); }, });*/ downloadWrapper(pictureUrl, pictureName); } } async function handleArticleDynamic(data) { const card = JSON.parse(data.card.card); const pictures = card.image_urls; // console.log(pictures); for (const [ index, picture ] of pictures.entries()) { // console.log(picture); const pictureUrl = picture; const originalName = pictureUrl.split('/')[pictureUrl.split('/').length - 1]; const pictureName = getPicName(GM_getValue('dlPicName', '{original}.{ext}'), originalName, index + 1, data); /*GM_download({ url: pictureUrl, name: pictureName, onerror: function(e) { console.log(e); alert('下载失败!'); }, ontimeout: function(e) { console.log(e); alert('下载超时!'); }, });*/ downloadWrapper(pictureUrl, pictureName); } } function getVideoInfo(bvid) { // console.log('getVideoInfo'); return oXMLHttpRequest('https://api.bilibili.com/x/web-interface/view?bvid=' + bvid, 'json'); } function getVideoDetail(aid, cid/*, cookies*/) { /*return new Promise(function(resolve, reject) { // console.log(aid, cid, cookies); GM_xmlhttpRequest({ method: 'GET', // 1080p dash --> https://api.bilibili.com/x/player/playurl?avid=1551880723&cid=1473551215&qn=80&fnval=4048 // 720p mp4 --> https://api.bilibili.com/x/player/playurl?avid=1551880723&cid=1473551215&qn=64 url: 'https://api.bilibili.com/x/player/wbi/playurl?avid=' + aid.toString() + '&cid=' + cid.toString() + '&fnval=64', responseType: 'json', anonymous: true, cookie: cookies, onload: function({ response }) { // console.log(response); resolve(response); }, onabort: function(e) { resolve(null); }, onerror: function(e) { resolve(null); }, ontimeout: function(e) { resolve(null); }, }); });*/ return oXMLHttpRequest('https://api.bilibili.com/x/player/wbi/playurl?avid=' + aid.toString() + '&cid=' + cid.toString() + '&fnval=64', 'json'); } async function downloadVideo(data) { const vidRes = await getVideoDetail(data.aid, data.cid/*, cookies*/); // console.log(vidRes); const vidUrl = vidRes.data.durl[0].url; // console.log(vidUrl); const originalName = vidUrl.split('?')[0].split('/')[vidUrl.split('?')[0].split('/').length - 1]; const vidName = getVidName(GM_getValue('dlVidName', '{original}.{ext}'), originalName, data); // console.log(vidName); /*GM_download({ url: vidUrl, name: vidName, headers: { Referer: 'https://www.bilibili.com', Origin: 'https://www.bilibili.com', }, onerror: function(e) { console.log(e); }, ontimeout: function(e) { console.log(e); }, }); const blob = await download2Blob(vidUrl); const url = URL.createObjectURL(blob); // console.log(url); GM_download({ url: url, name: vidName, onerror: function(e) { console.log(e); alert('下载失败!'); }, ontimeout: function(e) { console.log(e); alert('下载超时!'); }, }); saveAs(blob, vidName);*/ downloadWrapper(vidUrl, vidName); } async function handleVideoDownload(bvid) { const vidInfoRes = await getVideoInfo(bvid); // console.log(vidInfoRes); await downloadVideo(vidInfoRes.data); } async function handleVideoDynamic(data) { // console.log('handleVideoDynamic'); // console.log(data); // const card = JSON.parse(data.card.card); // const aid = card.aid; // const cid = card.cid; // console.log(aid, cid); // const cookies = await getAllCookies(); // const cookies = 'SESSDATA=' + await getCookie('SESSDATA'); // console.log(cookies); // await downloadVideo(aid, cid); const bvid = data.card.desc.bvid; await handleVideoDownload(bvid); } function getDynamicDetail(dynId) { /*return new Promise(function(resolve, reject) { GM_xmlhttpRequest({ method: 'GET', url: 'https://api.vc.bilibili.com/dynamic_svr/v1/dynamic_svr/get_dynamic_detail?dynamic_id=' + dynId, responseType: 'json', onload: function({ response }) { // console.log(response); resolve(response); }, onabort: function(e) { resolve(null); }, onerror: function(e) { resolve(null); }, ontimeout: function(e) { resolve(null); }, }); });*/ return oXMLHttpRequest('https://api.vc.bilibili.com/dynamic_svr/v1/dynamic_svr/get_dynamic_detail?dynamic_id=' + dynId, 'json'); // return oXMLHttpRequest('https://api.bilibili.com/x/polymer/web-dynamic/v1/detail?id=' + dynId, 'json'); } async function handleDynamicDownload(dynId) { // console.log('handleDynamicDownload: ' + dynId); const dynRes = await getDynamicDetail(dynId); // console.log(dynRes.data); if (dynRes.data.card.card) { const card = JSON.parse(dynRes.data.card.card); switch(dynRes.data.card.desc.type) { case 1: // 转发 break; case 2: // 图片 // console.log('picture'); handleImageDynamic(dynRes.data); break; case 4: // 文字 break; case 8: // 视频 // console.log('video'); handleVideoDynamic(dynRes.data); break; case 64: // 专栏 handleArticleDynamic(dynRes.data); break; case 256: // 音频 break; default: break; } } else { console.log('no content found!', dynRes); alert('无法下载!'); } } function addOpusDownloadButton(card) { if(card.getElementsByClassName('download-button').length == 0) { // console.log(card); const buttonBar = card.getElementsByClassName('bili-tabs__nav__items')[0]; let downloadButton = document.createElement('div'); downloadButton.textContent = '下载'; downloadButton.classList.add('bili-tabs__nav__item'); downloadButton.addEventListener('click', function(event) { const dynId = window.location.pathname.split('/')[window.location.pathname.split('/').length - 1]; // console.log(dynId); handleDynamicDownload(dynId); /*const content = document.body.querySelector('div.opus-module-content'); const list = content.querySelectorAll('div.bili-album__preview__picture__img'); // console.log(list); for (const item of list) { let imgUrl = item.style.backgroundImage.split(/"|@/)[1] || item.querySelector('img').src.split('@')[0]; if (imgUrl.startsWith('//')) { imgUrl = 'https:' + imgUrl; } const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1]; // console.log(imgUrl); // console.log(imgName); GM_download(imgUrl, imgName); } const topAlbum = document.body.querySelector('div.opus-module-top__album'); if (topAlbum) { const topAlbumIndicatorList = topAlbum.querySelectorAll('div.horizontal-scroll-album__indicator > div > img'); const topAlbumList = topAlbum.querySelectorAll('div.horizontal-scroll-album__pic__img > img'); let topList = topAlbumList; if (topAlbumIndicatorList.length > 0) topList = topAlbumIndicatorList; for (const item of topList) { let imgUrl = item.src.split(/@/)[0]; if (imgUrl.startsWith('//')) { imgUrl = 'https:' + imgUrl; } const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1]; // console.log(imgUrl); // console.log(imgName); GM_download(imgUrl, imgName); } }*/ }); buttonBar.appendChild(downloadButton); } } function addDownloadButton(card) { // console.log('addDownloadButton'); if(card.getElementsByClassName('download-button').length == 0) { if(card.getElementsByClassName('bili-dyn-item__footer').length > 0) { card.querySelectorAll('div.bili-dyn-item__footer > div.bili-dyn-item__action').forEach((ele) => { ele.style.marginRight = '48px'; }); let buttonBar = card.getElementsByClassName('bili-dyn-item__footer')[0]; let downloadButton = document.createElement('div'); downloadButton.classList.add('bili-dyn-item__action'); downloadButton.classList.add('download-button'); let span = document.createElement('div'); span.classList.add('bili-dyn-action'); let icon = document.createElement('i'); icon.style.width = '20px'; icon.style.height = '20px'; icon.style.transform = 'scale(0.8)'; icon.style.backgroundImage = downloadIcon; icon.style.backgroundRepeat = 'no-repeat'; icon.style.backgroundSize = '100% 100%'; icon.style.backgroundPosition = 'center'; let text = document.createElement('span'); text.textContent = '下载'; span.appendChild(icon); span.appendChild(text); downloadButton.appendChild(span); buttonBar.appendChild(downloadButton); downloadButton = buttonBar.getElementsByClassName('download-button')[0]; downloadButton.addEventListener('mouseover', function(event) { this.querySelector('i').style.backgroundImage = downloadIcon; // console.log('over'); }); downloadButton.addEventListener('mouseout', function(event) { this.querySelector('i').style.backgroundImage = downloadIcon; // console.log('out'); }); downloadButton.addEventListener('click', async function(event) { // console.log('click'); event.preventDefault(); const content = this.closest('div.bili-dyn-item__main'); // console.log(content); const opusCard = content.querySelector('[dyn-id]'); // console.log(opusCard.getAttribute('dyn-id')); const dynId = opusCard.getAttribute('dyn-id'); handleDynamicDownload(dynId); /*const list = content.querySelectorAll('div.bili-album__preview__picture,div.preview__picture__img.b-img'); // console.log(list); if (list.length > 0) { for (let j = 0; j < list.length; j++) { let imgUrl; if (list[j].querySelector('img')) { imgUrl = list[j].querySelector('img').src.split(/@/)[0]; } else { imgUrl = list[j].style.backgroundImage.split(/"|@/)[1]; } // console.log(imgUrl); if (imgUrl.startsWith('//')) { imgUrl = 'https:' + imgUrl; } const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1]; // console.log(imgName); GM_download(imgUrl, imgName); } }*/ }); } else if(GM_getValue('enableVideoDownload', false) && card.getElementsByClassName('bili-tabs__nav__items').length > 0) { // console.log('add video dynamic download button'); const buttonBar = card.getElementsByClassName('bili-tabs__nav__items')[0]; let downloadButton = document.createElement('div'); downloadButton.textContent = '下载'; downloadButton.classList.add('bili-tabs__nav__item'); downloadButton.addEventListener('click', function(event) { // console.log('click'); event.preventDefault(); const content = this.closest('div.card'); // console.log(content); const opusCard = content.querySelector('[dyn-id]'); // console.log(opusCard.getAttribute('dyn-id')); const dynId = opusCard.getAttribute('dyn-id'); handleDynamicDownload(dynId); /*const list = content.querySelectorAll('div.bili-album__preview__picture,div.preview__picture__img.b-img'); // console.log(list); if (list.length > 0) { for (let j = 0; j < list.length; j++) { let imgUrl; if (list[j].querySelector('img')) { imgUrl = list[j].querySelector('img').src.split(/@/)[0]; } else { imgUrl = list[j].style.backgroundImage.split(/"|@/)[1]; } // console.log(imgUrl); if (imgUrl.startsWith('//')) { imgUrl = 'https:' + imgUrl; } const imgName = imgUrl.split('/')[imgUrl.split('/').length - 1]; // console.log(imgName); GM_download(imgUrl, imgName); } }*/ }); buttonBar.appendChild(downloadButton); } } } function addPlayPageDownloadButton(buttonBar) { let buttonWrap = document.createElement('div'); buttonWrap.className = 'toolbar-left-item-wrap'; let button = document.createElement('div'); button.className = 'video-toolbar-left-item download-button'; let icon = document.createElement('i'); icon.className = 'video-toolbar-item-icon'; icon.style.width = '36px'; icon.style.height = '36px'; icon.style.backgroundImage = downloadIcon; icon.style.backgroundRepeat = 'no-repeat'; icon.style.backgroundSize = '100% 100%'; icon.style.backgroundPosition = 'center'; let text = document.createElement('span'); text.textContent = '下载'; button.appendChild(icon); button.appendChild(text); buttonWrap.appendChild(button); buttonBar.appendChild(buttonWrap); button.addEventListener('click', async function(event) { // console.log('click'); event.preventDefault(); const bvid = window.location.pathname.match(/BV[a-z|A-Z|0-9]{10}/g)[0]; // console.log(bvid); handleVideoDownload(bvid); }); } function handleOpusCard(card) { // console.log('handleOpusCard'); if (card.getElementsByClassName('bili-album').length > 0 || card.getElementsByClassName('horizontal-scroll-album').length > 0 || (GM_getValue('enableVideoDownload', false) && card.getElementsByClassName('bili-dyn-card-video').length > 0)) { addOpusDownloadButton(card); } } function handleCard(card) { // console.log('handleCard'); if (card.getElementsByClassName('bili-album').length > 0 || card.getElementsByClassName('bili-dyn-gallery').length > 0) { // console.log('add download button'); card.getElementsByClassName('bili-album__preview__picture__img').forEach((img) => { img.addEventListener('click', function(event) { addDownloadButton(card); }); }); addDownloadButton(card); } else if (GM_getValue('enableVideoDownload', false) && card.getElementsByClassName('bili-dyn-card-video').length > 0 && card.getElementsByClassName('bili-dyn-action like disabled').length === 0) { addDownloadButton(card); } } function bodyMouseOver(event) { // console.log('bodyMouseOver'); if (notLoaded) { // console.log('not loaded'); if (document.body.querySelector('div.bili-dyn-list')) { // console.log('feed'); const cards = document.body.querySelectorAll('div.bili-dyn-list div.bili-dyn-item'); // console.log(cards.length); if (cards.length > cardsTotal) { // console.log('cards'); cardsTotal = cards.length; // console.log(cardsTotal); for (let i = 0; i < cardsTotal; i++) { // console.log('card'); handleCard(cards[i]) // startIndex += 1; } if (cardsTotal > 0) { notLoaded = false; } // document.body.removeEventListener('mouseover', bodyMouseOver); } } else if (location.pathname.startsWith('/v/topic/detail') && document.body.querySelector('div.list-view.topic-list__flow-list')) { const cards = document.body.querySelectorAll('div.list-view.topic-list__flow-list div.bili-dyn-item'); console.log(cards.length); if (cards.length > cardsTotal) { // console.log('cards'); cardsTotal = cards.length; // console.log(cardsTotal); for (let i = 0; i < cardsTotal; i++) { // console.log('card'); handleCard(cards[i]) // startIndex += 1; } if (cardsTotal > 0) { notLoaded = false; } document.body.removeEventListener('mouseover', bodyMouseOver); } } else if(document.body.querySelector('div.bili-dyn-item')) { // console.log('found single card'); const card = document.body.querySelector('div.bili-dyn-item'); if (card) { handleCard(card); notLoaded = false; // document.body.removeEventListener('mouseover', bodyMouseOver); } } else if (document.body.querySelector('div.opus-detail')) { // console.log('found single opus card'); const card = document.body.querySelector('div.opus-detail'); if (card) { handleOpusCard(card); notLoaded = false; // document.body.removeEventListener('mouseover', bodyMouseOver); } } /* else if (GM_getValue('enableVideoDownload', false) && document.body.querySelector('div.video-toolbar-left-main')) { const buttonBar = document.body.querySelector('div.video-toolbar-left-main'); if (buttonBar) { addPlayPageDownloadButton(buttonBar); notLoaded = false; } }*/ } } if (location.pathname.startsWith('/v/topic/detail')) document.body.addEventListener('mouseover', bodyMouseOver); function showModal(event) { // console.log(addDlBtnMode); let bg = document.createElement('div'); bg.style.position = 'fixed'; bg.style.top = 0; bg.style.left = 0; bg.style.zIndex = 500; bg.style.backgroundColor = 'black'; bg.style.opacity = 0.5; let modal = document.createElement('div'); document.body.appendChild(bg); modal.style.position = 'fixed'; modal.style.width = '25rem'; modal.style.height = 'auto'; modal.style.maxHeight = '80vh'; modal.style.zIndex = 600; modal.style.backgroundColor = 'white'; modal.style.borderStyle = 'solid'; modal.style.borderWidth = '0.2rem'; modal.style.borderRadius = '0.5rem'; modal.style.borderColor = 'black'; modal.style.overflowX = 'hidden'; modal.style.overflowY = 'auto'; modal.style.fontSize = '1rem'; let titleBar = document.createElement('div'); titleBar.textContent = '欢迎使用“下载Bilibili动态页面图片”脚本'; titleBar.style.width = '100%'; titleBar.style.textAlign = 'center'; titleBar.style.backgroundColor = 'black'; titleBar.style.color = 'white'; titleBar.style.fontSize = '1rem'; titleBar.style.fontWeight = 'bold'; titleBar.style.paddingTop = '0.5rem'; titleBar.style.paddingBottom = '0.5rem'; titleBar.style.borderTopLeftRadius = '0.3rem'; titleBar.style.borderTopRightRadius = '0.3rem'; modal.appendChild(titleBar); let question1 = document.createElement('p'); question1.style.paddingLeft = '2rem'; question1.style.paddingRight = '2rem'; question1.style.marginTop = '1rem'; question1.style.marginBottom = '1rem'; let labelPicName = document.createElement('label'); labelPicName.textContent = '下载图片文件名'; labelPicName.setAttribute('for', 'dlPicName'); question1.appendChild(labelPicName); let inputPicName = document.createElement('input'); inputPicName.type = 'text'; inputPicName.id = 'dlPicName'; inputPicName.name = 'dlPicName'; inputPicName.style.marginTop = '0.5rem'; inputPicName.style.width = 'calc(100% - 1rem)'; inputPicName.style.padding = '0.1rem 0.2rem 0.1rem 0.2rem'; inputPicName.style.borderStyle = 'solid'; inputPicName.style.borderColor = 'gray'; inputPicName.style.borderWidth = '0.14rem'; inputPicName.style.borderRadius = '0.2rem'; inputPicName.defaultValue = GM_getValue('dlPicName', '{original}.{ext}'); question1.appendChild(inputPicName); let PicNameExplain1 = document.createElement('p'); PicNameExplain1.innerHTML = '{original} - 原文件名\n{username} - UP主名称\n{userid} - UP主ID\n{dynamicid} - 动态id\n{ext} - 文件后缀\n{index} - 图片序号\n{YYYY} {MM} {DD} {HH} {mm} {ss} - 原博发布时\n间的年份、月份、日期、小时、分钟、秒,可\n分开独立使用\n{content} - 动态文字内容(最多前25个字符)'; PicNameExplain1.style.marginTop = '0.5rem'; PicNameExplain1.style.marginBottom = '0'; PicNameExplain1.style.whiteSpace = 'pre'; PicNameExplain1.style.color = 'gray'; question1.appendChild(PicNameExplain1); /*let PicNameExplain2 = document.createElement('p'); PicNameExplain2.innerHTML = '<b>注意</b>:启用“打包下载”时,需区分多文件名称,\n避免重复而导致打包后只有一个文件,文件命\n名时,必须包含{original}、{index}中至少一个\n标签。'; PicNameExplain2.style.marginTop = '0.5rem'; PicNameExplain2.style.whiteSpace = 'pre'; PicNameExplain2.style.color = 'gray'; question1.appendChild(PicNameExplain2);*/ modal.appendChild(question1); let question2 = document.createElement('p'); question2.style.paddingLeft = '2rem'; question2.style.paddingRight = '2rem'; question2.style.marginTop = '1rem'; question2.style.marginBottom = '0'; let labelVideoDownload = document.createElement('label'); labelVideoDownload.setAttribute('for', 'enableVideoDownload'); labelVideoDownload.textContent = '开启视频下载'; labelVideoDownload.style.display = 'inline-block'; labelVideoDownload.style.paddingRight = '0.2rem'; question2.appendChild(labelVideoDownload); let inputVideoDownload = document.createElement('input'); inputVideoDownload.type = 'checkbox'; inputVideoDownload.id = 'enableVideoDownload'; inputVideoDownload.checked = GM_getValue('enableVideoDownload', false); question2.appendChild(inputVideoDownload); let videoDownloadExplain = document.createElement('p'); videoDownloadExplain.textContent = '目前Bilibili视频单文件下载最高只支持720P MP4格式。'; videoDownloadExplain.style.marginTop = '0.5rem'; videoDownloadExplain.style.marginBottom = '0'; videoDownloadExplain.style.color = 'gray'; question2.appendChild(videoDownloadExplain); let labelVidName = document.createElement('label'); labelVidName.textContent = '下载图片文件名'; labelVidName.setAttribute('for', 'dlVidName'); labelVidName.style.display = 'block'; labelVidName.style.marginTop = '0.5rem'; labelVidName.style.color = GM_getValue('enableVideoDownload', false) ? null : 'gray'; question2.appendChild(labelVidName); let inputVidName = document.createElement('input'); inputVidName.type = 'text'; inputVidName.id = 'dlVidName'; inputVidName.name = 'dlVidName'; inputVidName.style.marginTop = '0.5rem'; inputVidName.style.width = 'calc(100% - 1rem)'; inputVidName.style.padding = '0.1rem 0.2rem 0.1rem 0.2rem'; inputVidName.style.borderStyle = 'solid'; inputVidName.style.borderWidth = '0.14rem'; inputVidName.style.borderRadius = '0.2rem'; inputVidName.disabled = GM_getValue('enableVideoDownload', false) ? false : true; inputVidName.style.borderColor = GM_getValue('enableVideoDownload', false) ? 'gray' : 'lightgray'; inputVidName.defaultValue = GM_getValue('dlVidName', '{original}.{ext}'); question2.appendChild(inputVidName); let vidNameExplain1 = document.createElement('p'); vidNameExplain1.innerHTML = '{original} - 原文件名\n{username} - UP主名称\n{userid} - UP主ID\n{bvid} - 视频BVID\n{aid} - 视频AID\n{cid} - 视频CID\n{title} - 视频标题\n{ext} - 文件后缀\n{YYYY} {MM} {DD} {HH} {mm} {ss} - 原博发布时\n间的年份、月份、日期、小时、分钟、秒,可\n分开独立使用\n{content} - 视频简介(最多前25个字符)'; vidNameExplain1.style.marginTop = '0.5rem'; vidNameExplain1.style.marginBottom = '0'; vidNameExplain1.style.whiteSpace = 'pre'; vidNameExplain1.style.color = 'gray'; question2.appendChild(vidNameExplain1); modal.appendChild(question2); /*let question3 = document.createElement('p'); question3.style.paddingLeft = '2rem'; question3.style.paddingRight = '2rem'; question3.style.marginTop = '1rem'; question3.style.marginBottom = '0'; let labelZipMode = document.createElement('label'); labelZipMode.setAttribute('for', 'zipMode'); labelZipMode.textContent = '打包下载'; labelZipMode.style.display = 'inline-block'; labelZipMode.style.paddingRight = '0.2rem'; labelZipMode.style.color = GM_getValue('ariaMode', false) ? 'gray' : null; question3.appendChild(labelZipMode); let inputZipMode = document.createElement('input'); inputZipMode.type = 'checkbox'; inputZipMode.id = 'zipMode'; inputZipMode.checked = GM_getValue('zipMode', false); inputZipMode.disabled = GM_getValue('ariaMode', false); question3.appendChild(inputZipMode); let labelPackName = document.createElement('label'); labelPackName.textContent = '打包文件名'; labelPackName.setAttribute('for', 'packFileName'); labelPackName.style.display = 'block'; labelPackName.style.marginTop = '0.5rem'; labelPackName.style.color = (GM_getValue('zipMode', false) && !GM_getValue('ariaMode', false)) ? null : 'gray'; // labelPackName.style.display = GM_getValue('zipMode', false) ? 'block' : 'none'; question3.appendChild(labelPackName); let inputPackName = document.createElement('input'); inputPackName.type = 'text'; inputPackName.id = 'packFileName'; inputPackName.name = 'packFileName'; inputPackName.style.marginTop = '0.5rem'; inputPackName.style.width = 'calc(100% - 1rem)'; inputPackName.style.padding = '0.1rem 0.2rem 0.1rem 0.2rem'; inputPackName.style.borderStyle = 'solid'; inputPackName.style.borderColor = (GM_getValue('zipMode', false) && !GM_getValue('ariaMode', false)) ? 'gray' : 'lightgray'; inputPackName.style.borderWidth = '0.14rem'; inputPackName.style.borderRadius = '0.2rem'; inputPackName.defaultValue = GM_getValue('packFileName', '{mblogid}.zip'); // inputPackName.style.display = GM_getValue('zipMode', false) ? 'block' : 'none'; inputPackName.disabled = (GM_getValue('zipMode', false) && !GM_getValue('ariaMode', false)) ? false : true; question3.appendChild(inputPackName); let filePackExplain = document.createElement('p'); filePackExplain.textContent = '与“下载文件名称”规则相同,但{original}、{ext}、{index}除外'; filePackExplain.style.marginTop = '0.5rem'; filePackExplain.style.marginBottom = '0'; filePackExplain.style.color = 'gray'; // filePackExplain.style.display = GM_getValue('zipMode', false) ? 'block' : 'none'; question3.appendChild(filePackExplain); modal.appendChild(question3); let question4 = document.createElement('p'); question4.style.paddingLeft = '2rem'; question4.style.paddingRight = '2rem'; question4.style.marginTop = '1rem'; question4.style.marginBottom = '0'; let labelRetweetMode = document.createElement('label'); labelRetweetMode.setAttribute('for', 'retweetMode'); labelRetweetMode.textContent = '单独设置转发微博下载文件名称'; labelRetweetMode.style.display = 'inline-block'; labelRetweetMode.style.paddingRight = '0.2rem'; question4.appendChild(labelRetweetMode); let inputRetweetMode = document.createElement('input'); inputRetweetMode.type = 'checkbox'; inputRetweetMode.id = 'retweetMode'; inputRetweetMode.checked = GM_getValue('retweetMode', false); question4.appendChild(inputRetweetMode); let labelRetweetFileName = document.createElement('label'); labelRetweetFileName.textContent = '转发微博下载文件名称'; labelRetweetFileName.setAttribute('for', 'retweetFileName'); labelRetweetFileName.style.display = 'block'; labelRetweetFileName.style.marginTop = '0.5rem'; labelRetweetFileName.style.color = GM_getValue('retweetMode', false) ? null : 'gray'; // labelPackName.style.display = GM_getValue('retweetMode', false) ? 'block' : 'none'; question4.appendChild(labelRetweetFileName); let inputRetweetFileName = document.createElement('input'); inputRetweetFileName.type = 'text'; inputRetweetFileName.id = 'retweetFileName'; inputRetweetFileName.name = 'retweetFileName'; inputRetweetFileName.style.marginTop = '0.5rem'; inputRetweetFileName.style.width = 'calc(100% - 1rem)'; inputRetweetFileName.style.padding = '0.1rem 0.2rem 0.1rem 0.2rem'; inputRetweetFileName.style.borderStyle = 'solid'; inputRetweetFileName.style.borderColor = 'lightgray'; inputRetweetFileName.style.borderWidth = '0.14rem'; inputRetweetFileName.style.borderRadius = '0.2rem'; inputRetweetFileName.defaultValue = GM_getValue('retweetFileName', '{original}.{ext}'); // inputRetweetFileName.style.display = GM_getValue('retweetMode', false) ? 'block' : 'none'; inputRetweetFileName.disabled = GM_getValue('retweetMode', false) ? false : true; question4.appendChild(inputRetweetFileName); let retweetFileNameExplain = document.createElement('p'); retweetFileNameExplain.textContent = '除“下载文件名”规则外,额外标签如下:\n{re.mblogid} - 转博mblogid\n{re.username} - 转发博主名称\n{re.userid} - 转发博主ID\n{re.uid} - 转博uid\n{re.content} - 转发博文内容(最多前25个字符)\n{re.YYYY} {re.MM} {re.DD} {re.HH} {re.mm} {re.ss}\n - 原博发布时间的年份、月份、日期、小时、\n分钟、秒,可分开独立使用'; retweetFileNameExplain.style.marginTop = '0.5rem'; retweetFileNameExplain.style.whiteSpace = 'pre'; retweetFileNameExplain.style.marginBottom = '0'; retweetFileNameExplain.style.color = 'gray'; // retweetFileNameExplain.style.display = GM_getValue('retweetMode', false) ? 'block' : 'none'; question4.appendChild(retweetFileNameExplain); let labelRetweetPackName = document.createElement('label'); labelRetweetPackName.textContent = '转发微博打包文件名'; labelRetweetPackName.setAttribute('for', 'retweetPackFileName'); labelRetweetPackName.style.display = 'block'; labelRetweetPackName.style.marginTop = '0.5rem'; labelRetweetPackName.style.color = (GM_getValue('zipMode', false) && GM_getValue('retweetMode', false) && !GM_getValue('ariaMode', false)) ? null : 'gray'; // labelRetweetPackName.style.display = GM_getValue('zipMode', false) ? 'block' : 'none'; question4.appendChild(labelRetweetPackName); let inputRetweetPackName = document.createElement('input'); inputRetweetPackName.type = 'text'; inputRetweetPackName.id = 'retweetPackFileName'; inputRetweetPackName.name = 'retweetPackFileName'; inputRetweetPackName.style.marginTop = '0.5rem'; inputRetweetPackName.style.width = 'calc(100% - 1rem)'; inputRetweetPackName.style.padding = '0.1rem 0.2rem 0.1rem 0.2rem'; inputRetweetPackName.style.borderStyle = 'solid'; inputRetweetPackName.style.borderColor = 'lightgray'; inputRetweetPackName.style.borderWidth = '0.14rem'; inputRetweetPackName.style.borderRadius = '0.2rem'; inputRetweetPackName.defaultValue = GM_getValue('retweetPackFileName', '{mblogid}.zip'); // inputRetweetPackName.style.display = (GM_getValue('zipMode', false) && GM_getValue('retweetMode', false) && !GM_getValue('ariaMode', false)) ? 'block' : 'none'; inputRetweetPackName.disabled = (GM_getValue('zipMode', false) && GM_getValue('retweetMode', false) && !GM_getValue('ariaMode', false)) ? false : true; question4.appendChild(inputRetweetPackName); let retweetPackExplain = document.createElement('p'); retweetPackExplain.textContent = '与“转发微博下载文件名称”规则相同,但{original}、{ext}、{index}除外'; retweetPackExplain.style.marginTop = '0.5rem'; retweetPackExplain.style.marginBottom = '0'; retweetPackExplain.style.color = 'gray'; // retweetPackExplain.style.display = (GM_getValue('zipMode', false) && GM_getValue('retweetMode', false) && !GM_getValue('ariaMode', false)) ? 'block' : 'none'; question4.appendChild(retweetPackExplain); modal.appendChild(question4); let question5 = document.createElement('p'); question5.style.paddingLeft = '2rem'; question5.style.paddingRight = '2rem'; question5.style.marginTop = '1rem'; question5.style.marginBottom = '0'; let labelAriaMode = document.createElement('label'); labelAriaMode.setAttribute('for', 'ariaMode'); labelAriaMode.textContent = '使用Aria2c远程下载'; labelAriaMode.style.display = 'inline-block'; labelAriaMode.style.paddingRight = '0.2rem'; question5.appendChild(labelAriaMode); let inputAriaMode = document.createElement('input'); inputAriaMode.type = 'checkbox'; inputAriaMode.id = 'ariaMode'; inputAriaMode.checked = GM_getValue('ariaMode', false); question5.appendChild(inputAriaMode); let ariaModeExplain = document.createElement('p'); ariaModeExplain.textContent = '使用此方式下载,无法使用打包功能,无法在页面右下角显示下载进度和结果。'; ariaModeExplain.style.marginTop = '0.5rem'; ariaModeExplain.style.marginBottom = '0'; ariaModeExplain.style.color = 'gray'; question5.appendChild(ariaModeExplain); let labelAriaRpcUrl = document.createElement('label'); labelAriaRpcUrl.textContent = 'RPC接口地址'; labelAriaRpcUrl.setAttribute('for', 'ariaRpcUrl'); labelAriaRpcUrl.style.display = 'block'; labelAriaRpcUrl.style.marginTop = '0.5rem'; labelAriaRpcUrl.style.color = GM_getValue('ariaMode', false) ? null : 'gray'; question5.appendChild(labelAriaRpcUrl); let inputAriaRpcUrl = document.createElement('input'); inputAriaRpcUrl.type = 'text'; inputAriaRpcUrl.id = 'ariaRpcUrl'; inputAriaRpcUrl.name = 'ariaRpcUrl'; inputAriaRpcUrl.style.marginTop = '0.5rem'; inputAriaRpcUrl.style.width = 'calc(100% - 1rem)'; inputAriaRpcUrl.style.padding = '0.1rem 0.2rem 0.1rem 0.2rem'; inputAriaRpcUrl.style.borderStyle = 'solid'; inputAriaRpcUrl.style.borderColor = 'lightgray'; inputAriaRpcUrl.style.borderWidth = '0.14rem'; inputAriaRpcUrl.style.borderRadius = '0.2rem'; inputAriaRpcUrl.defaultValue = GM_getValue('ariaRpcUrl', 'http://localhost:6800/jsonrpc'); // inputAriaRpcUrl.style.display = GM_getValue('ariaMode', false) ? 'block' : 'none'; inputAriaRpcUrl.disabled = GM_getValue('ariaMode', false) ? false : true; question5.appendChild(inputAriaRpcUrl); let inputAriaExplain = document.createElement('p'); inputAriaExplain.textContent = '如果接口地址不是localhost,需手动将地址添加到XHR白名单。'; inputAriaExplain.style.marginTop = '0.5rem'; inputAriaExplain.style.marginBottom = '0'; inputAriaExplain.style.color = 'gray'; question5.appendChild(inputAriaExplain); modal.appendChild(question5); inputRetweetMode.addEventListener('change', function(event) { if (event.currentTarget.checked) { // labelRetweetFileName.style.display = 'block'; // inputRetweetFileName.style.display = 'block'; // retweetFileNameExplain.style.display = 'block'; inputRetweetFileName.disabled = false; labelRetweetFileName.style.color = null; inputRetweetFileName.style.borderColor = 'gray'; } else { // labelRetweetFileName.style.display = 'none'; // inputRetweetFileName.style.display = 'none'; // retweetFileNameExplain.style.display = 'none'; inputRetweetFileName.disabled = true; labelRetweetFileName.style.color = 'gray'; inputRetweetFileName.style.borderColor = 'lightgray'; } if (event.currentTarget.checked && inputZipMode.checked && !inputAriaMode.checked) { inputRetweetPackName.disabled = false; labelRetweetPackName.style.color = null; inputRetweetPackName.style.borderColor = 'gray'; } else { inputRetweetPackName.disabled = true; labelRetweetPackName.style.color = 'gray'; inputRetweetPackName.style.borderColor = 'lightgray'; } }); inputZipMode.addEventListener('change', function(event) { if (event.currentTarget.checked) { // labelPackName.style.display = 'block'; // inputPackName.style.display = 'block'; // filePackExplain.style.display = 'block'; inputPackName.disabled = false; labelPackName.style.color = null; inputPackName.style.borderColor = 'gray'; } else { // labelPackName.style.display = 'none'; // inputPackName.style.display = 'none'; // filePackExplain.style.display = 'none'; inputPackName.disabled = true; labelPackName.style.color = 'gray'; inputPackName.style.borderColor = 'lightgray'; } if (event.currentTarget.checked && inputRetweetMode.checked) { inputRetweetPackName.disabled = false; labelRetweetPackName.style.color = null; inputRetweetPackName.style.borderColor = 'gray'; } else { inputRetweetPackName.disabled = true; labelRetweetPackName.style.color = 'gray'; inputRetweetPackName.style.borderColor = 'lightgray'; } }); inputAriaMode.addEventListener('change', function(event) { if (event.currentTarget.checked) { // labelAriaRpcUrl.style.display = 'block'; // inputAriaRpcUrl.style.display = 'block'; inputAriaRpcUrl.disabled = false; labelAriaRpcUrl.style.color = null; inputAriaRpcUrl.style.borderColor = 'gray'; inputZipMode.disabled = true; labelZipMode.style.color = 'gray'; } else { // labelAriaRpcUrl.style.display = 'none'; // inputAriaRpcUrl.style.display = 'none'; inputAriaRpcUrl.disabled = true; labelAriaRpcUrl.style.color = 'gray'; inputAriaRpcUrl.style.borderColor = 'lightgray'; inputZipMode.disabled = false; labelZipMode.style.color = null; } if (!event.currentTarget.checked && inputZipMode.checked) { inputPackName.disabled = false; labelPackName.style.color = null; inputPackName.style.borderColor = 'gray'; } else { inputPackName.disabled = true; labelPackName.style.color = 'gray'; inputPackName.style.borderColor = 'lightgray'; } if (!event.currentTarget.checked && inputZipMode.checked && inputRetweetMode.checked) { inputRetweetPackName.disabled = false; labelRetweetPackName.style.color = null; inputRetweetPackName.style.borderColor = 'gray'; } else { inputRetweetPackName.disabled = true; labelRetweetPackName.style.color = 'gray'; inputRetweetPackName.style.borderColor = 'lightgray'; } });*/ inputVideoDownload.addEventListener('change', function(event) { if (event.currentTarget.checked) { // labelRetweetFileName.style.display = 'block'; // inputRetweetFileName.style.display = 'block'; // retweetFileNameExplain.style.display = 'block'; inputVidName.disabled = false; labelVidName.style.color = null; inputVidName.style.borderColor = 'gray'; } else { // labelRetweetFileName.style.display = 'none'; // inputRetweetFileName.style.display = 'none'; // retweetFileNameExplain.style.display = 'none'; inputVidName.disabled = true; labelVidName.style.color = 'gray'; inputVidName.style.borderColor = 'lightgray'; } }); let okButton = document.createElement('button'); okButton.textContent = '确定'; okButton.style.paddingTop = '0.5rem'; okButton.style.paddingBottom = '0.5rem'; okButton.style.margin = '2rem'; okButton.style.backgroundColor = 'darkblue'; okButton.style.color = 'white'; okButton.style.fontSize = '1.5rem'; okButton.style.fontWeight = 'bold'; okButton.style.width = '21rem'; okButton.style.borderStyle = 'solid'; okButton.style.borderRadius = '0.5rem'; okButton.style.borderColor = 'black'; okButton.style.borderWidth = '0.2rem'; okButton.addEventListener('mouseover', function(event) { okButton.style.backgroundColor = 'blue'; }); okButton.addEventListener('mouseout', function(event) { okButton.style.backgroundColor = 'darkblue'; }); okButton.addEventListener('mousedown', function(event) { okButton.style.backgroundColor = 'darkblue'; }); okButton.addEventListener('mouseover', function(event) { okButton.style.backgroundColor = 'blue'; }); function resizeWindow(event) { // console.log('resize'); bg.style.width = document.documentElement.clientWidth.toString() + 'px'; bg.style.height = document.documentElement.clientHeight.toString() + 'px'; modal.style.top = (( document.documentElement.clientHeight - modal.offsetHeight ) / 2).toString() + 'px'; modal.style.left = (( document.documentElement.clientWidth - modal.offsetWidth ) / 2).toString() + 'px'; } okButton.addEventListener('click', function(event) { /*if (document.getElementById('zipMode').checked && !document.getElementById('dlPicName').value.includes('{original}') && !document.getElementById('dlPicName').value.includes('{index}')) { alert('启用“打包下载”时,需区分多文件名称,避免重复而导致打包后只有一个文件,文件命名时,必须包含{original}、{index}中至少一个标签。'); document.getElementById('dlPicName').focus(); return; }*/ let refreshFlag = false; if (document.getElementById('enableVideoDownload').checked !== GM_getValue('enableVideoDownload', false)) { refreshFlag = true; } GM_setValue('dlPicName', document.getElementById('dlPicName').value); GM_setValue('enableVideoDownload', document.getElementById('enableVideoDownload').checked); GM_setValue('dlVidName', document.getElementById('dlVidName').value); // GM_setValue('retweetMode', document.getElementById('retweetMode').checked); // GM_setValue('retweetFileName', document.getElementById('retweetFileName').value); // GM_setValue('zipMode', document.getElementById('zipMode').checked); // GM_setValue('packFileName', document.getElementById('packFileName').value); // GM_setValue('retweetPackFileName', document.getElementById('retweetPackFileName').value); // GM_setValue('ariaMode', document.getElementById('ariaMode').checked); // GM_setValue('ariaRpcUrl', document.getElementById('ariaRpcUrl').value); GM_setValue('isSet', settingVersion); if (refreshFlag) { alert('已' + (document.getElementById('enableVideoDownload').checked ? '开启' : '关闭') + '视频下载功能,将在页面刷新后生效。'); location.reload(); } window.removeEventListener('resize', resizeWindow); document.body.removeChild(modal); document.body.removeChild(bg); }); modal.appendChild(okButton); document.body.appendChild(modal); /*bg.addEventListener('click', function(event) { document.body.removeChild(modal); document.body.removeChild(bg); window.removeEventListener('resize', resizeWindow); });*/ resizeWindow(); window.addEventListener('resize', resizeWindow); } if(GM_getValue('isSet', null) !== settingVersion) { showModal(); } bodyMouseOver(); new MutationObserver((mutationList, observer) => { for (const mutation of mutationList) { // console.log(mutation); if (mutation.type === 'childList') { // if (!['svg'].includes(mutation.target.tagName)) console.log(mutation.target); /*for (const node of mutation.addedNodes) { // console.log(node.nodeType, node); if (node.nodeType === 1 && Object.keys(mutation.target.getElementsByClassName('video-like')).length > 0) { console.log(mutation.target, node); } }*/ if (mutation.target.tagName === 'DIV' && ['bili-dyn-list__items', 'content'].includes(mutation.target.className)) { // console.log(mutation.addedNodes); for (const node of mutation.addedNodes) { // console.log(node); handleCard(node); } } else if (mutation.target.tagName === 'DIV' && mutation.target.parentNode.className === 'list-view topic-list__flow-list') { // vconsole.log(mutation.addedNodes); for (const node of mutation.addedNodes) { // console.log(node); handleCard(node); } } else if (GM_getValue('enableVideoDownload', false) && mutation.target.tagName === 'BODY') { for (const node of mutation.addedNodes) { if (node.nodeType === 1 && node.tagName === 'DIV' && node.classList.contains('lt-row')) { // console.log(mutation); const buttonBar = document.body.querySelector('div.video-toolbar-left-main'); if (buttonBar && !buttonBar.querySelector('div.download-button')) { // console.log(mutation); addPlayPageDownloadButton(buttonBar); } } } } } } }).observe(document.body, { attributes: false, childList: true, subtree: true }); let settingButton = document.createElement('button'); settingButton.textContent = '设置'; settingButton.style.position = 'fixed'; settingButton.style.top = '4rem'; settingButton.style.left = '0rem'; settingButton.style.fontSize = '0.7rem'; settingButton.style.backgroundColor = 'gray'; settingButton.style.color = 'white'; settingButton.style.borderWidth = '0.2rem'; settingButton.style.borderStyle = 'solid'; settingButton.style.borderRadius = '0.5rem'; settingButton.style.borderColor = 'lightgrey'; settingButton.style.zIndex = 400; settingButton.style.paddingLeft = '1rem'; settingButton.style.paddingRight = '1rem'; settingButton.style.paddingTop = '0.2rem'; settingButton.style.paddingBottom = '0.2rem'; settingButton.addEventListener('mouseover', function(event) { settingButton.style.backgroundColor = 'darkgray'; settingButton.style.color = 'black'; }); settingButton.addEventListener('mouseout', function(event) { settingButton.style.backgroundColor = 'gray'; settingButton.style.color = 'white'; }); settingButton.addEventListener('mousedown', function(event) { settingButton.style.backgroundColor = 'gray'; settingButton.style.color = 'white'; }); settingButton.addEventListener('mouseup', function(event) { settingButton.style.backgroundColor = 'darkgray'; settingButton.style.color = 'black'; }); settingButton.addEventListener('click', showModal); document.body.appendChild(settingButton); GM_registerMenuCommand('设置', showModal, "0"); })();