// ==UserScript==
// @name sciDownload
// @namespace http://tampermonkey.net/
// @version 0.6
// @description 修复bug
// @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
// ==/UserScript==
(function() {
'use strict'
const utils = {
api: 'https://service-lephiw86-1256272652.hk.apigw.tencentcs.com/sciDownload',
doiRegex: new RegExp(/10\.\d{4,9}\/[-\._;\(\)\/:A-z0-9]+/),
timeout: 25,
height: 40,
autoMax: {b: true, time: 0.5},
// 如果缩小状态下,sciState内容改变是否自动最大化以及弹出后自动收回的时间
// 这个b会自动记忆,如果最后一次运行状态是min,下次自动min
get switchState() {
return GM_getValue('sciDownload-state')
},
set switchState(value) {
GM_setValue('sciDownload-state', value)
},
color:
{
success: '#e74c3c',
flash: '#00b894',
fail: '#2c3e50'
},
svg:
{
doi: `<svg class="progressBox" width="25" height="25">
<circle class="progress-background" cx="12.5" cy="12.5" r="10" fill="transparent" stroke="rgba(255,255,255,0.15)" stroke-width="2"/>
<circle class="progress" cx="12.5" cy="12.5" r="10" fill="transparent" stroke="white" stroke-width="2"/>
</svg>`,
url: `<svg width="25" height="25">
<circle cx="12.5" cy="12.5" r="10" fill="transparent" stroke="white" stroke-width="2"/>
</svg>`,
pdf: `<svg t="1629675437209" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2519" width="25" height="25"><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 t="1629798290236" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5684" width="40" height="40">
<path d="M715 547.2H310c-19.3 0-35-15.7-35-35s15.7-35 35-35h405c19.3 0 35 15.7 35 35s-15.7 35-35 35z" p-id="5685" fill="#ffffff"></path>
<path class="switch" d="M715 547.2H310c-19.3 0-35-15.7-35-35s15.7-35 35-35h405c19.3 0 35 15.7 35 35s-15.7 35-35 35z" p-id="5685" fill="#ffffff"></path>
</svg>`
},
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()
this.log('total=' + total)
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 {
display: flex;
align-items: center;
justify-content: center;
width: ${this.height}px;
height: ${this.height}px;
opacity: 1;
}
#sciContent #sciText {
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;
}
`
},
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.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 = `
<div id="sciSwitch">${this.svg.switch}</div>
<div id="sciContent">
<div id="sciState">${this.svg.doi}</div>
<a href="" id="sciText" target="_blank">${doi}</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')
this.sciDownloadBox.style.right = -this.getElementWidth(this.sciDownloadBox) + 'px'
setTimeout(() => {
this.sciDownloadBox.style.right = '0px'
this.startProgress()
}, 10)
this.sciSwitch = this.sciDownloadBox.querySelector('#sciSwitch')
this.sciSwitch.onclick = this.switchEvent
window.onresize = () => {
this.sciSwitch.click()
this.sciSwitch.click()
}
}
},
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 = `
<div id="sciProgress">
<div id="sciProgressText">${ele.innerHTML}</div>
</div>
`
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)
},
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) { clearInterval(id); return }
let pdfURL = location.protocol + '//' + data.url.split('://')[1]
this.sciText.setAttribute('href', pdfURL)
let cookie = (data.method == 'Safari') ? '' : data.method
this.log('cookie: ' + cookie)
this.sciContent.classList.add('loading')
GM_xmlhttpRequest({
method: 'GET',
url: pdfURL,
responseType: 'blob',
cookie: cookie,
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 = () => {
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
}
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)
})();