// ==UserScript== // @name userChromeJS Manager // @include main // @author xiaoxiaoflood // @onlyonce // ==/UserScript== // original: https://github.com/alice0775/userChrome.js/blob/master/rebuild_userChrome.uc.xul UC.rebuild = { PREF_TOOLSBUTTON: 'userChromeJS.showtoolbutton', PREF_OPENWITHSYSTEMDEFAULT: 'userChromeJS.openWithSystemDefault', menues: [], onpopup: function (event) { let document = event.target.ownerDocument; if (event.target != document.getElementById('userChromejs_options')) return; while (document.getElementById('uc-menuseparator').nextSibling) { document.getElementById('uc-menuseparator').nextSibling.remove(); } let enabled = xPref.get(_uc.PREF_ENABLED); let mi = event.target.appendChild(this.elBuilder(document, 'menuitem', { label: enabled ? 'Enabled' : 'Disabled (click to Enable)', oncommand: 'xPref.set(_uc.PREF_ENABLED, ' + !enabled + ');', type: 'checkbox', checked: enabled })); if (Object.keys(_uc.scripts).length > 1) event.target.appendChild(this.elBuilder(document, 'menuseparator')); Object.values(_uc.scripts).sort((a, b) => a.name.localeCompare(b.name)).forEach(script => { if (script.filename === _uc.ALWAYSEXECUTE) { return; } mi = event.target.appendChild(this.elBuilder(document, 'menuitem', { label: script.name ? script.name : script.filename, onclick: 'UC.rebuild.clickScriptMenu(event)', onmouseup: 'UC.rebuild.shouldPreventHide(event)', type: 'checkbox', checked: script.isEnabled, class: 'userChromejs_script', restartless: !!script.shutdown })); mi.filename = script.filename; let homepage = script.homepageURL || script.downloadURL || script.updateURL || script.reviewURL; if (homepage) mi.setAttribute('homeURL', homepage); mi.setAttribute('tooltiptext', ` Left-Click: Enable/Disable Middle-Click: Enable/Disable and keep this menu open Right-Click: Edit Ctrl + Left-Click: Reload Script Ctrl + Middle-Click: Open Homepage Ctrl + Right-Click: Uninstall `.replace(/^\n| {2,}/g, '') + (script.description ? '\nDescription: ' + script.description : '') + (homepage ? '\nHomepage: ' + homepage : '')); event.target.appendChild(mi); }); document.getElementById('showToolsMenu').setAttribute('label', 'Switch to ' + (this.showToolButton ? 'button in Navigation Bar' : 'item in Tools Menu')); }, onHamPopup: function (aEvent) { const enabledMenuItem = aEvent.target.querySelector('#appMenu-userChromeJS-enabled'); enabledMenuItem.checked = xPref.get(_uc.PREF_ENABLED); // Clear existing scripts menu entries const scriptsSeparator = aEvent.target.querySelector('#appMenu-userChromeJS-scriptsSeparator'); while (scriptsSeparator.nextSibling) { scriptsSeparator.nextSibling.remove(); } // Populate with new entries let scriptMenuItems = []; Object.values(_uc.scripts).sort((a, b) => a.name.localeCompare(b.name)).forEach(script => { if (_uc.ALWAYSEXECUTE.includes(script.filename)) return; let scriptMenuItem = UC.rebuild.createMenuItem(scriptsSeparator.ownerDocument, null, null, script.name ? script.name : script.filename); scriptMenuItem.setAttribute('onclick', 'UC.rebuild.clickScriptMenu(event)'); scriptMenuItem.type = 'checkbox'; scriptMenuItem.checked = script.isEnabled; scriptMenuItem.setAttribute('restartless', !!script.shutdown); scriptMenuItem.filename = script.filename; let homepage = script.homepageURL || script.downloadURL || script.updateURL || script.reviewURL; if (homepage) scriptMenuItem.setAttribute('homeURL', homepage); scriptMenuItem.setAttribute('tooltiptext', ` Left-Click: Enable/Disable Middle-Click: Enable/Disable and keep this menu open Right-Click: Edit Ctrl + Left-Click: Reload Script Ctrl + Middle-Click: Open Homepage Ctrl + Right-Click: Uninstall `.replace(/^\n| {2,}/g, '') + (script.description ? '\nDescription: ' + script.description : '') + (homepage ? '\nHomepage: ' + homepage : '')); scriptMenuItems.push(scriptMenuItem); }); scriptsSeparator.parentElement.append(...scriptMenuItems); }, clickScriptMenu: function (event) { const { target } = event; const { gBrowser } = event.view; const script = _uc.scripts[target.filename]; switch (event.button) { case 0: this.toggleScript(script); if (event.ctrlKey) this.toggleScript(script); break; case 1: if (event.ctrlKey) { let url = target.getAttribute('homeURL'); if (url) { gBrowser.addTab(url, { triggeringPrincipal: Services.scriptSecurityManager.createNullPrincipal({}) }); } } else { this.toggleScript(script); if (target.tagName === 'toolbarbutton') target.setAttribute('checked', script.isEnabled); } break; case 2: if (event.ctrlKey) this.uninstall(script); else this.launchEditor(script); } }, shouldPreventHide: function (event) { if (event.button == 1 && !event.ctrlKey) { const menuitem = event.target; menuitem.setAttribute('closemenu', 'none'); menuitem.parentNode.addEventListener('popuphidden', () => { menuitem.removeAttribute('closemenu'); }, { once: true }); } }, launchEditor: function (script) { let editor = xPref.get('view_source.editor.path'); let useSystemDefault = xPref.get(this.PREF_OPENWITHSYSTEMDEFAULT); if (!editor && !useSystemDefault) { let obj = { value: 'C:\\WINDOWS\\system32\\notepad.exe' }; if (Services.prompt.prompt(null, 'userChromeJS', 'Editor not defined. Paste the full path of your text editor or click cancel to use system default.', obj, null, { value: 0 })) { editor = obj.value; xPref.set('view_source.editor.path', editor); } else useSystemDefault = xPref.set(this.PREF_OPENWITHSYSTEMDEFAULT, true); } if (useSystemDefault) { script.file.launch(); } else { let editorArgs = []; let args = Services.prefs.getCharPref('view_source.editor.args'); if (args) { const argumentRE = /"([^"]+)"|(\S+)/g; while (argumentRE.test(args)) { editorArgs.push(RegExp.$1 || RegExp.$2); } } editorArgs.push(script.file.path); try { let appfile = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsIFile); appfile.initWithPath(editor); let process = Cc['@mozilla.org/process/util;1'].createInstance(Ci.nsIProcess); process.init(appfile); process.run(false, editorArgs, editorArgs.length, {}); } catch { alert('Can\'t open the editor. Go to about:config and set editor\'s path in view_source.editor.path.'); } } }, restart: function () { Services.appinfo.invalidateCachesOnRestart(); let cancelQuit = Cc['@mozilla.org/supports-PRBool;1'].createInstance(Ci.nsISupportsPRBool); Services.obs.notifyObservers(cancelQuit, 'quit-application-requested', 'restart'); if (cancelQuit.data) return; if (Services.appinfo.inSafeMode) Services.startup.restartInSafeMode(Ci.nsIAppStartup.eAttemptQuit); else Services.startup.quit(Ci.nsIAppStartup.eAttemptQuit | Ci.nsIAppStartup.eRestart); }, toggleScript: function (script) { if (script.isEnabled) { xPref.set(_uc.PREF_SCRIPTSDISABLED, script.filename + ',' + xPref.get(_uc.PREF_SCRIPTSDISABLED)); } else { xPref.set(_uc.PREF_SCRIPTSDISABLED, xPref.get(_uc.PREF_SCRIPTSDISABLED).replace(new RegExp('^' + script.filename + ',|,' + script.filename), '')); } if (xPref.get(_uc.PREF_ENABLED) && script.isEnabled && !_uc.everLoaded.includes(script.id)) { this.install(script); } else if (script.isRunning && !!script.shutdown) { this.shutdown(script); } }, toggleUI: function (byaboutconfig = false, startup = false) { this.showToolButton = xPref.get(this.PREF_TOOLSBUTTON); if (!byaboutconfig && !startup) { this.showToolButton = xPref.set(this.PREF_TOOLSBUTTON, !this.showToolButton); } _uc.windows((doc) => { doc.getElementById('userChromebtnMenu').hidden = this.showToolButton; doc.getElementById('userChromejs_Tools_Menu').hidden = !this.showToolButton; if (this.showToolButton) { doc.getElementById('userChromejs_Tools_Menu').appendChild(doc.getElementById('userChromejs_options')); } else if (!startup) { doc.getElementById('userChromebtnMenu').appendChild(doc.getElementById('userChromejs_options')); } }); }, createMenuItem: function (doc, id, icon, label, command) { const menuItem = doc.createXULElement('toolbarbutton'); menuItem.className = 'subviewbutton subviewbutton-iconic'; if (id) menuItem.id = 'appMenu-userChromeJS-' + id; menuItem.label = label; menuItem.style.listStyleImage = icon; if (command) menuItem.setAttribute('oncommand', command); return menuItem; }, install: function (script) { script = _uc.getScriptData(script.file); Services.obs.notifyObservers(null, 'startupcache-invalidate'); _uc.windows((doc, win, loc) => { if (win._uc && script.regex.test(loc.href)) { _uc.loadScript(script, win); } }, false); }, uninstall: function(script) { if (!Services.prompt.confirm(null, 'userChromeJS', 'Do you want to uninstall this script? The file will be deleted.')) return; this.shutdown(script); script.file.remove(false); xPref.set(_uc.PREF_SCRIPTSDISABLED, xPref.get(_uc.PREF_SCRIPTSDISABLED).replace(new RegExp('^' + script.filename + ',|,' + script.filename), '')); }, shutdown: function (script) { if (script.shutdown) { _uc.windows((doc, win, loc) => { if (script.regex.test(loc.href)) { try { eval(script.shutdown); } catch (ex) { Cu.reportError(ex); } if (script.onlyonce) return true; } }, false); script.isRunning = false; } }, elBuilder: function (doc, tag, props) { let el = doc.createXULElement(tag); for (let p in props) { el.setAttribute(p, props[p]); } return el; }, init: function () { this.showToolButton = xPref.get(this.PREF_TOOLSBUTTON); if (this.showToolButton === undefined) { this.showToolButton = xPref.set(this.PREF_TOOLSBUTTON, false, true); } xPref.addListener(this.PREF_TOOLSBUTTON, function (value, prefPath) { UC.rebuild.toggleUI(true); }); xPref.addListener(_uc.PREF_ENABLED, function (value, prefPath) { Object.values(_uc.scripts).forEach(script => { if (script.filename == _uc.ALWAYSEXECUTE) return; if (value && script.isEnabled && !_uc.everLoaded.includes(script.id)) { UC.rebuild.install(script); } else if (!value && script.isRunning && !!script.shutdown) { UC.rebuild.shutdown(script); } }); }); if (AppConstants.MOZ_APP_NAME !== 'thunderbird') { const { CustomizableUI } = window; CustomizableUI.createWidget({ id: 'userChromebtnMenu', type: 'custom', defaultArea: CustomizableUI.AREA_NAVBAR, onBuild: (doc) => { return this.createButton(doc); } }); } else { const { document, location } = window; const btn = this.createButton(document); btn.setAttribute('removable', true); const toolbar = document.querySelector('toolbar[customizable=true].chromeclass-toolbar'); if (toolbar.parentElement.palette) toolbar.parentElement.palette.appendChild(btn); else toolbar.appendChild(btn); if (xPref.get('userChromeJS.firstRun') !== false) { xPref.set('userChromeJS.firstRun', false); if (!toolbar.getAttribute('currentset').split(',').includes(btn.id)) { toolbar.appendChild(btn); toolbar.setAttribute('currentset', toolbar.currentSet); Services.xulStore.persist(toolbar, 'currentset'); } } else { toolbar.currentSet = Services.xulStore.getValue(location.href, toolbar.id, 'currentset'); toolbar.setAttribute('currentset', toolbar.currentSet); } } }, createButton (aDocument) { let toolbaritem = UC.rebuild.elBuilder(aDocument, 'toolbarbutton', { id: 'userChromebtnMenu', label: 'userChromeJS', tooltiptext: 'userChromeJS Manager', type: 'menu', class: 'toolbarbutton-1 chromeclass-toolbar-additional', style: 'list-style-image: url()', }); let mp = UC.rebuild.elBuilder(aDocument, 'menupopup', { id: 'userChromejs_options', onpopupshowing: 'UC.rebuild.onpopup(event);', oncontextmenu: 'event.preventDefault();' }); toolbaritem.appendChild(mp); let mg = mp.appendChild(aDocument.createXULElement('menugroup')); mg.setAttribute('id', 'uc-menugroup'); let mi1 = UC.rebuild.elBuilder(aDocument, 'menuitem', { id: 'userChromejs_openChromeFolder', label: 'Open chrome directory', class: 'menuitem-iconic', flex: '1', style: 'list-style-image: url()', oncommand: 'Services.dirsvc.get(\'UChrm\', Ci.nsIFile).launch();' }); mg.appendChild(mi1); let tb = UC.rebuild.elBuilder(aDocument, 'menuitem', { id: 'userChromejs_restartApp', class: 'menuitem-iconic', tooltiptext: 'Restart ' + _uc.BROWSERNAME, style: 'list-style-image: url()', oncommand: 'UC.rebuild.restart();' }); mg.appendChild(tb); let mn = UC.rebuild.elBuilder(aDocument, 'menu', { id: 'uc-manageMenu', label: 'Settings', class: 'menuitem-iconic', style: 'list-style-image: url()' }); mp.appendChild(mn); let mp2 = mn.appendChild(aDocument.createXULElement('menupopup')); let mi2 = UC.rebuild.elBuilder(aDocument, 'menuitem', { id: 'showToolsMenu', label: 'Switch display mode', class: 'menuitem-iconic', style: 'list-style-image: url()', oncommand: 'UC.rebuild.toggleUI();' }); mp2.appendChild(mi2); let sep = mp.appendChild(aDocument.createXULElement('menuseparator')); sep.setAttribute('id', 'uc-menuseparator'); let mi = UC.rebuild.elBuilder(aDocument, 'menu', { id: 'userChromejs_Tools_Menu', label: 'userChromeJS Manager', tooltiptext: 'UC Script Manager', class: 'menu-iconic', image: '', }); aDocument.getElementById(AppConstants.MOZ_APP_NAME !== 'thunderbird' ? 'devToolsSeparator' : 'prefSep').insertAdjacentElement('afterend', mi);//taskPopup let menupopup = aDocument.getElementById('userChromejs_options'); UC.rebuild.menues.forEach(menu => { menupopup.insertBefore(menu, aDocument.getElementById('uc-menuseparator')); }) let pi = aDocument.createProcessingInstruction( 'xml-stylesheet', 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent(` #userChromejs_options menuitem[restartless="true"] { color: blue; } #userChromejs_restartApp { width: 34px; } `.replace(/[\r\n\t]/g, '')) + '"' ); aDocument.insertBefore(pi, aDocument.documentElement); aDocument.defaultView.setTimeout((() => UC.rebuild.toggleUI(false, true)), 1000); const viewCache = aDocument.getElementById('appMenu-viewCache')?.content || aDocument.getElementById('appMenu-multiView'); if (viewCache) { const userChromeJsPanel = aDocument.createXULElement('panelview'); userChromeJsPanel.id = 'appMenu-userChromeJsView'; userChromeJsPanel.className = 'PanelUI-subView'; userChromeJsPanel.addEventListener('ViewShowing', UC.rebuild.onHamPopup); const subviewBody = aDocument.createXULElement('vbox'); subviewBody.className = 'panel-subview-body'; subviewBody.appendChild(UC.rebuild.createMenuItem(aDocument, 'openChrome', 'url(chrome://browser/skin/folder.svg)', 'Open chrome directory', 'Services.dirsvc.get(\'UChrm\', Ci.nsIFile).launch();')); subviewBody.appendChild(UC.rebuild.createMenuItem(aDocument, 'restart', 'url(chrome://browser/skin/reload.svg)', 'Restart ' + _uc.BROWSERNAME, 'UC.rebuild.restart();')); subviewBody.appendChild(aDocument.createXULElement('toolbarseparator')); const enabledMenuItem = UC.rebuild.createMenuItem(aDocument, 'enabled', null, 'Enabled', 'xPref.set(_uc.PREF_ENABLED, !!this.checked)'); enabledMenuItem.type = 'checkbox'; subviewBody.appendChild(enabledMenuItem); const scriptsSeparator = aDocument.createXULElement('toolbarseparator'); scriptsSeparator.id = 'appMenu-userChromeJS-scriptsSeparator'; subviewBody.appendChild(scriptsSeparator); userChromeJsPanel.appendChild(subviewBody); viewCache.appendChild(userChromeJsPanel); const scriptsButton = aDocument.createXULElement('toolbarbutton'); scriptsButton.id = 'appMenu-userChromeJS-button'; scriptsButton.className = 'subviewbutton subviewbutton-iconic subviewbutton-nav'; scriptsButton.label = 'User Scripts'; scriptsButton.style.listStyleImage = 'url()'; scriptsButton.setAttribute('closemenu', 'none'); scriptsButton.setAttribute('oncommand', 'PanelUI.showSubView(\'appMenu-userChromeJsView\', this)'); const addonsButton = aDocument.getElementById('appMenu-extensions-themes-button') ?? aDocument.getElementById('appmenu_addons') ?? viewCache.querySelector('#appMenu-extensions-themes-button'); addonsButton.parentElement.insertBefore(scriptsButton, addonsButton); } return toolbaritem; } } UC.rebuild.init();