您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
基于大佬(ywzhaiqi)的AutoClick(火狐版)改造而来
// ==UserScript== // @name AutoClick // @namespace https://github.com/AutoClick // @version 0.1.11 // @description 基于大佬(ywzhaiqi)的AutoClick(火狐版)改造而来 // @author AutoClick // @license MIT // @match https://*/* // @match http://*/* // @match file:///* // @grant GM_setClipboard // @grant GM_setValue // @grant GM_getValue // @grant GM_addValueChangeListener // @grant GM_addStyle // @grant GM_registerMenuCommand // ==/UserScript== (function() { // 'use strict'; // 请不要在这里更改,更改在图标右键打开的设置里。 var BUTTON_TYPE = 1, // 0:地址栏按钮,1:可移动按钮,2:菜单 ENABLE = true, hovering_time = 500, // 悬停的时间(毫秒) clicked_elem = 'a, input, button, li', clicked_parent_elem = 'b, i, img, span, small, strong, em, big, sub, sup, s, font', ON_IMG = '', OFF_IMG = '' ; try { class AutoClick { constructor() { this.clicked_elems = []; this.clicked_parent_elems = []; } get enable() { return ENABLE; } set enable(bool) { bool = !!bool; console.log('setEnabel', bool, ENABLE) if (!this.isFirstRun && ENABLE === bool) return; this.prefs.setPrefVal('enable', ENABLE = bool); this.updateState(bool); } get BUTTON_TYPE() { return BUTTON_TYPE; } set BUTTON_TYPE(num) { if (!this.isFirstRun && BUTTON_TYPE === num) return; BUTTON_TYPE = num; this.prefs.setPrefVal('BUTTON_TYPE', BUTTON_TYPE); this.rebuildButton(); this.updateState(ENABLE); } get hovering_time() { return hovering_time; } set hovering_time(num) { if (hovering_time == num) return; hovering_time = num; this.prefs.setPrefVal('hovering_time', hovering_time); } get default_prefs() { return { clicked_elem: clicked_elem, clicked_parent_elem: clicked_parent_elem }; } get clicked_elem () { return this.clicked_elems; } set clicked_elem(str) { this.clicked_elems = str.split(',').map((x) => x.trim()); } set clicked_parent_elem(str) { this.clicked_parent_elems = str.split(',').map((x) => x.trim()); } init () { this. prefs = { hasPrefVal: (key) => { return GM_getValue(key) !== undefined; }, getPrefVal: (key, def) => { return GM_getValue(key, def); }, setPrefVal: (key, val) => { return GM_setValue(key, val); }, addObserver: (a, context, bool) => { GM_addValueChangeListener(name, function(name, old_value, new_value, remote) { }); } }; this.isFirstRun = true; this.loadSetting(); window.document.addEventListener('mouseover', this.handleEvent.bind(this), false); window.document.addEventListener('mouseout', this.handleEvent.bind(this), false); window.document.addEventListener('keydown', this.handleEvent.bind(this), false); window.document.addEventListener('keyup', this.handleEvent.bind(this), false); // this.prefs.addObserver('', this, false); GM_registerMenuCommand('配置管理', () => { this.openPref() }); this.isFirstRun = false; } uninit() { this.removeButton(); gBrowser.mPanelContainer.removeEventListener('mouseover', this, false); gBrowser.mPanelContainer.removeEventListener('mouseout', this, false); gBrowser.mPanelContainer.removeEventListener('keydown', this, false); gBrowser.mPanelContainer.removeEventListener('keyup', this, false); this.prefs.removeObserver('', this, false); } loadSetting(type) { // 载入设置并执行相应的任务 if (!type || type === 'BUTTON_TYPE') { this.BUTTON_TYPE = this.prefs.hasPrefVal('BUTTON_TYPE') ? this.prefs.getPrefVal('BUTTON_TYPE') : this.BUTTON_TYPE; } if (!type || type === 'enable') { this.enable = this.prefs.hasPrefVal('enable') ? this.prefs.getPrefVal('enable') : this.enable; } if (!type || type === 'hovering_time') { this.hovering_time = this.prefs.hasPrefVal('hovering_time') ? this.prefs.getPrefVal('hovering_time') : this.hovering_time; } if (!type || type === 'clicked_elem') { if (this.prefs.hasPrefVal('clicked_elem')) { this.clicked_elem = this.prefs.getPrefVal('clicked_elem'); } else { this.prefs.setPrefVal('clicked_elem', this.clicked_elem = clicked_elem); } } if (!type || type === 'clicked_parent_elem') { if (this.prefs.hasPrefVal('clicked_parent_elem')) { this.clicked_parent_elem = this.prefs.getPrefVal('clicked_parent_elem'); } else { this.prefs.setPrefVal('clicked_parent_elem', this.clicked_parent_elem = clicked_parent_elem); } } } removeButton() { if (this.menuitem) { this.menuitem.parentNode.removeChild(this.menuitem); this.menuitem = null; } if (this.button) { this.button.parentNode.removeChild(this.button); this.button = null; } if (this.style) { this.style.parentNode.removeChild(this.style); this.style = null; } } rebuildButton() { this.removeButton(); if (BUTTON_TYPE === 2) { let menuitem = $C('button', { id: 'AutoClick-enable-menuitem', class: 'menuitem-iconic', type: 'checkbox', autocheck: "false", checked: AutoClick.enable, tooltiptext: '自动点击开/关', onclick: 'AutoClick.iconClick(event);', }); let insPos = $('devToolsSeparator'); insPos.parentNode.insertBefore(menuitem, insPos); this.menuitem = menuitem; } else { let button = $C('button', { id: "AutoClick-icon", class: 'toolbarbutton-1 chromeclass-toolbar-additional', title: "自动点击开/关", click: this.iconClick.bind(this), // context: "AutoClick-popup", }); if (BUTTON_TYPE === 1) { ToolbarManager.addWidget(window, button, true); } else { $('urlbar-icons').appendChild(button); } this.button = button; // background: transparent !important;\ // background-image: none !important;\ let css = '\ #AutoClick-icon {\ -moz-appearance: none !important;\ border-style: none !important;\ border-radius: 0 !important;\ padding: 0 0 !important;\ margin: 0 3px !important;\ box-shadow: none !important;\ -moz-box-align: center !important;\ -moz-box-pack: center !important;\ min-width: 18px !important;\ min-height: 18px !important;\ width: 18px;\ z-index: 100000;\ }\ #AutoClick-icon > .toolbarbutton-icon {\ max-width: 18px !important;\ padding: 0 !important;\ margin: 0 !important;\ border: 0 !important;\ background-color: transparent !important;\ box-shadow: none !important;\ -moz-transition: none !important;\ }\ #AutoClick-icon dropmarker { display:none; }\ '; this.style = addStyle(css); } } updateState(enable, isTmpDisable) { let preLabel = isTmpDisable ? '临时' : '', label = 'AutoClick 已' + preLabel + (enable ? '启用' : '停用'); if (this.button) { console.log('label', label); this.button.style.background = 'url(' + (enable ? ON_IMG : OFF_IMG) + ') no-repeat #fff'; this.button.title = label; } else if (this.menuitem) { this.menuitem.setAttribute('checked', enable); this.menuitem.setAttribute('label', label); } } handleEvent(event) { if (!this.enable) return; switch(event.type) { case 'mouseover': if (!this.tmpDisable) this.mouseover(event); break; case 'mouseout': if (!this.tmpDisable) this.mouseout(event); break; case 'keydown': if (event.keyCode === 17) { // Ctrl_L this.ctrlKey = true; } break; case 'keyup': if (event.keyCode === 17) { this.ctrlKey = false; } break; // case 'keypress': // if (event.keyCode === 18) { // Alt_L // this.tmpDisable = !this.tmpDisable; // this.updateState(!this.tmpDisable, this.tmpDisable); // } break; } } observe(aSubject, aTopic, aData){ if (aTopic == 'nsPref:changed') { switch(aData) { case 'BUTTON_TYPE': case 'enable': case 'hovering_time': case 'clicked_elem': case 'clicked_parent_elem': this.loadSetting(aData); break; } } } mouseover(event){ const elem = this.findLink(event.target); if (elem && String(elem.id).indexOf('AutoClick') === -1) { console.log('mouseover.elem: ->', elem); this.timeoutID = setTimeout(() => { this.click(elem, event.view); }, this.hovering_time); } } mouseout(event) { if (this.timeoutID) { clearTimeout(this.timeoutID); this.timeoutID = null; } } findLink(node) { var selector = node.nodeName.toLowerCase(); // if (node.id) { // selector += '#' + node.id; // } // var classNames = node.getAttribute('class'); // if (classNames) { // selector += '.' + classNames.trim().replace(/\s+/ig, '.'); // } var match = function(list) { return list.some(x => selector === x); }; if (match(this.clicked_elems)) { return node; } else if (match(this.clicked_parent_elems)) { var parent = node.parentNode; return parent && this.findLink(parent); } return null; } click(element, win) { console.log('click--->', element, win, '创建点击事件'); let doc = win.document, e = doc.createEvent('MouseEvents'); e.initEvent('click', true, true, win, 0, 0, 0, 0, 0, this.ctrlKey, false, false, false, 0, element); return !element.dispatchEvent(e); } iconClick(event) { console.log('event', event.button, this.enable); if (event.button === 0) { this.enable = !this.enable; } else { this.openPref(); event.preventDefault(); event.stopPropagation(); } } openPref() { alert('待添加,请直接修改代码顶部配置。') /*let xul = '<?xml version="1.0"?>\ <?xml-stylesheet href="chrome://global/skin/" type="text/css"?>\ \ <prefwindow\ xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"\ id="AutoClick"\ windowtype="AutoClick:Preferences">\ <prefpane id="main" flex="1">\ \ <preferences>\ <preference id="hovering_time" type="int" name="userChromeJS.AutoClick.hovering_time"/>\ <preference id="BUTTON_TYPE" type="int" name="userChromeJS.AutoClick.BUTTON_TYPE"/>\ <preference id="clicked_elem" type="string" name="userChromeJS.AutoClick.clicked_elem"/>\ <preference id="clicked_parent_elem" type="string" name="userChromeJS.AutoClick.clicked_parent_elem"/>\ </preferences>\ \ <hbox>\ <label value="悬停的时间:" />\ <textbox size="10" preference="hovering_time" />\ <label value="毫秒" />\ </hbox>\ <hbox>\ <label value="按钮的类型:" />\ <menulist preference="BUTTON_TYPE">\ <menupopup>\ <menuitem label="地址栏按钮" value="0"/>\ <menuitem label="可移动按钮" value="1"/>\ <menuitem label="菜单" value="2"/>\ </menupopup>\ </menulist>\ </hbox>\ <label value="点击的节点名:" />\ <textbox id="clicked_elem" flex="1" preference="clicked_elem" multiline="true" rows="3" />\ <label value="需要查找父元素的节点名:" />\ <textbox flex="1" id="clicked_parent_elem" preference="clicked_parent_elem" \ multiline="true" rows="3" />\ <button label="重置" tooltiptext="重置上面2个设置为默认值" />\ \ </prefpane>\ <script>\ function reset() {\ $("clicked_elem").value = opener.AutoClick.default_prefs.clicked_elem;\ $("clicked_parent_elem").value = opener.AutoClick.default_prefs.clicked_parent_elem;\ }\ function $(id) document.getElementById(id); \ </script>\ </prefwindow>\ '; window.openDialog( "data:application/vnd.mozilla.xul+xml;charset=UTF-8," + encodeURIComponent(xul), '', 'chrome,titlebar,toolbar,centerscreen,dialog=no');*/ } }; const ToolbarManager = (function() { let layoutWidget = function(document, button, isFirstRun) { // Add to the customization palette let toolbox = document.getElementById('navigator-toolbox'); console.log('toolbox--->', toolbox, button); toolbox.appendChild(button); }; let addWidget = function(window, widget, isFirstRun) { try { layoutWidget(window.document, widget, isFirstRun); } catch(error) { console.log(error); } }; let removeWidget = function(window, widgetId) { try { let widget = window.document.getElementById(widgetId); widget.parentNode.removeChild(widget); } catch(error) { console.log(error); } }; const initLoayoutContainer = () => { const div = document.createElement('div'); div.id = 'navigator-toolbox'; div.style.height = '24px'; div.style.width = '24px'; //div.style.border = 'solid 2px #a94442'; div.style.position = 'fixed'; div.style.right = '10px'; div.style.top = '10px'; div.style.zIndex = '100000'; // div.style.backgroundColor = '#777'; div.innerHTML = ''; document.body.appendChild(div); }; initLoayoutContainer(); let exports = { addWidget: addWidget, removeWidget: removeWidget, }; return exports; })(); function $(id, doc) { return (doc || document).getElementById(id); }; function $C(name, attr) { var el = document.createElement(name); if (attr) { Object.keys(attr).forEach(function(n) { if (typeof attr[n] === 'function') { el.addEventListener(n, attr[n], false); } else { el.setAttribute(n, attr[n]); } }); } return el; } function addStyle(css) { return GM_addStyle(css); } (new AutoClick()).init(); } catch(e) { console.error('AutoClick. Error', e) } })();