firefox-scripts icon indicating copy to clipboard operation
firefox-scripts copied to clipboard

Several breaks in 136.0

Open marty60 opened this issue 11 months ago • 9 comments

Just a heads up, Mozilla broke several addons, including keyconfig and password manager (Saved Password Editor Redux), several script and custom buttons, and the userChromeJS Manager in the Jan 8th 136.0 nightly. There may be more unknown breaks depending on what addons or scripts someone is using.

The culprit is this bug:

https://phabricator.services.mozilla.com/D232982

If anyone has time to take a look at what changes can be made to restore the addons and userChromeJS Manager it would be appreciated. I especially hate losing keyconfig and password manager.

marty60 avatar Jan 13 '25 22:01 marty60

For your reference: https://github.com/onemen/TabMixPlus/issues/381#issuecomment-2581037140

117649 avatar Jan 14 '25 01:01 117649

That did it, toggled "security.browser_xhtml_csp.enabled" to false and now everything's back. Thanks for the link.

onemen is right though it would be better to fix the addons themselves rather than keep that as permanent since it is a security pref.

Interestingly, I had swapped into omni.ja the previous day's browser.xhtml and that fixed it as well without changing the pref. But that won't last since they make periodic changes to the file so it was only temporary.

marty60 avatar Jan 14 '25 02:01 marty60

Worring about a security pref while doing the most insecure from Mozilla's POV?😅

117649 avatar Jan 14 '25 04:01 117649

Worring about a security pref while doing the most insecure from Mozilla's POV?

😅

marty60 avatar Jan 14 '25 04:01 marty60

We plan on disabling inline event handlers in Beta with either version 136 or 137. I am happy to answer questions.

evilpie avatar Jan 23 '25 11:01 evilpie

We plan on disabling inline event handlers in Beta with either version 136 or 137. I am happy to answer questions.

Hi evilpie,

So this change is not just for nightlies but release versions as well?

Also will toggling "security.browser_xhtml_csp.enabled" to false continue to restore the broken addons?

thanks.

marty60 avatar Jan 23 '25 13:01 marty60

So this change is not just for nightlies but release versions as well?

We plan on shipping this change in release as soon as possible. But probably not in 136.

Also will toggling "security.browser_xhtml_csp.enabled" to false continue to restore the broken addons?

For now, most likely. We definitely want to remove the pref in the future.

evilpie avatar Jan 23 '25 14:01 evilpie

@evilpie When are your`s going to disconnect autoconfig

dimdamin avatar Jan 23 '25 18:01 dimdamin

Patched rebuild_userChrome.uc.js for inline event.

Can't find anywhere else for this to go right now. @xiaoxiaoflood Can you check this and see if you want to update the repo?

// ==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: event => UC.rebuild.clickScriptMenu(event),
        onmouseup: event => 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.onclick = event => 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.addEventListener('command', 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) {
      if(p.startsWith('on')) 
        el.addEventListener(p.slice(2), props[p]);
      else 
        el.setAttribute(p, props[p]);
    }
    return el;
  },

  setStyle: function () {
    _uc.sss.loadAndRegisterSheet(Services.io.newURI('data:text/css;charset=UTF-8,' + encodeURIComponent(`
      @-moz-document url('${_uc.BROWSERCHROME}') {
        #userChromejs_options menuitem[restartless="true"] {
          color: blue;
        }
        #userChromejs_restartApp {
          padding-right: 4px;
        }
        #userChromejs_restartApp > .menu-iconic-left {
          margin-inline-end: 0 !important;
          padding-inline-end: 0 !important;
        }

        #userChromejs_openChromeFolder {
          padding-inline-start: 12px;
        }

        #userChromejs_restartApp > .menu-accel-container {
          display: none;
        }

        /* bug 1828413: checkbox is only rendering on mouseover/mouseout */
        menuitem[type="checkbox"][checked="true"] .menu-iconic-icon {
          appearance: checkbox !important;
        }

        @media (-moz-platform: windows) {
          #userChromejs_openChromeFolder {
            padding-block: 0.5em;
          }
          #userChromejs_restartApp {
            padding: 0 8px !important;
          }
          #userChromejs_restartApp > .menu-iconic-left {
            padding-top: 0;
          }
        }

        @media (-moz-platform: linux) {
          #userChromejs_restartApp {
            padding-right: 4px !important;
          }
        }
      }
    `)), _uc.sss.USER_SHEET);
  },

  init: function () {
    this.setStyle();
    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) => {
          this.createPanel(doc);
          return this.createButton(doc);
        }
      });
    } else {
      this.createPanel(window.document);
    }
  },

  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: event => UC.rebuild.onpopup(event),
      oncontextmenu: event => 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'));            
    });

    aDocument.defaultView.setTimeout((() => UC.rebuild.toggleUI(false, true)), 1000);

    return toolbaritem;
  },

  createPanel (aDocument) {
    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.addEventListener('command', function() {Cu.getGlobalForObject(this).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);
    }
  }
}

UC.rebuild.init();

117649 avatar Jan 26 '25 14:01 117649