Several breaks in 136.0
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.
For your reference: https://github.com/onemen/TabMixPlus/issues/381#issuecomment-2581037140
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.
Worring about a security pref while doing the most insecure from Mozilla's POV?😅
Worring about a security pref while doing the most insecure from Mozilla's POV?
😅
We plan on disabling inline event handlers in Beta with either version 136 or 137. I am happy to answer questions.
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.
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 When are your`s going to disconnect autoconfig
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();