sciDownload

适配ipad端userscripts

目前為 2022-03-14 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name        sciDownload
// @namespace   http://tampermonkey.net/
// @version     2.2
// @description 适配ipad端userscripts
// @author      Polygon
// @icon        
// @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 1024 1024" 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.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() {
            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]

        },
        LocalSearch(doi) {
            // 1 开启动画效果
            this.initBox(doi)
            setTimeout(() => {
                this.startProgress()
            }, 230)
            // 2 生成data,这里按顺序查询,不太会用js的并发只能按照顺序了,应该不会很慢
            let data = undefined
            let totalSource = 4
            let failSource = 0
            // 2.1 查询unpaywall
            let setData = (value) => {
                if (value.url) {
                    if (data == undefined) {
                        data = value
                    }
                } else {
                    failSource += 1
                }
            }
            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 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) {
                        const resData = res.response['result']['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
                        })
                    }
                })
            }
            // 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 (data != undefined) {
                    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) {
            // 进入下载,收回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) {
                    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) / 1024 / 1024 * 5
                        rippleUp(size > 1 ? 1 : size)
                        lastTime = currentTime
                        lastDone = currentDone
                        let tip
                        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 = `${(res.done / 1024 / 1024).toFixed(2)}M | ${(res.total / 1024 / 1024).toFixed(2)}M | ${(percent * 100).toFixed(2)}% | ${data.message}`
                        } else {
                            utils.sciContent.setAttribute('download-noprogress', '')
                            tip = `${(res.done / 1024 / 1024).toFixed(2)}M | --M | --% | ${data.message}`
                        }
                        utils.sciContent.setAttribute('title', tip)
                        utils.sciSwitch.setAttribute('title', tip)
                    }
                },
                onload: function(res) {
                    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);
                        console.log(res.response)
                        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')
                        utils.sciText.onclick = () => {
                            setTimeout(() => {
                                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 = (event) => {
                            let aTag = document.createElement('a')
                            aTag.setAttribute('href', fileURL)
                            aTag.setAttribute('download', `${title}.pdf`)
                            aTag.click()
                        }
                    }, 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)
})();