// ==UserScript== // @name sciDownload // @namespace http://tampermonkey.net/ // @version 0.7 // @description 涟漪效果 // @author Polygon // @icon  // @match *://*/* // @grant unsafeWindow // @grant GM_xmlhttpRequest // @grant GM_download // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @grant unsafeWindow // @connect * // @run-at document-end // @downloadURL none // ==/UserScript== (function() { 'use strict' const utils = { api: 'https://service-8d4l1brd-1256272652.sh.apigw.tencentcs.com/sciDownload', doiRegex: new RegExp(/10\.\d{4,9}\/[-\._;\(\)\/:A-z0-9]+/), timeout: 25, height: 40, autoMax: {b: true, time: 1}, // 如果缩小状态下,sciState内容改变是否自动最大化以及弹出后自动收回的时间 // 这个b会自动记忆,如果最后一次运行状态是min,下次自动min get switchState() { return GM_getValue('sciDownload-state', 'max') }, set switchState(value) { GM_setValue('sciDownload-state', value) }, color: { success: '#e74c3c', flash: '#00b894', fail: '#2c3e50' }, svg: { doi: ` `, url: ` `, pdf: ``, switch: ` ` }, style() { let div = document.createElement('div') div.innerHTML = this.svg.doi div.style.opacity = '0' document.body.appendChild(div) const total = div.querySelector('.progress').getTotalLength() document.body.removeChild(div) return ` .sciTool { display: flex; position: fixed; height: ${this.height}px; 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; } .sciTool * { box-sizing: border-box; } #sciContent { box-sizing: border-box; display: flex; 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; } #sciSwitch { width: ${this.height}px; height: ${this.height}px; 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: ${this.height}px; height: ${this.height}px; 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: 1 !important; } #sciSwitch:hover { opacity: 1 !important; } /* left svg progress */ #sciContent #sciState .progressBox { transform: rotate(-90deg); } #sciContent #sciState .progress { stroke-dasharray: ${total}; animation: progressOffset ${this.timeout}s linear; } @keyframes progressOffset { from { stroke-dashoffset: ${total}; } to { stroke-dashoffset: 0; } } /* right div progress */ #sciProgressBox { right: 0px; } #sciProgress { position: relative; display: flex; justify-content: center; align-items: center; width: 0px; background-color: ${this.color.flash}; height: 40px; overflow: hidden; transition: width ${this.timeout}s linear; } #sciProgressText { position: absolute; color: white; right: 0px; padding-left: 5px; padding-right: 10px; } /* flash */ @keyframes sciFlash { from { opacity: 1; background-color: ${this.color.success}; } 50% { opacity: 0.72; background-color: ${this.color.flash}; } to { opacity: 1; background-color: ${this.color.success}; } } #sciDownloadBox .loading { animation: sciFlash 1.2s infinite !important; } /* switch button animation */ #sciSwitch svg .switch { transform: rotate(0deg); transform-origin: center center; transition: transform .23s ease-out .15s; } /* ripple effect */ .sciTool .ripple { position: absolute; background: #fff; transform: translate(-50%, -50%); pointer-events: none; border-radius: 50%; animation: ripple 1s linear infinite; } @keyframes ripple{ 0% { width: 0px; height: 0px; opacity: 0.5; } 100% { width: 500px; height: 500px; opacity: 0; } } ` }, initBox(doi) { if (this.sciContent) { if (this.sciContent.classList.contains('loading')) {this.sciContent.classList.remove('loading')} let sciProgressBox = document.querySelector('#sciProgressBox') if (sciProgressBox) {sciProgressBox.parentNode.removeChild(sciProgressBox)} this.sciContent.style['background-color'] = this.color.fail this.sciText.removeAttribute('href') this.changeContent(this.svg.doi, doi, () => { this.startProgress() }) } else { this.sciDownloadBox = document.createElement('div') this.sciDownloadBox.setAttribute('id', 'sciDownloadBox') this.sciDownloadBox.setAttribute('class', 'sciTool') this.sciDownloadBox.innerHTML = `
${this.svg.switch}
${this.svg.doi}
${doi}
` document.body.appendChild(this.sciDownloadBox) this.sciContent = this.sciDownloadBox.querySelector('#sciContent') this.sciState = this.sciDownloadBox.querySelector('#sciState') this.sciText = this.sciDownloadBox.querySelector('#sciText') this.sciDownloadBox.style.right = -this.getElementWidth(this.sciDownloadBox) + 'px' setTimeout(() => { this.sciDownloadBox.style.right = '0px' setTimeout(() => { this.startProgress() }, 230*2) }, 230) this.sciSwitch = this.sciDownloadBox.querySelector('#sciSwitch') this.sciSwitch.onclick = this.switchEvent // 涟漪效果点击事件 this.sciState.addEventListener('click', this.rippleClickEvent) this.sciText.addEventListener('click', this.rippleClickEvent) // 调整窗口重适应 window.onresize = () => { setTimeout(() => { this.sciSwitch.click() this.sciSwitch.click() }) } } }, rippleClickEvent(event) { let parent // 过滤内部元素 for (let i=0;i { ripple.remove() }, 1000) }, switchEvent(event) { if (utils.switchState == 'max') { utils.sciDownloadBox.style.left = getComputedStyle(utils.sciDownloadBox).left utils.sciDownloadBox.style.left = document.body.clientWidth - utils.height * 2 + 'px' utils.sciDownloadBox.style.right = -utils.getElementWidth(utils.sciDownloadBox) + 'px' let sciProgressBox = document.querySelector('#sciProgressBox') if (sciProgressBox) { sciProgressBox.style.right = -utils.getElementWidth(utils.sciState) + 'px' sciProgressBox.style.left = document.body.clientWidth + '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' let sciProgressBox = document.querySelector('#sciProgressBox') if (sciProgressBox) { sciProgressBox.style.left = '' sciProgressBox.style.right = '0px' } utils.sciSwitch.querySelector('svg .switch').style.transform = 'rotate(0deg)' utils.switchState = 'max' } }, startProgress() { let ele = this.sciText let sciProgressBox = document.createElement('div') sciProgressBox.setAttribute('id', 'sciProgressBox') sciProgressBox.setAttribute('class', 'sciTool') let totalWidth = this.getElementWidth(ele) sciProgressBox.innerHTML = `
${ele.innerHTML}
` document.body.appendChild(sciProgressBox) let sciProgress = sciProgressBox.querySelector('#sciProgress') sciProgressBox.querySelector('#sciProgressText').style.width = `${totalWidth}px` setTimeout(() => { sciProgress.style.width = `${totalWidth}px` if (this.switchState == 'min') {this.switchState = 'max'; this.sciSwitch.click()} }, 100) sciProgress.addEventListener('click', this.rippleClickEvent) }, 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) { this.sciState.innerHTML = state let ele = this.sciText let sciProgressBox = document.querySelector('#sciProgressBox') if (sciProgressBox) {sciProgressBox.parentNode.removeChild(sciProgressBox)} let oldWidth = this.getElementWidth(ele) let newWidth = this.getContentWidth(ele, text) utils.log(`newWidth=${newWidth}, oldWidth=${oldWidth}`) 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) }, getDoi() { let selection = window.getSelection().toString() let sourceText = document.body.innerHTML let matches = selection.match(this.doiRegex) || sourceText.match(this.doiRegex) let doi = null if (matches) { doi = matches[0].replace('.pdf', '') } return doi }, getApi(doi) { this.log('初始化sciDownloadBox') this.initBox(doi) this.log('初始化sciDownloadBox完成') GM_xmlhttpRequest({ method: 'POST', url: this.api, responseType: 'json', headers: {"Content-Type": "application/x-www-form-urlencoded"}, data: `doi=${doi}&timeout=${this.timeout}`, onload: (res) => { console.log(res.response) utils.getPdf(res.response, doi) } }) }, getPdf(data, doi) { this.changeContent(this.svg.url, data.message) this.sciState.setAttribute('href', data.url) let exit = true switch (data.message) { case 'NotSupport': this.log('不支持该文章,退出...') break case 'Timeout': this.log('请求超时,退出...') break case 'DoiError': this.log('doi错误,退出...') break default: this.log('请求pdf中...') exit = false } if (exit) { return } let pdfURL = location.protocol + '//' + data.url.split('://')[1] utils.sciText.onclick = () => { setTimeout(() => { window.open(pdfURL) }, 1000) } this.sciContent.classList.add('loading') // 开始缓存同时尝试打开链接,可将下行反注释即可 // window.open(pdfURL) GM_xmlhttpRequest({ method: 'GET', url: pdfURL, responseType: 'blob', onload: function(res) { console.log(res.response) let fileURL = URL.createObjectURL(new Blob([res.response], {type: 'application/pdf'})) let title = doi.split('/').slice(1).join('/') utils.log('res.responseHeaders') console.log(res.responseHeaders) if (res.responseHeaders.search('application') !== -1) { let titleRes = res.responseHeaders.match(/filename=(.+)/) if (titleRes) { title = titleRes[1].replace(';', '').replace('.pdf', '').replace('"', '').replace('"', '') } } else { utils.sciContent.classList.remove('loading') return } utils.sciText.removeAttribute('href') utils.sciText.onclick = () => { setTimeout(() => { let win = window.open() win.document.write(``) win.document.title = title }, 1000) } utils.log('缓存pdf成功') utils.sciContent.classList.remove('loading') utils.sciContent.style['background'] = utils.color.success utils.changeContent(utils.svg.pdf, title) utils.sciState.onclick = (event) => { let aTag = document.createElement('a') aTag.setAttribute('href', fileURL) aTag.setAttribute('download', `${title}.pdf`) aTag.click() } } }) }, log(text) { console.log(`[sciDownload] ${text}`) } } try{ GM_addStyle(utils.style()) } catch { utils.log('添加style失败,退出...') return } let lastDoi = '' setInterval(function () { let doi = utils.getDoi() if (doi == lastDoi | doi == null) { return } lastDoi = doi utils.log('发现doi = ' + doi) utils.getApi(doi) }, 500) })();