任何页面,sci论文下载,显示期刊信息
// ==UserScript== // @name sciDownload // @namespace http://tampermonkey.net/ // @version 4.1 // @description 任何页面,sci论文下载,显示期刊信息 // @author Polygon // @icon ###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###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 // @match *://*/* // @grant unsafeWindow // @grant GM_xmlhttpRequest // @grant GM_download // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @grant GM_openInTab // @grant unsafeWindow // @grant window.close // @connect * // @run-at document-idlm // ==/UserScript== (function() { 'use strict' const notification = (function() { 'use strict'; GM_addStyle(` #notification { box-sizing: border-box; position: fixed; left: calc(50% - 365.65px / 2); display: flex; flex-direction: row; align-items: center; justify-content: center; height: 50px; background-color: #ff7675; border-radius: 50px; padding: 0 0px 0px 20px; top: -50px; transition: top .5s ease-out; z-index: 9999999999; } #notification .content { display: flex; align-items: center; justify-content: center; color: white; font-size: 25px; } #notification .closeBox { margin: 0 10px; transform: rotate(90deg); cursor: pointer; } #notification .closeBox .progress { margin: 0 10px; cursor: pointer; } #notification .closeBox .progress .circle { stroke-dasharray: 100; animation: progressOffset 0s linear; } @keyframes progressOffset { from { stroke-dashoffset: 100; } to { stroke-dashoffset: 0; } } `) return { open(info, timeout, autoClose=true) { let eles = document.querySelectorAll('#notification') for (let i=0;i<eles.length;i++) { this.close(eles[i]) } this.box = document.createElement('div') this.box.setAttribute('id', 'notification') this.box.innerHTML = ` <div class="content"></div> <svg class="closeBox" width="40" height="40"> <g class="close" style="stroke: white; stroke-width: 2; stroke-linecap: round;"> <line x1="13" y1="13" x2="27" y2="27"/> <line x1="13" y1="27" x2="27" y2="13"/> </g> <g class="progress" fill="transparent" stroke-width="3"> <circle class="background" cx="20" cy="20" r="16" stroke="rgba(255,255,255,0.15)"/> <circle class="circle" cx="20" cy="20" r="16" stroke="rgba(255,255,255,1)"/> </g> </svg> ` document.body.appendChild(this.box) this.box.querySelector('.content').innerHTML = info let width = getComputedStyle(this.box).width this.box.style.left = `clac(50%-${width}/2)` this.box.querySelector('.closeBox .progress .circle').style['animation-duration'] = `${timeout}s` this.box.style.top = '100px' this.box.querySelector('.closeBox .progress').addEventListener('click', () => { console.log('you close...') this.close() console.log('you clear...') }) if (autoClose) { setTimeout(() => { console.log('timeout close...') this.close() console.log('timeout clear ...') }, timeout * 1000) } }, close(ele=null) { if (!ele) {ele=this.box} ele.style['transition-duration'] = '.23s' ele.style['transition-timing-function'] = 'eaer-out' ele.style.top = '-50px' setTimeout(() => { try { document.body.removeChild(this.box) } catch { console.log('clear') } }, 1000) } } })(); const utils = { api: 'http://muise.icu:5000/sciDownload', doiRegex: new RegExp(/10\.\d{4,9}\/[-\._;\(\)\/:A-z0-9]+/), pdfRegex: /(content-type|Content-Type).+(pdf|binary|application|stream)/g, timeout: 25, autoMax: {b: true, time: 1}, scihubURL: 'sci-hub.ee', // 适配ipad,userscript暂不支持GM_getValue,GM_setValue get switchState() { if (typeof(GM_getValue) == 'undefined') { return this._switchSate } else { return GM_getValue('sciDownload-state', 'max') } }, set switchState(value) { if (typeof(GM_setValue) == 'undefined') { this._switchSate = value } else { GM_setValue('sciDownload-state', value) } }, _switchSate: 'max', color: { success: '#e74c3c', flash: '#00b894', fail: '#2c3e50' }, svg: { doi: `<svg class="progressBox" width="40" height="40"> <g fill="transparent" stroke-width="2.5"> <circle class="progress-background" cx="20" cy="20" r="11" stroke="rgba(255,255,255,0.23)"/> <circle class="progress" cx="20" cy="20" r="11" stroke="rgba(255,255,255,1)" stroke-linecap="round"/> </g> </svg>`, pdf: `<svg viewBox="0 0 #### ####" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2519" width="30" height="30"><path d="M478.08 192.192a58.88 58.88 0 0 1 46.208 23.168c23.104 32.064 21.312 99.84-8.832 199.68A536.832 536.832 0 0 0 625.664 557.44c37.312-7.04 74.688-12.416 112-12.416 83.52 1.792 96 41.024 94.144 64.128 0 60.544-58.688 60.544-88.896 60.544v0.064a226.048 226.048 0 0 1-131.52-53.504c-72.832 16-142.144 39.168-211.456 67.712C344.96 782.08 293.312 832 248.96 832c-8.96 0-19.584-1.792-26.752-7.168A52.48 52.48 0 0 1 192 776.704c0-16 3.52-60.608 172.352-133.568a1267.776 1267.776 0 0 0 94.208-221.056c-21.312-42.752-67.584-147.84-35.584-201.344 10.688-19.648 32-30.272 55.168-28.544z m-118.4 475.84c-42.688 20.736-95.232 58.368-90.112 82.368 3.392 16 21.888 10.56 55.424-16.384 21.056-27.072 24.512-41.088 34.688-65.984z m386.304-79.808c-17.152 0-32.384 0-49.536 7.68 19.072 19.2 29.504 28.608 48.64 32.448 13.312 3.84 39.104 10.752 46.72-9.28 7.68-19.968-7.616-30.848-45.824-30.848zM494.08 492.8a2572.16 2572.16 0 0 1-46.592 104.704l104.704-34.88c-21.76-22.272-41.344-43.392-58.112-69.824z m-16.64-223.488c-10.496 1.664-16.192 15.04-18.88 24.512-5.504 27.008 8.96 57.088 27.072 78.528 14.912-22.528 18.688-43.328 12.928-73.92-6.976-20.48-14.016-30.208-21.12-29.12z" fill="#ffffff" p-id="2520"></path></svg>`, switch: `<svg width="40" height="40"> <g stroke="white" stroke-width="3" stroke-linecap="round"> <line x1="10" y1="20" x2="30" y2="20"/> <line class="switch" x1="10" y1="20" x2="30" y2="20"/> </g> </svg>` }, style() { let div = document.createElement('div') div.innerHTML = this.svg.doi div.style.opacity = '0' document.body.appendChild(div) try{ this.progressTotaLength = div.querySelector('.progress').getTotalLength() } catch { this.progressTotaLength = 68.66967010498047 } document.body.removeChild(div) return ` #sciDownloadBox { display: flex; position: fixed; height: 40px; bottom: 75px; font-family: NexusSans,Arial,Helvetica,Lucida Sans Unicode,Microsoft Sans Serif,Segoe UI Symbol,STIXGeneral,Cambria Math,Arial Unicode MS,sans-serif; font-size: 18px; cursor: pointer; box-shadow: 0px 0px 20px rgba(0, 0, 0, .1); transition: left .23s ease-out, opacity .23s, right .23s ease-out; z-index: 9999999999; } #sciDownloadBox * { box-sizing: border-box; } #sciContent { position: relative; overflow: hidden; box-sizing: border-box; display: flex; height: 40px; align-items: center; justify-content: space-around; vertical-align: middle; white-space: nowrap; color: white; background-color: ${this.color.fail}; opacity: 0.72; transition: width .23s ease-out, opacity .23s, background-color .23s; } #sciContent[loading] #sciState .progress { opacity: 0; } #sciContent[loading]::before { content: ''; position: absolute; left: 0; top: 0; width: 100%; height: 100%; opacity: 0; background-color: white; z-index: -1; animation: loading 2.3s linear infinite; } @keyframes loading { from { opacity: 0; } 50% { opacity: 0.3; } to { opacity: 0; } } #sciSwitch { width: 40px; height: 40px; color: white; background-color: #00b894; opacity: 0.72; transition: width .23s ease-out, opacity .23s; z-index: 1; } #sciContent #sciState { position: relative; overflow: hidden; display: flex; align-items: center; justify-content: center; width: 40px; height: 40px; opacity: 1; } #sciContent #sciText { position: relative; overflow: hidden; height: 100%; display: flex; align-items: center; justify-content: center; color: white; padding-left: 5px; padding-right: 10px; opacity: 1; text-decoration: none; transition: width .23s ease-out; } #sciContent:hover { opacity: 0.99 !important; } #sciSwitch:hover { opacity: 1 !important; } /* left svg progress */ #sciContent #sciState .progressBox { transform: rotate(-90deg); } #sciContent #sciState .progress { stroke-dasharray: ${this.progressTotaLength}; stroke-dashoffset: ${this.progressTotaLength}; transition: stroke-dashoffset .23s linear; } #sciContent[progress] #sciState .progress { animation: progressOffset ${this.timeout}s linear forwards; } @keyframes rotator { from { transform: rotate(-90deg); } to { transform: rotate(180deg); } } #sciContent[download-noprogress] #sciState .progressBox { animation: rotator 2.3s linear infinite; } @keyframes dash { from { stroke-dashoffset: ${this.progressTotaLength}; } 50% { stroke-dashoffset: ${this.progressTotaLength / 4}; transform:rotate(135deg); } to { stroke-dashoffset: ${this.progressTotaLength}; transform:rotate(450deg); } } #sciContent[download-noprogress] #sciState .progress { stroke-dashoffset: 0; transform-origin: center; animation: dash 2.3s ease-in-out infinite; } @keyframes progressOffset { from { stroke-dashoffset: ${this.progressTotaLength}; } to { stroke-dashoffset: 0; } } @keyframes progressRecover { to { stroke-dashoffset: ${this.progressTotaLength}; } } /* progress animation */ @keyframes progress { to { width: 100%; } } /* switch button animation */ #sciSwitch svg .switch { transform: rotate(0deg); transform-origin: center center; transition: transform .23s ease-out .15s; } /* ripple effect */ #sciDownloadBox .ripple { position: absolute; background: #fff; transform: translate(-50%, -50%); pointer-events: none; border-radius: 50%; animation: ripple 1s linear; } @keyframes ripple{ from { width: 0px; height: 0px; opacity: 0.5; } to { width: 500px; height: 500px; opacity: 0; } } ` }, initBox(doi) { let createBox = () => { this.sciDownloadBox = document.createElement('div') this.sciDownloadBox.setAttribute('id', 'sciDownloadBox') this.sciDownloadBox.setAttribute('doi', doi) this.sciDownloadBox.innerHTML = ` <div id="sciSwitch">${this.svg.switch}</div> <div id="sciContent"> <div id="sciState"></div> <a id="sciText"></a> </div> ` document.body.appendChild(this.sciDownloadBox) // 绑定变量 this.sciContent = this.sciDownloadBox.querySelector('#sciContent') this.sciState = this.sciDownloadBox.querySelector('#sciState') this.sciText = this.sciDownloadBox.querySelector('#sciText') // 改变doi文字 this.changeContent(this.svg.doi, doi) // 缓入准备 this.sciDownloadBox.style.right = -this.getElementWidth(this.sciDownloadBox) + 'px' setTimeout(() => { // 设置right属性,触发缓入动画 this.sciDownloadBox.style.right = '0px' }, 230) this.sciSwitch = this.sciDownloadBox.querySelector('#sciSwitch') // 最大化/最小化按钮点击事件绑定 this.sciSwitch.addEventListener('click', this.switchEvent) // 涟漪效果点击事件 this.sciState.addEventListener('click', this.rippleClickEvent) this.sciText.addEventListener('click', this.rippleClickEvent) // 当窗口调整时,自适应 window.onresize = () => { setTimeout(() => { this.sciSwitch.click() this.sciSwitch.click() }) } } // 添加sciTool逻辑 if (this.sciDownloadBox) { // 存在,设置left属性缓出 this.sciDownloadBox.style.left = getComputedStyle(utils.sciDownloadBox).left this.sciDownloadBox.style.left = document.body.clientWidth + 'px' GM_addStyle(` #sciContent[progress] #sciState .progress { animation: progressOffset 25s linear forwards; } `) setTimeout(() => { // 缓出结束,让其消失,并创建新的 this.sciDownloadBox.remove() createBox.apply(this) }, 230); } else { // 可能首次打开页面,直接创建 createBox.apply(this) } }, rippleClickEvent(event) { // 这一步让sciState内svg的点击事件传播到sciState,而svg本身不产生动画效果 let parent for (let i=0;i<event.path.length;i++) { if (event.path[i].id.match(/(sciText|sciState)/)) { parent = event.path[i] break } } let x = event.offsetX let y = event.offsetY let ripple = document.createElement("span") ripple.setAttribute('class', 'ripple') ripple.style.left = `${x}px` ripple.style.top = `${y}px` parent.appendChild(ripple) // timeout数值越大涟漪扩散越慢 setTimeout(() => { ripple.remove() }, 1000) event.stopPropagation(); }, switchEvent(event) { if (utils.switchState == 'max') { utils.sciDownloadBox.style.left = getComputedStyle(utils.sciDownloadBox).left utils.sciDownloadBox.style.left = document.body.clientWidth - 40 * 2 + 'px' utils.sciDownloadBox.style.right = -utils.getElementWidth(utils.sciDownloadBox) + 'px' utils.sciSwitch.querySelector('svg .switch').style.transform = 'rotate(90deg)' utils.switchState = 'min' } else if (utils.switchState == 'min') { utils.sciDownloadBox.style.left = '' utils.sciDownloadBox.style.right = '0px' utils.sciSwitch.querySelector('svg .switch').style.transform = 'rotate(0deg)' utils.switchState = 'max' } }, startProgress(doi) { if (document.querySelectorAll(`#sciDownloadBox[doi="${doi}"]`).length == 0) return this.sciContent.setAttribute('progress', '') // 有progress属性的有进度条动画 GM_addStyle(` /* 背景色进度条 */ #sciContent[progress] #sciText::before { content: ""; position: absolute; right: 0; bottom: 0; height: 40px; width: 0%; background-color: ${this.color.flash}; opacity: 1; z-index: -1; animation: progress ${this.timeout}s linear forwards; } `) }, getContentWidth(ele, content) { let oldContent = ele.innerHTML ele.innerHTML = content let width = this.getElementWidth(ele) ele.innerHTML = oldContent return width }, getElementWidth(ele) { return parseFloat(window.getComputedStyle(ele).width.replace('px', '')) }, changeContent(state, text, callback=null) { if (state) {this.sciState.innerHTML = state} let ele = this.sciText let oldWidth = this.getElementWidth(ele) let newWidth = this.getContentWidth(ele, text) ele.style.width = oldWidth + 'px' setTimeout(() => { ele.style.width = newWidth + 'px' ele.innerHTML = text setTimeout(() => { ele.style.width = 'fit-content' if (callback) {callback()} }, 230) if (this.switchState == 'min' & this.autoMax.b){ this.sciSwitch.click() setTimeout(() => { this.sciSwitch.click() }, this.autoMax.time * 1000 > 230 ? this.autoMax.time * 1000 : 230) } }, 230) }, getDoi() { let doi, select, res let selection = window.getSelection().toString() let sourceText = document.body.innerHTML res = selection.match(this.doiRegex) if (res) { doi = res[0] select = true } else { res = sourceText.match(this.doiRegex) if (res) { doi = res[0] select = false } } if (doi) { doi = doi.replace(/[\/\.]\w*?pdf/, '').split(';')[0] } return [doi, select] }, check_update_journal_info(journal_name) { let key = `sciDownload-journal-${journal_name}` console.log(key) if (!GM_getValue(key, undefined)) { GM_xmlhttpRequest({ method: 'POST', url: "https://easyscholar.cc/homeController/getQueryTable.ajax", data: `page=1&limit=10&sourceName=${journal_name}`, headers: { "Content-Type": "application/x-www-form-urlencoded" }, responseType: "json", onload: function (res) { let value if (res.response.data.length) { let info = res.response.data[0] value = `${info.paperName} | ${info.sciif}/${info.sciif5} | ${info.sciBase} | ${info.sciUp}` } else { value = journal_name || "No More Information" } console.log(value) GM_setValue(key, value) } }) } else { console.log('read from GM_getValue') console.log(GM_getValue(key, undefined)) } }, LocalSearch(doi) { // 1 开启动画效果 this.initBox(doi) setTimeout(() => { this.startProgress(doi) }, 230) // 2 生成data,这里按顺序查询,不太会用js的并发只能按照顺序了,应该不会很慢 let data = {journal_name: ""} let totalSource = 4 let failSource = 0 let setData = (value) => { if (value.url) { if (!Object.keys(data).includes("url")) { data = {...data, ...value} } } else { failSource += 1 } } // 2.1 查询unpaywall const unpaywall = `https://api.unpaywall.org/v2/${doi}[email protected]` GM_xmlhttpRequest({ method: 'GET', url: unpaywall, responseType: 'json', onload: function (res) { const unpaywallData = res.response let journal_name = unpaywallData['journal_name'] setTimeout(() => { utils.check_update_journal_info(journal_name) data.journal_name = journal_name }, 0); let url try { url = unpaywallData['best_oa_location']['url_for_pdf'] || unpaywallData['best_oa_location']['url_for_landing_page'] } catch { url = '' if (Object.prototype.hasOwnProperty.call(unpaywallData, 'title')) { console.log(`unpaywall: ${unpaywallData['title']}`) researchgate(unpaywallData['title']) // 交给researchgate } } setData({ message: 'unpaywall.org', url: url }) } }) // 2.2 查询scihub const scihub = `https://${this.scihubURL}/${doi}` GM_xmlhttpRequest({ method: 'GET', url: scihub, onload: function (res) { let url = res.response.match(/\/\/(.+pdf)[^\'\"]/) if (url) { url = 'https://' + url[1] } else { url = '' } setData({ message: utils.scihubURL, url: url }) } }) // 2.3 查询researchgate let isConsistent = (s1, s2) => { if (s1 == null || s2 == null) return false let matchArray = [], strArray = [s1, s2] for (let i=0;i<strArray.length;i++) { matchArray.push(strArray[i].toLowerCase().match(/\w+/g)) } [s1, s2] = matchArray if (s1.length != s2.length) return false let b = true for (let i=0;i<s1.length;i++) { if (s1[i] != s2[i]) return false } return b } let researchgate = (paperTitle) => { const url = `https://www.researchgate.net/search.SearchBox.html?query=${paperTitle}&activeTab=publication` GM_xmlhttpRequest({ method: 'GET', url: url, headers: {'accept': 'application/json'}, responseType: 'json', onload: function (res) { try { const resData = res.response['r###lt']['state']['searchSearch']['publication']['items'] console.log(resData) let url = '' let i = 0 while (i < resData.length) { let item = resData[i] if (!Object.prototype.hasOwnProperty.call(item['urls'], 'download')){ console.log(`researchgate: no url`) } else { const pdfURL = 'https://www.researchgate.net/' + item['urls']['download'] // 标题是否一致 if (isConsistent(item['title'], paperTitle)) { url = pdfURL console.log(`researchgate[${i}]: best title`) console.log(paperTitle) console.log(item['title']) break } else { console.log(`researchgate: not consistent`) } } i += 1 } setData({ message: 'researchgate.net', url: url }) } catch { console.log(res) } } }) } // 2.4 出版商地址 GM_xmlhttpRequest({ method: 'GET', url: `https://doi.org/${doi}`, onload: function (res) { let publisherURL = res.finalUrl // https://linkinghub.elsevier.com/retrieve/pii/S016980952100421X if (publisherURL.includes('linkinghub.elsevier.com')) { let pdfURL = 'https://www.sciencedirect.com/science/article/pii/' + publisherURL.match(/pii\/(.+)/)[1] + '/pdfft' console.log(pdfURL) GM_xmlhttpRequest({ method: 'GET', url: pdfURL, onload: function (res) { let url = '' res = res.response.match(/"(https?:\/\/pdf.+)"/) if (res && res.length == 2) { url = res[1] } else { url = '' } setData({ message: 'sciencedirect.com', url: url }) } }) } else if (publisherURL.includes('springer.com')) { let pdfURL = 'https://link.springer.com/content/pdf/' + publisherURL.match(/article\/(.+)/)[1] + '.pdf' console.log(pdfURL) let func = (res) => { let url let contentType = res.responseHeaders.match(/content-type:(.+)/)[1] console.log(contentType) if (contentType.includes('pdf')) { url = pdfURL } else { url = '' } setData({ message: 'springer.com', url: url }) } GM_xmlhttpRequest({ method: 'GET', url: pdfURL, onprogress: func, onload: func }) } else { setData({ message: 'sciencedirect.com', url: null }) } // 补充其他 } }) // 3 等待data结果,并提交至下一个函数 let total = 0 const interval = 10 let id = setInterval(() => { if (Object.keys(data).includes('url')) { console.log(data) utils.getPdf(data, doi) clearInterval(id) } else if (failSource == totalSource) { data = { message: 'NotSupport', url: '' } utils.getPdf(data, doi) clearInterval(id) } else if (total > this.timeout * 1000) { data = { message: 'Timeout', url: '' } utils.getPdf(data, doi) clearInterval(id) } total += interval }, interval) }, getPdf(data, doi) { if (document.querySelectorAll(`#sciDownloadBox[doi="${doi}"]`).length == 0) return // 进入下载,收回timeout倒计时的进度条 let progress = this.sciState.querySelector('svg .progress') let currentOffset = getComputedStyle(progress)['stroke-dashoffset'] GM_addStyle(` #sciContent[progress] #sciState .progress { stroke-dashoffset: ${currentOffset}; animation: progressRecover .23s linear; } `) setTimeout(() => { // 动画结束后改变sciContent状态 this.sciContent.removeAttribute('progress') }, 230) this.changeContent(null, data.message) // api相应速度太快,可能清除不掉过一段时间才出现的进度条, 检测一秒钟 let exit = true switch (data.message) { case 'NotSupport': this.log('不支持该文章,退出...') break case 'Timeout': this.log('请求超时,退出...') break default: this.log('请求pdf中...') exit = false } setTimeout(() => { progress.style['stroke-dashoffset'] = '0' }, 230*2) if (exit) { return } utils.sciContent.style['background-color'] = utils.color.flash let openInTab = (url) => { if (typeof(GM_openInTab) == 'undefined') { var a = document.createElement('a') a.href = url a.target = '_blank' document.body.appendChild(a) a.click() a.remove() } else { GM_openInTab(url, {active: false, insert: true}) } } utils.sciText.onclick = () => { openInTab(data.url) } utils.sciState.onclick = () => { openInTab(data.url) } // 开始缓存同时尝试打开链接,可将下行反注释即可 // window.open(pdfURL) let failSetting = () => { progress.style['stroke-dashoffset'] = '0' utils.sciContent.style['background-color'] = utils.color.fail } let lastTime, currentTime, lastDone=0, currentDone, size setTimeout(() => { this.sciContent.setAttribute('loading', '') progress.style['stroke-dashoffset'] = utils.progressTotaLength }, 230*2) GM_xmlhttpRequest({ method: 'GET', url: data.url, responseType: 'blob', onprogress: function(res) { if (document.querySelectorAll(`#sciDownloadBox[doi="${doi}"]`).length == 0) return utils.sciContent.removeAttribute('loading') if (!res.responseHeaders.match(utils.pdfRegex)) { failSetting() return } // 波浪冲击效果待完成 let rippleUp = (opacity) => { let ripple = document.createElement("span") ripple.setAttribute('class', 'ripple') ripple.style.backgroundColor = 'white' ripple.style.zIndex = -1 ripple.style.opacity = opacity ripple.style.left = `${utils.getElementWidth(utils.sciContent)}px` ripple.style.top = '20px' utils.sciContent.appendChild(ripple) // timeout数值越大涟漪扩散越慢 setTimeout(() => { ripple.remove() }, 1E3) } currentTime = new Date().getTime() currentDone = res.done if (!lastTime | currentTime - lastTime >= 300) { size = (currentDone - lastDone) / #### / #### * 5 rippleUp(size > 1 ? 1 : size) lastTime = currentTime lastDone = currentDone let tip let key = `sciDownload-journal-${data.journal_name}` let journal_info = GM_getValue(key, "...") if (res.lengthComputable) { utils.sciContent.setAttribute('download-progress', '') let percent = res.done / res.total progress.style['stroke-dashoffset'] = utils.progressTotaLength * (1 - percent) // 下载进度条 GM_addStyle(` #sciContent[download-progress] #sciText::before { content: ""; position: absolute; right: 0; bottom: 0; height: 40px; width: ${percent * 100}%; background-color: white; opacity: 0.5; z-index: -1; transition: width .23s linear; } `) tip = `${journal_info} | ${(res.done / #### / ####).toFixed(2)}M | ${(res.total / #### / ####).toFixed(2)}M | ${(percent * 100).toFixed(2)}% | ${data.message}` } else { tip = `${journal_info} | ${(res.done / #### / ####).toFixed(2)}M | --M | --% | ${data.message}` utils.sciContent.setAttribute('download-noprogress', '') } utils.sciContent.setAttribute('title', tip) utils.sciSwitch.setAttribute('title', tip) } }, onload: function(res) { if (document.querySelectorAll(`#sciDownloadBox[doi="${doi}"]`).length == 0) return let size = (res.response.size / #### / ####).toFixed(2) let key = `sciDownload-journal-${data.journal_name}` let journal_info = GM_getValue(key, "...") let tip = `${journal_info} | ${size}M | ${data.message}` utils.sciContent.setAttribute('title', tip) utils.sciSwitch.setAttribute('title', tip) setTimeout(() => { utils.sciContent.removeAttribute('loading') if (!res.responseHeaders.match(utils.pdfRegex)) { failSetting() notification.open('pdf加载失败了,亲自点一下吧~', 3) return } utils.sciContent.removeAttribute('download-progress') utils.sciContent.removeAttribute('download-noprogress') setTimeout(() => { utils.sciContent.style['background-color'] = utils.color.success }, 230); let fileURL = URL.createObjectURL(new Blob([res.response], {type: 'application/pdf'})) let title = doi.split('/').slice(1).join('/') let titleRes = res.responseHeaders.match(/filename=(.+)/) if (titleRes) { title = decodeURI(titleRes[1].split(';')[0]).replace('.pdf', '').replace('"', '').replace('"', '') } utils.sciText.removeAttribute('href') let downloadPdf = (fileURL, title) => { let aTag = document.createElement('a') aTag.setAttribute('href', fileURL) aTag.setAttribute('download', `${title}.pdf`) aTag.click() } utils.sciText.onclick = () => { setTimeout(() => { if ((typeof(GM_setValue) == 'undefined')) { downloadPdf(fileURL, title) } else { let win = window.open() win.document.write(`<iframe name="${title}" src="${fileURL}" frameborder="0" style="border:0; top:0px; left:0px; bottom:0px; right:0px; width:100%; height:100%;" allowfullscreen></iframe>`) win.document.title = title } }, 1E3) } utils.log('缓存pdf成功') utils.sciContent.style['background'] = utils.color.success utils.changeContent(utils.svg.pdf, title) utils.sciState.onclick = () => { downloadPdf(fileURL, title) } // 防止此时还没变更 let key = `sciDownload-journal-${data.journal_name}` let value if (utils.sciContent.getAttribute('title').includes('...')) { // 设置一个循环等着 let id = setInterval(() => { value = GM_getValue(key, undefined) if (value !== undefined) { let tip = utils.sciContent.getAttribute('title') .replace('...', value) utils.sciContent.setAttribute('title', tip) utils.sciSwitch.setAttribute('title', tip) clearInterval(id) } }, 200); } }, 230*3); } }) }, log(text) { console.log('[sciDownload]', text) } } try{ GM_addStyle(utils.style()) } catch { utils.log('添加style失败,退出...') return } let lastDoi = null let lastIsSelect = false setInterval(function () { let [doi, select] = utils.getDoi() if (!doi | doi == lastDoi | (lastIsSelect && !select)) { return } lastDoi = doi lastIsSelect = select let a = 'background: #00b894; color: #fff; opacity: 0.75;' let b = 'background: #2c3e50; color: #fff; opacity: 0.75;' console.log(`%c sciDownload %c ${doi} `, a, b) utils.LocalSearch(doi) }, 500) })();