Greasy Fork is available in English.

sciDownload

任何页面,sci论文下载,显示期刊信息

// ==UserScript==
// @name        sciDownload
// @namespace   http://tampermonkey.net/
// @version     4.1
// @description 任何页面,sci论文下载,显示期刊信息
// @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.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=polygon@unpaywall.com`
            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['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
                            })
                        } 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) / 1024 / 1024 * 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 / 1024 / 1024).toFixed(2)}M | ${(res.total / 1024 / 1024).toFixed(2)}M | ${(percent * 100).toFixed(2)}% | ${data.message}`
                        } else {
                            tip = `${journal_info} | ${(res.done / 1024 / 1024).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 / 1024 / 1024).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)
})();