// ==UserScript== // @name 网盘智能识别助手 // @namespace https://github.com/syhyz1990/panAI // @version 1.8.1 // @author YouXiaoHou // @description 智能识别选中文字中的🔗网盘链接和🔑提取码,识别成功打开网盘链接并自动填写提取码,省去手动复制提取码在输入的烦恼。支持识别 ✅百度网盘 ✅阿里云盘 ✅腾讯微云 ✅蓝奏云 ✅天翼云盘 ✅和彩云 ✅迅雷云盘 ✅123云盘 ✅360云盘 ✅115网盘 ✅奶牛快传 ✅城通网盘 ✅夸克网盘 ✅Chrome 扩展商店 ✅Edge 扩展商店 ✅Firefox 扩展商店 ✅Windows 应用商店。 // @license AGPL // @homepage https://www.youxiaohou.com/tool/install-panai.html // @supportURL https://github.com/syhyz1990/panAI // @match *://*/* // @require https://unpkg.com/sweetalert2@10.16.6/dist/sweetalert2.min.js // @require https://unpkg.com/hotkeys-js/dist/hotkeys.min.js // @resource swalStyle https://unpkg.com/sweetalert2@10.16.6/dist/sweetalert2.min.css // @run-at document-idle // @grant GM_openInTab // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant GM_getResourceText // @icon  // @downloadURL none // ==/UserScript== (function () { 'use strict'; const customClass = { container: 'panai-container', popup: 'panai-popup', }; let toast = Swal.mixin({ toast: true, position: 'top', showConfirmButton: false, timer: 3500, timerProgressBar: false, didOpen: (toast) => { toast.addEventListener('mouseenter', Swal.stopTimer); toast.addEventListener('mouseleave', Swal.resumeTimer); } }); let util = { clog(c) { console.group('[网盘智能识别助手]'); console.log(c); console.groupEnd(); }, parseQuery(name) { let reg = new RegExp("(^|&)" + name + "=([^&]*)(&|$)", "i"); let r = location.search.substr(1).match(reg); if (r != null) return (r[2]); return null; }, getValue(name) { return GM_getValue(name); }, setValue(name, value) { GM_setValue(name, value); }, sleep(time) { return new Promise((resolve) => setTimeout(resolve, time)); }, addStyle(id, tag, css) { tag = tag || 'style'; let doc = document, styleDom = doc.getElementById(id); if (styleDom) return; let style = doc.createElement(tag); style.rel = 'stylesheet'; style.id = id; tag === 'style' ? style.innerHTML = css : style.href = css; document.head.appendChild(style); }, isHidden(el) { try { return el.offsetParent === null; } catch (e) { return false; } }, query(selector) { if (Array.isArray(selector)) { let obj = null; for (let i = 0; i < selector.length; i++) { let o = document.querySelector(selector[i]); if (o) { obj = o; break; } } return obj; } return document.querySelector(selector); } }; let opt = { 'baidu': { reg: /((?:https?:\/\/)?(?:e?yun|pan)\.baidu\.com\/(doc\/|enterprise\/)?(?:s\/[\w~]*(((-)?\w*)*)?|share\/\S{4,}))/, host: /(pan|e?yun)\.baidu\.com/, input: ['#accessCode', '.share-access-code', '#wpdoc-share-page > .u-dialog__wrapper .u-input__inner'], button: ['#submitBtn', '.share-access .g-button', '#wpdoc-share-page > .u-dialog__wrapper .u-btn--primary'], name: '百度网盘', storage: 'hash' }, 'aliyun': { reg: /((?:https?:\/\/)?(?:(?:www\.)?aliyundrive\.com\/s|alywp\.net)\/[a-zA-Z0-9]+)/, host: /www\.aliyundrive\.com|alywp\.net/, input: ['.ant-input', 'input[type="text"]'], button: ['.button--fep7l', 'button[type="submit"]'], name: '阿里云盘', storage: 'hash' }, 'weiyun': { reg: /((?:https?:\/\/)?share\.weiyun\.com\/[a-zA-Z0-9]+)/, host: /share\.weiyun\.com/, input: ['.mod-card-s input[type=password]'], button: ['.mod-card-s .btn-main'], name: '微云', storage: 'hash' }, 'lanzou': { reg: /((?:https?:\/\/)?(?:[a-zA-Z0-9\-.]+)?lanzou[a-z]\.com\/[a-zA-Z0-9_\-]+)/, host: /(?:[a-zA-Z0-9-.]+)?lanzou[a-z]\.com/, input: ['#pwd'], button: ['.passwddiv-btn', '#sub'], name: '蓝奏云', storage: 'hash' }, 'tianyi': { reg: /((?:https?:\/\/)?cloud\.189\.cn\/(?:t\/|web\/share\?code=)?[a-zA-Z0-9]+)/, host: /cloud\.189\.cn/, input: ['.access-code-item #code_txt'], button: ['.access-code-item .visit'], name: '天翼云', storage: 'hash' }, 'caiyun': { reg: /((?:https?:\/\/)?caiyun\.139\.com\/m\/i\?[a-zA-Z0-9]+)/, host: /caiyun\.139\.com/, input: ['.token-form input[type=text]'], button: ['.token-form .btn-token'], name: '和彩云', storage: 'local', storagePwdName: 'tmp_caiyun_pwd' }, 'xunlei': { reg: /((?:https?:\/\/)?pan\.xunlei\.com\/s\/[\w-]{10,})/, host: /pan\.xunlei\.com/, input: ['.pass-input-wrap .td-input__inner'], button: ['.pass-input-wrap .td-button'], name: '迅雷云盘', storage: 'hash' }, '123pan': { reg: /((?:https?:\/\/)?www\.123pan\.com\/s\/[\w-]{6,})/, host: /www\.123pan\.com/, input: ['.ca-fot input'], button: ['.ca-fot button'], name: '123云盘', storage: 'hash' }, '360': { reg: /((?:https?:\/\/)?(?:[a-zA-Z0-9\-.]+)?yunpan\.360\.cn(\/lk)?\/surl_[\w]{6,})/, host: /yunpan\.360\.cn/, input: ['.pwd-input'], button: ['.submit-btn'], name: '360云盘', storage: 'hash' }, '115': { reg: /((?:https?:\/\/)?115\.com\/s\/[a-zA-Z0-9]+)/, host: /115\.com/, input: ['.form-decode input'], button: ['.form-decode .submit a'], name: '115网盘', storage: 'hash' }, 'cowtransfer': { reg: /((?:https?:\/\/)?(?:[a-zA-Z0-9-.]+)?cowtransfer\.com\/s\/[a-zA-Z0-9]+)/, host: /(?:[a-zA-Z0-9-.]+)?cowtransfer\.com/, input: ['.receive-code-input input'], button: ['.open-button'], name: '奶牛快传', storage: 'hash' }, 'ctfile': { reg: /((?:https?:\/\/)?(?:[a-zA-Z0-9-.]+)?ctfile\.com\/\w+\/[a-zA-Z0-9-]+)/, host: /(?:[a-zA-Z0-9-.]+)?ctfile\.com/, input: ['#passcode'], button: ['.card-body button'], name: '城通网盘', storage: 'hash' }, 'quark': { reg: /((?:https?:\/\/)?pan\.quark\.cn\/s\/[a-zA-Z0-9-]+)/, host: /pan\.quark\.cn/, input: ['.ant-input'], button: ['.ant-btn-primary'], name: '夸克网盘', storage: 'local', storagePwdName: 'tmp_quark_pwd' }, 'chrome': { reg: /^https?:\/\/chrome.google.com\/webstore\/.+?\/([a-z]{32})(?=[\/#?]|$)/, host: /chrome\.google\.com/, replaceHost: "chrome.crxsoso.com", name: 'Chrome商店', }, 'edge': { reg: /^https?:\/\/microsoftedge.microsoft.com\/addons\/.+?\/([a-z]{32})(?=[\/#?]|$)/, host: /microsoftedge\.microsoft\.com/, replaceHost: "microsoftedge.crxsoso.com", name: 'Edge商店', }, 'firefox': { reg: /^https?:\/\/(reviewers\.)?(addons\.mozilla\.org|addons(?:-dev)?\.allizom\.org)\/.*?(?:addon|review)\/([^/<>"'?#]+)/, host: /addons\.mozilla\.org/, replaceHost: "addons.crxsoso.com", name: 'Firefox商店', }, 'microsoft': { reg: /^https?:\/\/(?:apps|www).microsoft.com\/store\/.+?\/([a-zA-Z\d]{10,})(?=[\/#?]|$)/, host: /(apps|www)\.microsoft\.com/, replaceHost: "apps.crxsoso.com", name: 'Windows商店', }, }; let main = { lastText: "lorem&", //初始化配置数据 initValue() { let value = [{ name: 'setting_success_times', value: 0 }, { name: 'setting_auto_click_btn', value: true }, { name: 'setting_active_in_front', value: true }, { name: 'setting_timer_open', value: false }, { name: 'setting_timer', value: 5000 }]; value.forEach((v) => { if (util.getValue(v.name) === undefined) { util.setValue(v.name, v.value); } }); }, // 监听选择事件 addPageListener() { document.addEventListener("mouseup", this.smartIdentify.bind(this), true); }, smartIdentify(event, str = '') { let selection = window.getSelection(); let text = str || selection.toString(); if (text !== this.lastText && text !== '') { //选择相同文字或空不识别 let start = performance.now(); this.lastText = text; //util.clog(`当前选中文字:${text}`); let linkObj = this.parseLink(text); let link = linkObj.link; let name = linkObj.name; let pwd = this.parsePwd(text); if (!link) { linkObj = this.parseParentLink(selection); link = linkObj.link; name = linkObj.name; } if (link) { if (!/https?:\/\//.test(link)) { link = 'https://' + link; } let end = performance.now(); let time = (end - start).toFixed(3); util.clog(`文本识别结果:${name} 链接:${link} 密码:${pwd} 耗时:${time}毫秒`); let option = { toast: true, showCancelButton: true, position: 'top', title: `发现${name}链接`, html: `${!!pwd ? '密码:' + pwd : '是否打开?'}`, confirmButtonText: '打开', cancelButtonText: '关闭', customClass }; if (util.getValue('setting_timer_open')) { option.timer = util.getValue('setting_timer'); option.timerProgressBar = true; } util.setValue('setting_success_times', util.getValue('setting_success_times') + 1); Swal.fire(option).then((res) => { this.lastText = 'lorem&'; selection.empty(); if (res.isConfirmed || res.dismiss === 'timer') { if (name === '和彩云') { //和彩云无法携带参数和Hash util.setValue('tmp_caiyun_pwd', pwd); } if (name === '夸克网盘') { util.setValue('tmp_quark_pwd', pwd); } let active = util.getValue('setting_active_in_front'); if (pwd) { let extra = `${link}?pwd=${pwd}#${pwd}`; if (~link.indexOf('?')) { extra = `${link}&pwd=${pwd}#${pwd}`; } GM_openInTab(extra, {active}); } else { GM_openInTab(`${link}`, {active}); } } }); } } }, addHotKey() { hotkeys('f1', (event, handler) => { event.preventDefault(); this.showIdentifyBox(); }); }, //正则解析网盘链接 parseLink(text = '') { let obj = {name: '', link: ''}; if (text) { text = text.replace(/[点點]/g, '.'); text = text.replace(/[\u4e00-\u9fa5\u200B()(),,]/g, ''); text = text.replace(/lanzous/g, 'lanzouw'); //修正lanzous打不开的问题 for (let name in opt) { let val = opt[name]; if (val.reg.test(text)) { let matches = text.match(val.reg); obj.name = val.name; obj.link = matches[0]; if (val.replaceHost) { obj.link = obj.link.replace(val.host, val.replaceHost); } return obj; } } } return obj; }, //正则解析超链接类型网盘链接 parseParentLink(selection) { let anchorNode = selection.anchorNode.parentElement.href; let focusNode = selection.focusNode.parentElement.href; if (anchorNode) return this.parseLink(anchorNode); if (focusNode) return this.parseLink(focusNode); return this.parseLink(); }, //正则解析提取码 parsePwd(text) { text = text.replace(/\u200B/g, ''); let reg = /(?<=\s*(密|提取|访问|訪問|key|password|pwd|#)[码碼]?[::=]?\s*)[a-zA-Z0-9]{3,8}/i; if (reg.test(text)) { let match = text.match(reg); return match[0]; } return ''; }, //根据域名检测网盘类型 panDetect() { let hostname = location.hostname; for (let name in opt) { let val = opt[name]; if (val.host.test(hostname)) { return name; } } return ''; }, //自动填写密码 autoFillPassword() { let url = location.href; let query = util.parseQuery('pwd'); let hash = location.hash.slice(1); let pwd = query || hash; let panType = this.panDetect(); for (let name in opt) { let val = opt[name]; if (panType === name) { if (val.storage === 'local') { pwd = util.getValue(val.storagePwdName) ? util.getValue(val.storagePwdName) : ''; pwd && this.doFillAction(val.input, val.button, pwd); } if (val.storage === 'hash') { if (!/^[a-zA-Z0-9]{3,8}$/.test(pwd)) { //过滤掉不正常的Hash return; } pwd && this.doFillAction(val.input, val.button, pwd); } } } }, doFillAction(inputSelector, buttonSelector, pwd) { let maxTime = 10; let ins = setInterval(async () => { maxTime--; let input = util.query(inputSelector); let button = util.query(buttonSelector); if (input && !util.isHidden(input)) { clearInterval(ins); Swal.fire({ toast: true, position: 'top', showCancelButton: false, showConfirmButton: false, title: 'AI已识别到密码!正自动帮您填写', icon: 'success', timer: 2000, customClass }); let lastValue = input.value; input.value = pwd; //Vue & React 触发 input 事件 let event = new Event('input', {bubbles: true}); let tracker = input._valueTracker; if (tracker) { tracker.setValue(lastValue); } input.dispatchEvent(event); if (util.getValue('setting_auto_click_btn')) { await util.sleep(1000); //1秒后点击按钮 button.click(); } } else { maxTime === 0 && clearInterval(ins); } }, 800); }, //重置识别次数 clearIdentifyTimes() { let res = Swal.fire({ showCancelButton: true, title: '确定要重置识别次数吗?', icon: 'warning', confirmButtonText: '确定', cancelButtonText: '取消', customClass }).then(res => { this.lastText = 'lorem&'; if (res.isConfirmed) { util.setValue('setting_success_times', 0); history.go(0); } }); }, //识别输入框中的内容 showIdentifyBox() { Swal.fire({ title: '识别剪切板中文字', input: 'textarea', inputPlaceholder: '若选方式一,请按 Ctrl+V 粘贴要识别的文字', html: `