返回首頁 

sciDownload

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


Install this script?
// ==UserScript==// @name        sciDownload// @namespace   http://tampermonkey.net/// @version     4.1// @description 任何页面,sci论文下载,显示期刊信息// @author      Polygon// @icon        ###JTPJF2zNyNlaxMBCUgG0BzHdwG8bwZJ4FJ+AEP/2IcGB4cCG4tdEwISkJSHBkfRXijR/SnLcyzj40nXvpBjA2I9hoAEJOXhwOXo9WDy/arVPrW7MGWTTcr4MtK1Y/08RJ2WgAQkJSmO4w8C/MmU5RPLeoafT2vu+A0r9WYQvu3sA/4x6doid70osxCQgKSkxbH6JoC3pSwfV0YXkNanbvpLz5A8Rto8za0fospKQAKSkhjHyr6n8dKU5VvKCD+hqplwYe8ZuIWk9c8y90UEmQlIQFIi41hxytLxZV8mbd41/i+53L8EXPqhkydjKRmzwkkrokwEJCApcbkHpPnDhhxHfwXo6Sm7sKWMcTYZc1ZmnQgyE5CApETmHBDi11K19v3JmuE4/iXAu6fswtiyT5E2pznoRJKRgAQkJTDngDAOI2NumDwg0QBAfSm7MKaMLiGtT8quE0VWAhKQlMQ8AnIMGXP55AFRtwKopOzCljLC16hq/j2zTgSZCUhAUiLzCMh7yJhJn9/i2DEgwNWkzVtSdl3KPAhIQFLC8whI0wtq94DQdaT1G1J2Xco8CEhAUsLjWD0MYH7K8rEfh1ZQ1SwN/BHrZqqaJZn7IoLMBCQgKZFxrO4A0JuyfGzZfaTNi5sE5BEAz8juSctJ6yOz60SRlYAEJCUxjuMrAD4qZflWZaTNBM6s1J4g/MLFD4wzyJhznbQiykRAApIS1+iUoh9JWT6ujCY8GsLlaCmYljv5lfB6us1c76QVUSYCEpCUuLgcHQWmK1KWjw/I8aT1Vu9weAVu9Mlgt76IKgsBCUhKWqxUL+wk004brSStDx4rZaXuBWHSa5MWTWwkbbI/nuLUbxFJQFIeA9zXNx+zeuw3WW7bcH0Xuv32h6yYlTochP9xM0KNtJl8OQVHQ5E1JyAByXB0ePzWB4jfTdXaFxsBiZV9EvdNGZoeW/p50uYER63IMhKQgGQAxkp9GIRzMkjGlt5A2hzG/f27oaf0gKMHQKUjqVp1u7h3brS4QglIhrHnOH4RwO6rQfUkL8Fw6QgQzsjQ7JZS5icxb/vt6aabnnDSiygzAQlIRmQcx3cCfEBG2Ug54Vww3glgByc9+Luka64fzdyaLLhKApLxAOBydDqYlmWUbSr/FYDdHLUAkmNJD1zmrhdlVgISkIzEOI73A3gwo8y/nLAe2zy5kFbf6f5Nmn8vCucgAXEYco7jAYAdXnRyaGyL5DzS5n1eDiLOTEACkhlZ4z6GXcnW8bEThwYb1y+l/ahavdtRLTJHAhIQB3BcLu8BTn4KYK6D3EUiL0i5UAugkYA4QuSy+ioY73CUZ5TR60jrGzOKpDwAAQmII0SO41cBvMpRnkHGVdK1xRkEUhqQgATEAybH6mYAB3lYpJDSu0jrL6colJIcCEhAPKCyUseC8DUPixZSegBDQ7IeSH6AWzpLQFoial7Ahx46B4/+3V6s57Mkm8yg6DE6YaQSEE+OrNRHQMhjGtCN6Bnexy6Z4NlFkXsQkIB4wLNSrlR2QX3YLuz5LE+rcfKtl0wI6y1uaQlIQNKSmqKOlVoGwukBrLZY1JN9aWDgJ0E9xSwzAQlIZmQTBVzp2x31Hvt81vYB7Oxt82+R1keH8RIXHwISEB96Y7QcR+cBtHkVKS9bxivImDVeHiIOQkACEgQjwFG0ECWyZ5E5nparSJtXe3qIPBABCUggkI0L9rK6FAzP98XpNNL6UwG7JVYeBCQgHvDGS1mp/UFY62XJeAhJEtHAwG+9fEQchIAEJAjGEROO408D/P4AlpOuaxjAVywyEpCAZATWrJzjeEcw3w3Cs4NYykKdQTD6mkhAfAmO6rmszgDjY4HsrM09mPP4Ylo1+LeAnmKVkYAEJCOwycp50aIFmDPbfoO1awC7MRb0adL6g2E9xS0LAQlIFlpNajmOPwjwJwNYTbRgvJqMacN7J7n0vutNJSCeQ8hRNG/0/scenlbN5LeRNtkX+sypM0WzlYB4jjjH0SkAne9p00JOHyatP55vG+I+GQEJiMdxMfo+iL32WOhhk0b6OHq4TGtqfvdY0rQkNVsRkIB4HBCs1AkgXOphkUHK15KuvTGDQEoDEJCAOEJkO9NuHA0CtK+jRXYZ4xwy5r+zC0XhSkAC4kiO4/g/AG7/ZAqMt5IxVzl2W2QZCUhAMgLbVO6xLLRji5tlfwTjEDLmHl8j0bcmIAFpzWhCBSt1DAjfcJAGktAtGBo6hAYHnwpkKDZNCEhAHA4NVqpqr0AcpAEldAlpfVJAQ7GahIAEJONhwXF8JMDu1wCEh8BYB2CrVW8zdmO0nE4grT/vphVVGgISkDSUxtT4z6ZI70SSrEMPXQP2fvL3HwAdQlrflnE3pDwlAQlISlC2jMvRv4Dpuxkk40v/TNo0pgfiOD4e4M95eG2S3jUakj8F8BKLcQQkIBkOCY7jGwF+TQbJ1qWM95AxF236S47Vl4DGmoW+21WkzVt9TUQ/kYAEJOVRwUq9DoTrU5ZPVvYX0mb+2H/g3t5nY/bs650XBR1rRriIquY9Hv0T6SQEJCApDwuO1fcAuD/qQfwhqtY+Mb45jqJDUaLrAMxK2ZXmZYQzqGrO9fYRg80EJCApDgYu9y8Bl36YorRZyd9Jm6aTyrFSHwbhHA//LVLGcWSM/egmWwACEpAUEFmp5SAsTVHapIQ/Sro25ZqGHMffAfhf3dsYo2S8nIxp/0q8QTrfWSYSkBbjwXG8GGCfWQ6HoM08ApKpmuJKZXfUh+1Z6gXehwjRw1TVz/T2EQNIQFoGJLocII95cukzpPUH0hxrXI6OANPVaWpb1jDuJ2Ne2LJOCqYkIAGZAg8vXtyHpD7gcQw9gW3/aQGtXPlYWg+OlX23PcxEDYTrqWpen7ZtqZtIQAIyVUB8V7IlfI6q5sSsBx7H6gYAr82qm7SecRYZc3YQrwKaSECaXVb7TyM6DNBOpHXmO9xcLj8HnNh7LvuEOSbpZNL64jBexXKRgDQPyBdAOM75cGB8kYx5t6ueF/f3ISnZM8lWNxdd/cA4ioy50lmfUcj7778N5s1bgHp9PkrJAiSlBaBkAbjxZwkJbQTZn2Rj47933FHTihX1jM3kXi4BmQQxl8v7ghO7rJr7Vk+e5zsBNUfRUpRouXsnxilzmM6U+/p2xqxZe4N5LxDvDaa9QdgLwHMz9nsYwCoQakDpTqpWV2bU51IuAZksILGyDxEe706cvkJah3jGCqzUe0H4rHtfwoSEy+U9wLzPJEGYF6xvWxnxjwGsQE99xXQuZCoBGX/8KLUPCH5rA/bUX0hrbr8/1IHDcXw+wKeE8kOLMwlXKs9Evd4LJIsAWgTgQAD/HKz9bEaPgfBVMJ3jcj2XramJ1RKQiQG5BITM3zyNsfk6afMO34EZq+dKZS7qw5cDeFMw39GQcKUyayQM6Ae4f+RP7BCsnXBGP8fIrC5tu46yXZeAjBlALpdfAk5+6jemtD9p7Xf9MtnHvpE77VcAsAdzoI1+D/AugczaZMOXka4d26bGJCBb/aaOo4sA8nnP+3LS5pi8Bo+VqqCEK8DYOa82usR3gLSJ2tFXOYOMUuY4fhHA93lBTzimWs14ebQQcxwfDbD9uFX07f9Im53yhiAB2RQQpS4Awf2FI8aVZMxReQ+Y9edydDqYlrWjrQ5v4w+kTa5nUwmIPeAqfXuj3vNzr4OBkldRdeAWL48mYo4ihVKpbKMBwP5sm0c7XenJOJuMOSuvvktAGhMoROcBdKo7ZFpOWh/prt+i5ErlaUieKgNUBjfm3rLfKsk2FQHGO8iYr+cBqfABYaX2BOEXXnATXkK12s0uHhxFOzXODsQxGArAS118Cq55EHPmvpRWrQq+nqMERCnf+x7fIW3enPYA5SjaC6VSDOZ4dHbG56fVSt1UBPKZRK/QAeFK3+6o9/zS68BjHErG/KCZR+P6ZrhnCYDFo4FozIslW3AChrQJPh1ssQPiPy/VNaTN4WOHmnt7t8fs2UuA5GCA7PSizwl+KIjh5AQoOZiqA0EfcixsQFipF4DwK79jjV5HWt/Y+JaJ6BAQDgGwv5/njFH/Doz7QbwBoPUgbECCDSDaAGb7WPt8EC0A8wJQY/I8/3sahNOpaoKuNlzcgMTxZQD/m/PhaCeRSzBrNBSF5QjgURDWgXkdmO4FYB/VuYeM+UtathzHO4L5OBDeB+BpaXUT6hjfJmOOcNZPIizkwHLlwOehPut/Q4IskNeTYCwD0Tr0DK8L+tSyUr0gnOfx1fYvSZs9Q45FMQMSK/tE6FtCgiyg1+8B+njo5RcaZxMkPwfo6Q5Mh0mbbRx0TSWFCwhH0XNRot+GhFhwr7sw8npxsNkcWamzQJhyor0###fPnh9qTIoXkJAzGIYahZnhMwjGl3yDwkq9C4QvuiGhB0jrPdy0Tb4YC2nW6V7c378rekq/6fR+Buyf/ahiL5rt+ymDANvf8v4zN07dQbs09q2g5Faq1r6fZl9GnoUrHQ6QnXrV/VtAwlqqmgPStJm2plBnEC6r68A4LC2cLqv7HWAnPOAaEtyBnXa6e/wsIXzQQU/HE098E+B2TSb3GBh24r0/Nb7eRTIElOzbijsA/OzRNxft/28XiPWE+1K+voUJCMfxqwBe5Qusg/T2s//tjVAw16hWezBt31ipZSCcnra+e+r4VNK1C0L2d0YHhKNoHkr4TxCd2N1v4fHDAK0F+E5Qj8HcubUs05lOdsAEXAIu5PHo51VP9qWBAb8JN8b1YEYGpHG2ILah2OoxED/6bVXbd1PWgu0PD1CttjaP1jmKXo1S6ZMA75uHf1s9c5qse8YEZPPZAmRf6A/6TUbOAz0E8NrGGcIGolS6g6rVtt3EHJniZ/hTANo2EUIuPIk/QNXaZ0J7d31ARq4t7LM8HPQRg9Cgx/htAFiDsBYJrcVTT91JP/rRxhzbS2XNcXwSwDYoc1MJOqtoDWnzijy61JUBGTlblOwyyjYU++UBJqgnYx2A1SjxLZi73Wrf64egfRtjxoujGEnjI1dfXm3k49tYK95nibym3eqqgIyeLY4E2L7e6v5QWz6jNN51NUC3gHk1GfOj9jTp30rjld/6U2cClGrRH/8WPR2K/k5642xB9DaQPVuQnbCgU7c/A7gZ4FtAPavbeR2RBxAu95fBdBpA7uvC59GxsZ45h8M21bFnEI6iA1EqHTH6MWrHvFk7+j8I8EowrcTcuSvzeCfasV/BZBxHdgkHG5TnBTMNYdSGcHRkQFipgwCcDOrQO972JaASRkIxNLSSBgefCjHenezBld5dUJ9lQ2JfbJozrX0deffkUtK1L7SjHx1zBuE43g9ITgLo7e3Y8Yxt2LvWK9HDK+m22q0ZtTOmfHTCiaMAthPk5f1M13huDwJ0CYaGLqXBwaF2QZ32gIy+vGTnw7U/QZ/l94R4HxjfAXBTN11ke+5zKjkvWbIdnhg6CnUcPToRRSqdY9FdAC3H7NnfoNWr1zt6OMumNSCs1LEoYRkY9sG1ztlKXKbbarpzOtS5PWGl7Hv4i1FCBdxYRyTQRsvBvJyM+V4gQyebaQuIx0sxTjuaXsQXk66dnL5eKjcRaLwNSPRKcGJv3trH1u0TDa2nSR25rruz8Vg+0zrU6/fSwMCGTiDb9oDwoYfOwaOPXgewnSuq87Z6sjsNDHjOdtJ5uzVdPeJX9u2MJ3v2ANHujcfbKVkPLv0RSbIePT3r8fDD6+m++56crv61aretAeHFiw9AUre/KTpzI3yJqsZ9ZdvO3CvplQeBtgWksfgLocO/AaKFpPXPPHiKdIYRaEtAuiIccvaYYYd2mN3JPSCj4bgh4GuVYfZ8vAvV96Pq7XfnYy6u3Uog14Bwpf/FqJeuA9DZM5jL2aNbj9/c+51vQMrqq2AEXRJ5AhHCCjBSLz8wKVHGy8mYwdxpSwNdRyC3gHC5XAYnt+VDhP8G0JWkzQlcVreA4fGyjNz3yGeMZoZrfgFRajkISwNjuhfEV1K19gnry0rZhxov9GjjEVDpQKpW/dYI8eiASDubQC4B4XL0GjDdGHDXJ8zaNzqFaBXArs7tEM6man4LQDr3S4QdQyCngAS79mg6nSXH8fkAn+JB8leYPaeXVq9+2MNDpDOcQD4BiZV9VMPvcWjGcc3meR15d5rs2cN9I5xCVePz8cy9bVF2DYHgARl5r4P9vhFiLCVjVjSjyHF8rd/0mfRjaH0gAUnXjJR0dFoIhA9IOXo/mD7tvDetwlFWp4HRuEh33hjHkDGXO+tFWBgC4QMSK/u8VcWJ4BQfq6xfmEdWeCXpml1cUzYh0JJAHgGxH69c5qr6A2mzc9OPVZXKXNSHH2+5R60KGIeRMfbRF9mEQEsCeQTkAQC7tWx5fEGLWSo4Vnadi5dk9t1acDVpI0uveUIskjyPgNg3wZ7lAHHS6SO5XD4CnHwlyERxVIqoWrXrVcgmBFIRyCMg/3CfGob/i3RtWeN6I45fNDrLiZ2XyX8jfI6q5kR/I3EoEoE8AvILAO5L8RIeAjcmtPNfWH7LSG5Ez6xeWrPm/iINruyrP4EcAhKfB/Cp/l0L6EA4l6rmjICOYlUQAuEDotRhINh3QDpjY9yMefMOo5tueqIzOiS96CYC4QPS1zcfs3oe6pB1Jh4B4wAy5tfdNCjS184hEDwgIxfY0TUAvWHadzPhmGo1M+39kA50LYF8AqJULwj2YcLZ00aG+Giq1r41be1LwzOCQC4BaZxFlDoLhI9MDyX6Fml99PS0La3OJAL5BWThwtmY/wx7FultKzBuTDbduYu+tBWGNOZLILeAjFyLxG8A+BrfTmbQX07aHJOhXkqFwJQEcg3I6EetN4Pw7fzHgT5GWp+ZfzvSQpEI5B6QNoRkAwgfoqr5WpEGTva1PQTaEpDcQsK4EKXShd2+YGZ7hlpacSHQtoCMXpMsAvGJYPheJ1wFxoWy8pPLkIsmC4G2BmRTxxrrnY8E5fAMnf01wNfaxTPJmB9k0EmpEHAmMC0B2RyUSt/eGO5ZBGARCAc0/gQ2jv7Y2RM3AslalLBClkRzHmMRehCY1oB49FukQqAtBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrAQlIt46c9LstBCQgbcEsjXQrgf8HoYS0IywK1C0AAAAASUVORK5CYII=// @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 = infolet width = getComputedStyle(this.box).widththis.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_setValueget switchState() {if (typeof(GM_getValue) == 'undefined') {return this._switchSate} else {return GM_getValue('sciDownload-state', 'max')}},set switchState(value) {if (typeof(GM_setValue) == 'undefined') {this._switchSate = value} else {GM_setValue('sciDownload-state', value)}},_switchSate: 'max',color:{success: '#e74c3c',flash: '#00b894',fail: '#2c3e50'},svg:{doi: `<svg class="progressBox" width="40" height="40"><g fill="transparent" stroke-width="2.5"><circle class="progress-background" cx="20" cy="20" r="11" stroke="rgba(255,255,255,0.23)"/><circle class="progress" cx="20" cy="20" r="11" stroke="rgba(255,255,255,1)" stroke-linecap="round"/></g></svg>`,pdf: `<svg viewBox="0 0 #### ####" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2519" width="30" height="30"><path d="M478.08 192.192a58.88 58.88 0 0 1 46.208 23.168c23.104 32.064 21.312 99.84-8.832 199.68A536.832 536.832 0 0 0 625.664 557.44c37.312-7.04 74.688-12.416 112-12.416 83.52 1.792 96 41.024 94.144 64.128 0 60.544-58.688 60.544-88.896 60.544v0.064a226.048 226.048 0 0 1-131.52-53.504c-72.832 16-142.144 39.168-211.456 67.712C344.96 782.08 293.312 832 248.96 832c-8.96 0-19.584-1.792-26.752-7.168A52.48 52.48 0 0 1 192 776.704c0-16 3.52-60.608 172.352-133.568a1267.776 1267.776 0 0 0 94.208-221.056c-21.312-42.752-67.584-147.84-35.584-201.344 10.688-19.648 32-30.272 55.168-28.544z m-118.4 475.84c-42.688 20.736-95.232 58.368-90.112 82.368 3.392 16 21.888 10.56 55.424-16.384 21.056-27.072 24.512-41.088 34.688-65.984z m386.304-79.808c-17.152 0-32.384 0-49.536 7.68 19.072 19.2 29.504 28.608 48.64 32.448 13.312 3.84 39.104 10.752 46.72-9.28 7.68-19.968-7.616-30.848-45.824-30.848zM494.08 492.8a2572.16 2572.16 0 0 1-46.592 104.704l104.704-34.88c-21.76-22.272-41.344-43.392-58.112-69.824z m-16.64-223.488c-10.496 1.664-16.192 15.04-18.88 24.512-5.504 27.008 8.96 57.088 27.072 78.528 14.912-22.528 18.688-43.328 12.928-73.92-6.976-20.48-14.016-30.208-21.12-29.12z" fill="#ffffff" p-id="2520"></path></svg>`,switch: `<svg width="40" height="40"><g stroke="white" stroke-width="3" stroke-linecap="round"><line x1="10" y1="20" x2="30" y2="20"/><line class="switch" x1="10" y1="20" x2="30" y2="20"/></g></svg>`},style() {let div = document.createElement('div')div.innerHTML = this.svg.doidiv.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).leftthis.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 parentfor (let i=0;i<event.path.length;i++) {if (event.path[i].id.match(/(sciText|sciState)/)) {parent = event.path[i]break}}let x = event.offsetXlet y = event.offsetYlet 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).leftutils.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) returnthis.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.innerHTMLele.innerHTML = contentlet width = this.getElementWidth(ele)ele.innerHTML = oldContentreturn 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.sciTextlet oldWidth = this.getElementWidth(ele)let newWidth = this.getContentWidth(ele, text)ele.style.width = oldWidth + 'px'setTimeout(() => {ele.style.width = newWidth + 'px'ele.innerHTML = textsetTimeout(() => {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, reslet selection = window.getSelection().toString()let sourceText = document.body.innerHTMLres = 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 valueif (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 = 4let failSource = 0let setData = (value) => {if (value.url) {if (!Object.keys(data).includes("url")) {data = {...data, ...value}}} else {failSource += 1}}// 2.1 查询unpaywallconst unpaywall = `https://api.unpaywall.org/v2/${doi}[email protected]`GM_xmlhttpRequest({method: 'GET',url: unpaywall,responseType: 'json',onload: function (res) {const unpaywallData = res.responselet journal_name = unpaywallData['journal_name']setTimeout(() => {utils.check_update_journal_info(journal_name)data.journal_name = journal_name}, 0);let urltry {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 查询scihubconst 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 查询researchgatelet isConsistent = (s1, s2) => {if (s1 == null || s2 == null) return falselet matchArray = [], strArray = [s1, s2]for (let i=0;i<strArray.length;i++) {matchArray.push(strArray[i].toLowerCase().match(/\w+/g))}[s1, s2] = matchArrayif (s1.length != s2.length) return falselet b = truefor (let i=0;i<s1.length;i++) {if (s1[i] != s2[i]) return false}return b}let researchgate = (paperTitle) => {const url = `https://www.researchgate.net/search.SearchBox.html?query=${paperTitle}&activeTab=publication`GM_xmlhttpRequest({method: 'GET',url: url,headers: {'accept': 'application/json'},responseType: 'json',onload: function (res) {try {const resData = res.response['r###lt']['state']['searchSearch']['publication']['items']console.log(resData)let url = ''let i = 0while (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 = pdfURLconsole.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/S016980952100421Xif (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 urllet 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 = 0const interval = 10let 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 = trueswitch (data.message) {case 'NotSupport':this.log('不支持该文章,退出...')breakcase 'Timeout':this.log('请求超时,退出...')breakdefault:this.log('请求pdf中...')exit = false}setTimeout(() => {progress.style['stroke-dashoffset'] =  '0'}, 230*2)if (exit) {return}utils.sciContent.style['background-color'] = utils.color.flashlet openInTab = (url) => {if (typeof(GM_openInTab) == 'undefined') {var a = document.createElement('a')a.href = urla.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, sizesetTimeout(() => {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) returnutils.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 = -1ripple.style.opacity = opacityripple.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.doneif (!lastTime | currentTime - lastTime >= 300) {size = (currentDone - lastDone) / #### / #### * 5rippleUp(size > 1 ? 1 : size)lastTime = currentTimelastDone = currentDonelet tiplet 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.totalprogress.style['stroke-dashoffset'] =  utils.progressTotaLength * (1 - percent)// 下载进度条GM_addStyle(`#sciContent[download-progress] #sciText::before {content: "";position: absolute;right: 0;bottom: 0;height: 40px;width: ${percent * 100}%;background-color: white;opacity: 0.5;z-index: -1;transition: width .23s linear;}`)tip = `${journal_info} | ${(res.done / #### / ####).toFixed(2)}M | ${(res.total / #### / ####).toFixed(2)}M | ${(percent * 100).toFixed(2)}% | ${data.message}`} else {tip = `${journal_info} | ${(res.done / #### / ####).toFixed(2)}M | --M | --% | ${data.message}`utils.sciContent.setAttribute('download-noprogress', '')}utils.sciContent.setAttribute('title', tip)utils.sciSwitch.setAttribute('title', tip)}},onload: function(res) {if (document.querySelectorAll(`#sciDownloadBox[doi="${doi}"]`).length == 0) returnlet size = (res.response.size / #### / ####).toFixed(2)let key = `sciDownload-journal-${data.journal_name}`let journal_info = GM_getValue(key, "...")let tip = `${journal_info} | ${size}M | ${data.message}`utils.sciContent.setAttribute('title', tip)utils.sciSwitch.setAttribute('title', tip)setTimeout(() => {utils.sciContent.removeAttribute('loading')if (!res.responseHeaders.match(utils.pdfRegex)) {failSetting()notification.open('pdf加载失败了,亲自点一下吧~', 3)return}utils.sciContent.removeAttribute('download-progress')utils.sciContent.removeAttribute('download-noprogress')setTimeout(() => {utils.sciContent.style['background-color'] = utils.color.success}, 230);let fileURL = URL.createObjectURL(new Blob([res.response], {type: 'application/pdf'}))let title = doi.split('/').slice(1).join('/')let titleRes = res.responseHeaders.match(/filename=(.+)/)if (titleRes) {title = decodeURI(titleRes[1].split(';')[0]).replace('.pdf', '').replace('"', '').replace('"', '')}utils.sciText.removeAttribute('href')let downloadPdf = (fileURL, title) => {let aTag = document.createElement('a')aTag.setAttribute('href', fileURL)aTag.setAttribute('download', `${title}.pdf`)aTag.click()}utils.sciText.onclick = () => {setTimeout(() => {if ((typeof(GM_setValue) == 'undefined')) {downloadPdf(fileURL, title)} else {let win = window.open()win.document.write(`<iframe name="${title}" src="${fileURL}" frameborder="0" style="border:0; top:0px; left:0px; bottom:0px; right:0px; width:100%; height:100%;" allowfullscreen></iframe>`)win.document.title = title}}, 1E3)}utils.log('缓存pdf成功')utils.sciContent.style['background'] = utils.color.successutils.changeContent(utils.svg.pdf, title)utils.sciState.onclick = () => {downloadPdf(fileURL, title)}// 防止此时还没变更let key = `sciDownload-journal-${data.journal_name}`let valueif (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 = nulllet lastIsSelect = falsesetInterval(function () {let [doi, select] = utils.getDoi()if (!doi | doi == lastDoi | (lastIsSelect && !select)) {return}lastDoi = doilastIsSelect = selectlet 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)})();