Dumby
Я имел ввиду выделять userChoice, а не наоборот, у себя изменил.
По поводу пунктов true-false:
Текст значения после заголовка добавьте пожалуйста, что бы к единому стилю привести. Например после:
pref: ["javascript.enabled", "Выполнять скрипты Java", "", "javascript.enabled"], userChoice: true, refresh: true, values: [[true, "Да"], [false, "Нет"]]
user ? node.style.setProperty("font-weight", "bold", "important") : node.style.removeProperty("font-weight");
Отредактировано _zt (13-08-2020 18:33:41)
Отсутствует
Вот здесь же bold для user задается, или я чего то не понял.
это для текущего значения - prefHasUserValue, а чтобы проверить другие значения (в подменю) нужно определить его дефолтное значение,
не переключать же настройку и проверять каждое...
Отсутствует
Vitaliy V.
Теперь понятно. Тогда может на базе вашего кода решение найдется.
Dumby
maroon я у помянул так как он у меня не работает. Сами видите какое у меня меню, так что не удивительно, что цвета в нем не работают.
Вы все правильно сделали, только, на мой взгляд, логичнее UserChoice выделять, а не отличные от него.
Хотя, ради баловства, можно для каждого значения свою иконку назначить - UserChoice / notUserChoice.
Отредактировано _zt (13-08-2020 16:12:57)
Отсутствует
Dumby
Все таки добавил иконку на notUserChoice. Казалось, что будет сложнее.
Теперь вопрос появился, а нельзя ли на иконку (или весь пункт) клик ЛКМ повесить? (ПКМ там и так работает, сбрасывает значение на "по умолчанию" (я так понимаю)).
Дело в том, что подменю может появится и справа и слева, и все это приводит к бестолковым перемещениям мыши. А так можно будет только информацию снимать глазами с подменю, а переключать щелчком мыши.
Последовательное переключение неприемлемо (из-за refresh: true, restart: true), но можно добавить какой нибудь UserAlt и переключать по клику ЛКМ (UserChoice > UserAlt, UserAlt > UserChoice), если UserAlt назначен пункту.
В этом случае, UserAlt тоже можно будет свою иконку приделать.
Отредактировано _zt (13-08-2020 18:29:59)
Отсутствует
Добрый день. Посмотрите пожалуйста код для FF71 " Удалить куки текущего сайта в контекстном меню на странице" для контекстного меню.
// Удалить куки текущего сайта в контекстном меню на странице, от 06.03.2017. ................ (()=> { (contextMenu=> { var menu = contextMenu.appendChild(document.createElement("menu")); menu.id = "content-removeCookies." menu.setAttribute("label", "Удалить куки текущего сайта"); menu.setAttribute("class", "menu-iconic"); addEventListener("popupshowing", ()=> menu.setAttribute("image", gBrowser.mCurrentTab.image), false, contextMenu); addDestructor(()=> menu.remove() ); var menuPopup = menu.appendChild(document.createElement("menupopup")); var menuitem = menuPopup.appendChild(document.createElement("menuitem")); menuitem.setAttribute("label", "Удалить"); menuitem.onclick =()=> { var host = Services.eTLD.getBaseDomain(gBrowser.currentURI); for ( var en = Services.cookies.enumerator; en.hasMoreElements(); ) { var cookie = en.getNext().QueryInterface(Ci.nsICookie); ~cookie.host.indexOf(host.trim()) && Services.cookies.remove(cookie.host, cookie.name, cookie.path, false); } var reversedDomain = host.split("").reverse().join("") + "."; Cu.import("resource://gre/modules/FileUtils.jsm"); var file = FileUtils.getFile("ProfD", ["webappsstore.sqlite"]); var dbConn = Services.storage.openDatabase(file); dbConn.executeSimpleSQL("DELETE FROM webappsstore2 WHERE scope LIKE \"%" + reversedDomain +"%\""); dbConn.close(); var host = content.document.domain; for( var tab of gBrowser.tabs ) { var tabHost = tab.linkedBrowser.contentDocument.location.host; if ( host == tabHost ) gBrowser.reloadTab(tab); } // очистить кэш .... try { Services.cache.evictEntries(1); Services.cache.evictEntries(2); } catch(e) { Services.cache2.clear() }; // удалить все Flash куки var dir = Services.dirsvc.get('Home', Ci.nsIFile); dir.append("Application Data"); dir.append("Macromedia"); //dir.remove(true); //dir.create(Ci.nsIFile.DIRECTORY_TYPE, 0777); dir.exists() && dir.remove(true); // Удалить Flash кукисы222 .... !dir.exists() && dir.create(Ci.nsIFile.DIRECTORY_TYPE, 0777); // Всплывающее сообщение .... var win = openDialog("chrome://global/content/alerts/alert.xul", "", "popup=yes", (gBrowser.mCurrentTab.image || "chrome://global/skin/icons/Portrait.png"), "", "Очистил куки, кеш текущего сайта и удалил Flash куки", false, null, 4); setTimeout(()=> win && win.close(), 2500); }; })(document.getElementById("contentAreaContextMenu")); })();
Отредактировано Duche (14-08-2020 13:41:07)
Отсутствует
пробовал в начало кода добавлять это:
gBrowser.selectedTab = gBrowser.addTab('http://site.ru');
Не работает)
Направление мысли верное, но следует дождаться пока загрузится.
Попробуй разместить в Инициализации, и если «нужный сайт»
прям после загрузки ещё не готов быть снапшотнутым, тогда
подними таймаут с нуля до какого-нибудь эмпирического значения.
(exec => addEventListener("click", e => { if (e.button) return; e.preventDefault(); var url = "http://site.ru"; var br = (gBrowser.selectedTab = gBrowser.addTab(url)).linkedBrowser; br.addEventListener("pageshow", () => setTimeout(exec, 0, br, url), {once: true}); }, true, this))((br, url) => gBrowser.selectedBrowser == br && br.currentURI.spec == url && custombutton.buttonCbExecuteCode(null, this, this.cbCommand) );
Хотелось бы совместить оба варианта - для _zt и для меня с методом londPress, они почему-то сильно различаются…
Ну, ты выразил пожелание чтобы менюшки закрывались,
а _zt к этому пожеланию не присоединился, и направился
куда-то в сторону единостилия. Поэтому и различаются.
Но londPress достаточно автономен, там только здесь вписан "mousedown":
for(var type of ["command", "contextmenu", "mousedown"])
и, в конце, метод mousedown, обрабатывающий это событие,
и метод londPress, куда передаётся результат обработки.
может как то так
Да! Это работает! Спасибо!
И ведь что-то подобное в коде есть, но мысль проверить
что возвращает defaultBranch мою тупую голову не посетила.
у себя изменил
Смотрю много измененений.
Скинь тогда свой код целиком, чтобы было с чем возиться.
С проставленными userAlt и userAltImg.
Резюмирую, скорее для себя, что попробовать сделать:
1. italic в субменюшках для недефолтных значений,
учитывая наставление Виталия.
2. длинное тире + лейбл и для true-false <menu>
при наличии "values".
3. Последовательное переключение по клику на <menu>
если отсутствуют restart и refresh,
иначе, переключение между userChoice и userAlt,
если присутствуют оба. Плюс иконка userAlt.
(Если текущее значение не userChoice и не userAlt
переключать на userChoice или на userAlt ?)
Всё это займёт какое-то время, полагаю.
как то можно отключить backup
Наверно да, %Profile%\extensions\custombuttons@xsms.org\components\CustomButtonsService.js
function backupProfile (phase) {if(true)return;
Удалить куки текущего сайта в контекстном меню на странице
Может подойдёт в контекстном меню не на странице, а на кнопке?
Отсутствует
firepox пишетпробовал в начало кода добавлять это:gBrowser.selectedTab = gBrowser.addTab('http://site.ru');Не работает)Направление мысли верное, но следует дождаться пока загрузится.Попробуй разместить в Инициализации, и если «нужный сайт»прям после загрузки ещё не готов быть снапшотнутым, тогдаподними таймаут с нуля до какого-нибудь эмпирического значения.
код вставил в инициализацию - ничего не сохраняется вообще
Отредактировано firepox (15-08-2020 11:40:37)
Отсутствует
Dumby
Резюмирую, скорее для себя, что попробовать сделать:
1. italic в субменюшках для недефолтных значений,
учитывая наставление Виталия.
2. длинное тире + лейбл и для true-false <menu>
при наличии "values".
3. Здесь, переключение только между userChoice и userAlt, если userAlt существует.
Если сделать иначе (например, последовательно), будет путаница и проблемы с restart и refresh.
... Думаю не стоит усложнять. Более двух никогда не нужно, и ведь еще есть уже работающий ПКМ.
Да и вообще, при последовательном переключении смысл UserAlt теряется,
он был выдуман, что бы избежать проблем последовательного переключения.
Если текущее значение не userChoice и не userAlt
переключать на userChoice или на userAlt ?
На userChoice. Так как он основной.
UserAlt в немногих пунктах потребуется, а userChoice можно прописать везде.
Смотрю много измененений.
Вы мне льстите. Не много.
Иконку UserAlt прописал и она работает, но вы проверьте за мной, я просто интуитивно сделал. В трех пунктах добавил новый атрибут.
data:text/plain;charset=utf-8;base64,// Быстрое переключение параметров about:config
(async (name, id, func) =&gt; {
	if (name == &quot;Object&quot;) return CustomizableUI.createWidget(func());
	var win = name == &quot;Window&quot;, g = Components.utils.import(&quot;resource://gre/modules/Services.jsm&quot;, {});
	if (g[id]) {if (win) return;} else g[id] = func();
	if (win) return CustomizableUI.createWidget(g[id]);
	addDestructor(r =&gt; r[5] == &quot;e&quot; &amp;&amp; delete g[id]);
	g[id].onCreated(this);
})(this.constructor.name, &quot;QuickToggleAboutConfigSettings&quot;, () =&gt; {

	var {prefs} = Services, db = prefs.getDefaultBranch(&quot;&quot;);
	var pv = parseInt(Services.appinfo.platformVersion);
	var xul_ns = &quot;http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul&quot;;

	var localhost = &quot;127.0.0.1&quot;;
	var anticensority = &quot;https://git.io/ac-anticensority-pac&quot;;
	var antizapret = &quot;https://antizapret.prostovpn.org/proxy.pac&quot;;
	var pacfile = prefs.getStringPref(&quot;user.pacfile&quot;, &quot;file:///etc/proxy.pac&quot;);
	var useragent = &quot;Mozilla/5.0 (Windows NT 6.1; WOW64; rv:56.0) Gecko/20100101 Firefox/56.0&quot;;

//=====================================================================================

	// refresh:
	//	false - reload current tab
	//	true - reload current tab skip cache
	//
	// restart:
	//	false - restart browser
	//	true - restart browser with confirm

	var primary = [{

			pref: [&quot;network.proxy.type&quot;, &quot;Настройки прокси&quot;, &quot;&quot;, &quot;network.proxy.type&quot;],
			userChoice: 5, userAlt: 1, refresh: true,
			values: [
				[0, &quot;Не проксировать&quot;, &quot;0&quot;], [5, &quot;Системные (из IE)&quot;, &quot;5&quot;], [2, &quot;Авто (pacfile)&quot;, &quot;2&quot;],
				[1, &quot;Прописанные&quot;, &quot;1&quot;], [4, &quot;Автоопределение&quot;, &quot;4&quot;]
	]},
			null,
	{
			pref: [&quot;permissions.default.image&quot;, &quot;Загружать графику&quot;, &quot;&quot;, &quot;permissions.default.image&quot;],
			userChoice: 1, refresh: true,
			values: [[1, &quot;Да&quot;], [3, &quot;С сайта&quot;], [2, &quot;Нет&quot;]]
	},{
			pref: [&quot;browser.display.use_document_fonts&quot;, &quot;Загружать web-шрифты&quot;, &quot;&quot;, &quot;browser.display.use_document_fonts&quot;],
			userChoice: 1, refresh: true,
			values: [[1, &quot;Да&quot;], [0, &quot;Нет&quot;]]
	},{
			pref: [&quot;javascript.enabled&quot;, &quot;Выполнять скрипты Java&quot;, &quot;&quot;, &quot;javascript.enabled&quot;],
			userChoice: true, refresh: true,
			values: [[true, &quot;Да&quot;], [false, &quot;Нет&quot;]]
	},{
			pref: [&quot;media.autoplay.default&quot;, &quot;Автозапуск медиа&quot;, &quot;&quot;, &quot;media.autoplay.default&quot;],
			userChoice: 5, refresh: true,
			values: [
				[5, &quot;Блокировать все&quot;, &quot;5&quot;],
				[1, &quot;Блокировать не приглушенное&quot;, &quot;1&quot;],
				[0, &quot;Разрешить все&quot;, &quot;0&quot;]
	]},{
			pref: [&quot;media.autoplay.blocking_policy&quot;, &quot;Автозапуск (политика)&quot;, &quot;&quot;, &quot;media.autoplay.blocking_policy&quot;],
			userChoice: 1, userAlt: 2, refresh: true,
			values: [
				[1, &quot;Временная&quot;, &quot;1&quot;],
				[2, &quot;По действию&quot;, &quot;2&quot;],
				[0, &quot;Постоянная&quot;, &quot;0&quot;]
	]},{
			pref: [&quot;network.cookie.cookieBehavior&quot;, &quot;Cookies&quot;, &quot;&quot;, &quot;network.cookie.cookieBehavior&quot;],
			userChoice: 1, userAlt: 3, refresh: false,
			values: [
				[1, &quot;Не принимать сторонние&quot;], [3, &quot;Не принимать с не посещенных&quot;], [4, &quot;Не принимать от трекеров&quot;],
				[2, &quot;Не принимать со всех&quot;], [0, &quot;Принимать со всех&quot;]
	]},
			null,
	{
			pref: [&quot;dom.storage.enabled&quot;, &quot;Локальное хранилище&quot;, &quot;&quot;, &quot;dom.storage.enabled&quot;],
			userChoice: true
	}
];

//=====================================================================================

	var secondary = [{

			pref: [&quot;dom.serviceWorkers.enabled&quot;, &quot;Видео dom.serviceWorkers&quot;, &quot;&quot;, &quot;dom.serviceWorkers.enabled&quot;],
			userChoice: false
	},{
			pref: [&quot;dom.enable_performance&quot;, &quot;Статус загрузки страницы&quot;, &quot;&quot;, &quot;dom.enable_performance&quot;],
			userChoice: false
	},
			null,
	{
			pref: [&quot;browser.cache.memory.enable&quot;, &quot;Кэш в оперативной памяти&quot;, &quot;&quot;, &quot;browser.cache.memory.enable&quot;],
			userChoice: true
	},
			null,
	{
			pref: [&quot;intl.accept_languages&quot;, &quot;Язык для веб-страниц&quot;, &quot;&quot;, &quot;intl.accept_languages&quot;],
			userChoice: &quot;en-US, en&quot;,
			values: [[&quot;en-US, en&quot;, &quot;en-US, en&quot;], [&quot;en-US, en, ru-RU, ru&quot;, &quot;en-US, en, ru-RU, ru&quot;]]
	},{
			pref: [&quot;browser.display.document_color_use&quot;, &quot;Использовать цвета сайтов&quot;, &quot;&quot;, &quot;browser.display.document_color_use&quot;],
			userChoice: 0,
			values: [[0, &quot;Авто&quot;, &quot;0&quot;], [1, &quot;Всегда&quot;, &quot;1&quot;], [2, &quot;Никогда&quot;, &quot;2&quot;]]
	},
			null,
	{
			pref: [&quot;network.http.sendRefererHeader&quot;, &quot;Referer - для чего&quot;, &quot;&quot;, &quot;network.http.sendRefererHeader&quot;],
			userChoice: 1,
			values: [[0, &quot;Ни для чего&quot;, &quot;0&quot;], [1, &quot;Только ссылки&quot;, &quot;1&quot;], [2, &quot;Ссылки и изобр.&quot;, &quot;2&quot;]]
	},{
			pref: [&quot;network.http.referer.trimmingPolicy&quot;, &quot;Referer - что&quot;, &quot;&quot;, &quot;network.http.referer.trimmingPolicy&quot;],
			userChoice: 0,
			values: [[0, &quot;Полный URL&quot;, &quot;0&quot;], [1, &quot;scheme+host+port+path&quot;, &quot;1&quot;], [2, &quot;scheme+host+port&quot;, &quot;2&quot;]]
	},{
			pref: [&quot;network.http.referer.XOriginPolicy&quot;, &quot;RefererXO - когда&quot;, &quot;&quot;, &quot;network.http.referer.XOriginPolicy&quot;],
			userChoice: 0,
			values: [[0, &quot;В любом случае&quot;, &quot;0&quot;], [1, &quot;При совп. баз. домена&quot;, &quot;1&quot;], [2, &quot;При совпадении адреса&quot;, &quot;2&quot;]]
	},{
			pref: [&quot;network.http.referer.XOriginTrimmingPolicy&quot;, &quot;RefererXO - что&quot;, &quot;&quot;, &quot;network.http.referer.XOriginTrimmingPolicy&quot;],
			userChoice: 0,
			values: [[0, &quot;Полный URL&quot;, &quot;0&quot;], [1, &quot;scheme+host+port+path&quot;, &quot;1&quot;], [2, &quot;scheme+host+port&quot;, &quot;2&quot;]]
	},{
			pref: [&quot;network.http.referer.spoofSource&quot;, &quot;Referer - корень сайта&quot;, &quot;&quot;, &quot;network.http.referer.spoofSource&quot;],
			userChoice: false
	},
			null,
	{
			pref: [&quot;media.peerconnection.enabled&quot;, &quot;WebRTC утечка IP&quot;, &quot;&quot;, &quot;media.peerconnection.enabled&quot;],
			userChoice: false
	}
	];

	return {
		label: &quot;Quick toggle&quot;,
		id: &quot;QuickToggleAboutConfigSettings&quot;,
		localized: false,
		image: &quot;data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAFPSURBVDhPnVPNSsNAEJ5tamnBk+A7BMEcikohp/gi9i18nRafQoT00kK16tnn8CJo4vp9m5m41eSgH0zmd3d+diJ/gfdedvl0QW7k1PcLwem+3TjowZagK+Nn6yc3gNCJ+LCCh9LzzTMdqeoNbkQSFZmpJLGCW5Gx2hZaQahMbZ52VuDSfHpnAVJXOb4FKzianbzCXiLrHLYl5C3t5NTRwtxaKJxUTZ/JcOTr6q2Rxwf0aVb2/hLs4KqL3Isckz/mmSdRRksjyg+z7IM6LzDCBfuvAEOJmIKBOqAWmo1YodxLlQNsyO0rsIV/AZM+JGc2a4GgzhZ2p9mFmrpboCMqNYBDdMlwompoTWN6F2nFT5hBXb0PEpmo/Gk+oH+RrqNZINPeIlnZViV1jQuLFJQuWGB0gJdwgShvqdPXZv6J+F/QS7oXqQ9xBTFZ5oa8fAGs9fed5YhPSwAAAABJRU5ErkJggg==&quot;,
		onCreated(btn) {
			btn.setAttribute(&quot;image&quot;, this.image);
			var doc = btn.ownerDocument;

			btn.btn = true;
			btn.domParent = null;
			btn.popups = new btn.ownerGlobal.Array();
			this.createPopup(doc, btn, &quot;primary&quot;, primary);
			this.createPopup(doc, btn, &quot;secondary&quot;, secondary);

			btn.linkedObject = this;
			for(var type of [&quot;command&quot;, &quot;contextmenu&quot;])
				btn.setAttribute(&quot;on&quot; + type, `linkedObject.${type}(event)`);
		},
		createPopup(doc, btn, name, data) {
			var popup = doc.createElementNS(xul_ns, &quot;menupopup&quot;);
			var prop = name + &quot;Popup&quot;;
			btn.popups.push(btn[prop] = popup);
			popup.id = this.id + &quot;-&quot; + prop;
			popup.setAttribute(&quot;onpopupshowing&quot;, &quot;parentNode.linkedObject.popupshowing(event)&quot;);
			for(var obj of data) popup.append(this.createElement(doc, obj));
			btn.append(popup);
		},
		createElement(doc, obj) {
			if (!obj) return doc.createElementNS(xul_ns, &quot;menuseparator&quot;);
			var pref = doc.ownerGlobal.Object.create(null), node, img;
			for(var [key, val] of Object.entries(obj)) {
				if (key == &quot;pref&quot;) {
					var [apref, lab, akey, ttt] = val;
					pref.pref = apref; pref.lab = lab || apref;
					if (ttt) pref.ttt = ttt;
				}
				else if (key == &quot;image&quot;) img = val, pref.img = true;
				else if (key != &quot;values&quot;) pref[key] = val;
			}
			var type = prefs.getPrefType(pref.pref), str, bool;
			var pint = type == prefs.PREF_INT, inv = type == prefs.PREF_INVALID;
			if (inv &amp;&amp; obj.values || pint || type == prefs.PREF_STRING)
				str = pint || (inv &amp;&amp; typeof obj.values[0][0] == &quot;number&quot;) ? &quot;Int&quot; : &quot;String&quot;,
				pref.bool = false;
			else
				str = &quot;Bool&quot;, pref.bool = true;
			pref.get = prefs[`get${str}Pref`];
			pref.set = prefs[`set${str}Pref`];

			node = doc.createElementNS(xul_ns, &quot;menu&quot;);
			node.className = &quot;menu-iconic&quot;;
			node.setAttribute(&quot;closemenu&quot;, &quot;none&quot;);
			img &amp;&amp; node.setAttribute(&quot;image&quot;, img);
			akey &amp;&amp; node.setAttribute(&quot;accesskey&quot;, akey);
			(node.pref = pref).vals = doc.ownerGlobal.Object.create(null);
			this.createRadios(doc,
				pref.bool ? [[true, &quot;true&quot;], [false, &quot;false&quot;]] : obj.values,
				node.appendChild(doc.createElementNS(xul_ns, &quot;menupopup&quot;))
			);
			return node;
		},
		UserChoiceImg: &quot;data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACRUlEQVR42qWTS2gTQRjH/zPJZo2BDfgkB62vYg5CjQdvRUE8WB8HKyahYotIQQOeFKGCF7EgvQlBqOCjWFKlevBRDyL4uHloLXhYqWJroaEqlETSptndGb+ZPDH0ojN8O8vA98vv+/Itw38u1nSRYRE6UhKyg4FF1R292/Q+xhhLi4TIrgjgIzwe4MbgmR2nrZMtxxEN63zYORuPZp7gzpehvCu9XjfuPmwCUHIiEtw4fK89zVvD2+FHAD56quXBpV3CVO4ret6nxHzxZxdBRuqADCKmz7RHD9y2tlmbYbBVfwE8DXBkEd/y39H5+mze8ZyoSIpsFXC9u/VE3/ldpyg5AAMEoJMTQi2hANKBgyKdy0h/eoChqcf9MimvVAHjd/fdiG0NRwgQpDAp1agBpDZwdHKJLKZzc+h5e3kCSexh5c5j8fmhW0HGpNZXFkqfsQpAKgcFKGmAlAxHXp5bkkmsrhosjh4cCIJ5pG5qfQXgtMslqO1qgEMWTPrR+eriEhoA4wN7L8TWh0KVXzfgY2RQAUjVBenWLH4UCrj04Wa9BNXEo5va+w63xCiJ6WRVP2MVgNQIgjgEA17MTODZ7Lt+AtSaGDG4377a1m2FAkKrc11/dUwIID1dSKHEcW3yft4RbpR6kK1PYgaJtaY1nNp5jIeMsjKNbs1AWRUcjvTnp2Kh9LtLJGTDINUhcTIZ3L+hzdq9ZgvWmSF9/Wu5gI8L03gzP5l3hddL/3/zKDdA9MdE0UERrdzaFGMUaap75Y/pX9Yfap76EQYaCeEAAAAASUVORK5CYII=&quot;,
		notUserChoiceImg: &quot;data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACNUlEQVR42qXTT2jTUBwH8G9e0ta2ks5ND0Ww/qX1oGyevEwQnOBA6EHXwLwoUh3eRGRs501RPE6xgp6KrexQECZY8DBPXmbRS4Z/K0gRtmmDa2yb5PlLX7p2Gwpi4EfCy8sn3/dPwn9e0saGnCRF6XYFnA9DkhKtRs51ep6jmtEcp/JHIM9Yivl9mf0XNDV2Ngn1YLzVxdB1lJ8U8P5hzuCWnR6xrPwmIM8kLRjdkT326BaLHNgLKAFA9ouXdhOw6qi++4j589cd89vSaMqyc2sAPUXlgF8fyt9WI3t2Av4w4NsCMJ8AHAKav6hqqH76iuLINcNuNhOawyttYCp+7tRE/6XT9GFQAG4CpniATQkIaNQIMVG6/xSL2WfTGueTbWDh5L2rA9t29QGBrQQEPUDuAuoEmFSr+P5lGc8v33mtAUfaQO3M4/GgDOoYCIkUin8D0BDDoBQ2ZMxqN0wCQh0gMxaUnYb4uzt+dwIZ8wBHTGQbYAHMpu+uAxaGxpMDvSr3AHcFaAIlD+BtoN6ag5WqhOLNwrohTMWPH5roP7pdLIwb3wW6EziWGAZhpVdLWHzxdpqAyc4yKrJ+4uKg2oMV+lgRK9CdwAVsG1WpF8UH84Zt2QkCKmsbiRAtFAllB5OHWQ9fpr81OwlcgBL9kPrwsvDGMQ1zNMV5ZyN1ISlZYZl9/TE1tjuMSFgA1VWO8uef+FAqG47tpFMcm7dyFyIOEzBMlfCadao5qhk39l9P479evwFbiOcRSXKueAAAAABJRU5ErkJggg==&quot;,
		UserAltImg: &quot;data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAIfSURBVDhPpVM9b9NQFD02cSKKFVGKIFFREFsmFiCA1AEQQrQjC0gssLCkC0jJj0iFWFoGdiT4Ay0SEmJBhPI5FBokikL4SCC0Te2UxInt13Mdu1CVrVc6713fe86x/T6w09DCeTNKJ7Q0pzyUmoCmZYOiUhXms8RMcd6vB7UwthhM5fTLWix2L3PmYjKdOwdz9DDFGtr1KuovnqD29JGlfO9Goew9DCV/DaZy2pXE8PD9Y/mbupk6xE6cVWPQhEujHo2+4fXdO76ztnq1UPYfSCcwKB1HWjeMyunCraSZOsDqbkBPsBOTNoMGvkOTLto/f+F56bble262OK/qesiYzIydTJojFLjrgEe4bcIOEdXWYe6LITN2KgmFSRFGBuPpo0eAngX0SexTJHD5LOgLpEZTcgKuUuMijAyy5l7+r0NSb21g5LQI5gGYSy2AjYBLjQyRAUniTnJXsBoayBzm3RBBndwwIoOK3fgNdIREQYCV/0DqLdiNZqCRITKYayx85RfQuUNiZ3k7IgO+vfGeXGpkiAyma++WLHtZhZ8o5H/FEVpos1V7u8TFwLQId8nw+Afa51Oq2vyycmlk9KCW0GXFZeu63IUO8z88Bx5s28SbuUXfddzrxVcobxpI0GTh7H734/dPzQt9d0/CGBpCPB7nbhmwbAPVxR4+PPtsuU7/WuElth/lKORUcsoTE8TgMg0WbJaY4Zu3XKYdBrABARUk+ls4DfQAAAAASUVORK5CYII=&quot;,
		upd(node) {
			var {pref} = node, def = false, user = false, val;
			if (prefs.getPrefType(pref.pref) != prefs.PREF_INVALID) {
				var pn = pref.pref;
				try {val = db[pref.get.name](pn); def = true} catch(ex) {def = false;}
				var user = prefs.prefHasUserValue(pn);
				if (user) try {val = pref.get(pn, undefined);} catch(ex) {}
			}
			if (val == pref.val &amp;&amp; def == pref.def &amp;&amp; user == pref.user) return;
			pref.val = val; pref.def = def; pref.user = user;
			var exists = def || user;

			var ttt = exists ? val : &quot;Этого префа не существует&quot;;
			if (ttt === &quot;&quot;) ttt = &quot;[ empty_string ]&quot;;
			if (pref.ttt) ttt += &quot;\n&quot; + pref.ttt;
			node.tooltipText = ttt;

			if (&quot;userChoice&quot; in pref)
				if (val == pref.userChoice)
					node.style.removeProperty(&quot;color&quot;),
					pref.img || node.setAttribute(&quot;image&quot;, this.UserChoiceImg);
				else
					node.style.setProperty(&quot;color&quot;, &quot;maroon&quot;, &quot;important&quot;),
					pref.img || node.setAttribute(&quot;image&quot;, this.notUserChoiceImg);
			else (&quot;userAlt&quot; in pref)
				if (val == pref.userAlt)
					pref.img || node.setAttribute(&quot;image&quot;, this.UserAltImg);
			user
				? node.style.setProperty(&quot;font-style&quot;, &quot;italic&quot;, &quot;important&quot;)
				: node.style.removeProperty(&quot;font-style&quot;);

			var {lab} = pref;
			if (exists &amp;&amp; !pref.bool) {
				if (val in pref.vals) var sfx = pref.vals[val] || val;
				else var sfx = user ? &quot;Другое&quot; : &quot;По умолчанию&quot;;
				lab += ` &mdash; &quot;${sfx}&quot;`;
			}
			node.setAttribute(&quot;label&quot;, lab);
		},
		createRadios(doc, vals, popup) {
			for(var arr of vals) {
				if (!arr) {
					popup.append(doc.createElementNS(xul_ns, &quot;menuseparator&quot;));
					continue;
				}
				var [val, lab, key, ttt] = arr;
				var menuitem = doc.createElementNS(xul_ns, &quot;menuitem&quot;);
				menuitem.setAttribute(&quot;type&quot;, &quot;radio&quot;);
				menuitem.setAttribute(&quot;closemenu&quot;, &quot;none&quot;);
				menuitem.setAttribute(&quot;label&quot;, popup.parentNode.pref.vals[val] = lab);
				key &amp;&amp; menuitem.setAttribute(&quot;accesskey&quot;, key);
				var tip = menuitem.val = val;
				if (ttt) tip += &quot;\n&quot; + ttt;
				menuitem.tooltipText = tip;
				popup.append(menuitem);
			}
		},
		openPopup(popup) {
			var btn = popup.parentNode;
			if (btn.domParent != btn.parentNode) {
				btn.domParent = btn.parentNode;
				var pos;
				if (btn.matches(&quot;.widget-overflow-list &gt; :scope&quot;))
					pos = &quot;after_start&quot;;
				else var win = btn.ownerGlobal, {width, height, top, bottom, left, right} =
					btn.closest(&quot;toolbar&quot;).getBoundingClientRect(), pos = width &gt; height
						? `${win.innerHeight - bottom &gt; top ? &quot;after&quot; : &quot;before&quot;}_start`
						: `${win.innerWidth - right &gt; left ? &quot;end&quot; : &quot;start&quot;}_before`;
				for(var p of btn.popups) p.setAttribute(&quot;position&quot;, pos);
			}
			popup.openPopup(btn);
		},
		maybeRestart(node, conf) {
			var msgRest = &quot;Перезапустить браузер?&quot;, msgAbort = &quot;Запрос на выход отменен.&quot;;
			if (pv &gt;= 77) {
				var title = node.closest(&quot;toolbarbutton&quot;).label;
				var pp = domWin =&gt; Services.prompt.wrappedJSObject.pickPrompter({
					domWin, modalType: Ci.nsIPrompt.MODAL_TYPE_WINDOW
				});
				var confirm = win =&gt; pp(win).confirm(title, msgRest);
				var alert = win =&gt; pp(win).alert(title, msgAbort);
			} else {
				var confirm = win =&gt; win.confirm(msgRest);
				var alert = win =&gt; win.alert(msgAbort);
			}
			return (this.mayBeRestart = (node, conf) =&gt; {
				var win = node.ownerGlobal;
				if (conf &amp;&amp; !confirm(win)) return;
				if (win.BrowserUtils.restartApplication() === false) alert(win);
				else return true;
			})(node, conf);
		},
		regexpRefresh: /^(?:view-source:)?(?:https?|ftp)/,
		maybeRe(node, fe) {
			var {pref} = node;
			if (&quot;restart&quot; in pref) {
				if (this.maybeRestart(node, pref.restart)) return;
			}
			else this.popupshowing(fe, node.parentNode);
			if (&quot;refresh&quot; in pref) {
				var win = node.ownerGlobal;
				if (this.regexpRefresh.test(win.gBrowser.currentURI.spec)) pref.refresh
					? win.BrowserReloadSkipCache() : win.BrowserReload();
			}
		},
		command(e) {
			var trg = e.target;
			if (trg.btn) return this.openPopup(trg.primaryPopup);

			var menu = trg.closest(&quot;menu&quot;), newVal = trg.val;
			if (newVal != menu.pref.val)
				menu.pref.set(menu.pref.pref, newVal),
				this.maybeRe(menu, true);
		},
		popupshowing(e, trg = e.target) {
			if (trg.id) {
				for(var node of trg.children) {
					if (node.nodeName.endsWith(&quot;r&quot;)) continue;
					this.upd(node);
					!e &amp;&amp; node.open &amp;&amp; this.popupshowing(null, node.querySelector(&quot;menupopup&quot;));
				}
				return;
			}
			var {val} = trg.closest(&quot;menu&quot;).pref;
			var checked = trg.querySelector(&quot;[checked]&quot;);
			if (checked) {
				if (checked.val == val) return;
				else checked.removeAttribute(&quot;checked&quot;);
			}
			for(var node of trg.children) if (&quot;val&quot; in node &amp;&amp; node.val == val) {
				node.setAttribute(&quot;checked&quot;, true);
				break;
			}
		},
		contextmenu(e) {
			var trg = e.target;
			if (trg.btn) {
				if (e.ctrlKey || e.shiftKey) return;
				if (e.detail == 2) return trg.secondaryPopup.hidePopup();
				this.openPopup(trg.secondaryPopup);
			}
			else if (&quot;pref&quot; in trg &amp;&amp; trg.pref.user)
				prefs.clearUserPref(trg.pref.pref),
				this.maybeRe(trg);
			e.preventDefault();
		}
	};
});
Отредактировано _zt (14-08-2020 21:39:11)
Отсутствует
Может подойдёт в контекстном меню не на странице, а на кнопке?
Это не кнопка,это комбайн какой то . Нет нужно попроще, только из контекстного меню. Желательно по возможности ,реанимировать старый код. Просто у некоторых, руки работаю быстрее головы, жмут на всё подряд, а до контекстного меню не дотягиваются, тонкой тайм-аут. Может у кого получится починить код.
Отсутствует
код вставил инициализацию - ничего не сохраняются вообще
Ну не знаю, может CB какой-то несоответствующий.
Тогда попробуй изменить вкладку Код.
(f => { var url = "http://site.ru"; var br = (gBrowser.selectedTab = gBrowser.addTab(url)).linkedBrowser; br.addEventListener("pageshow", () => setTimeout( () => gBrowser.selectedBrowser == br && br.currentURI.spec == url && f() , 0), {once: true}); })(() => { //====================== // Здесь расположить // весь код вкладки Код. //====================== });
Иконку UserAlt прописал и она работает, но вы проверьте за мной
Вот эта строка:
else ("userAlt" in pref)
не делает ничего (значимого), просто по else вычисляет есть ли у этого объекта такое свойство,
вычисляет в пустоту. Ну а дальше там просто установка "image", если совпадение с userAlt.
Всё бы ничего, но бывает, перед этим, чуть выше по коду, ставится notUserChoiceImg,
и сразу UserAltImg. То есть двойная установка, а это, наверно, нехорошо. Переделал.
И ещё, немного похозяйничал. Заметил, что в каждом(!) pref'е, в конце,
продублировано имя настройки, чтобы оно отображалось в тултипе.
Показалось похожим на концепцию, поэтому поудалял их все, а отображение
имени настройки в тултипе добавил в код.
Ну и оставшиеся пустые "" под accesskey тоже удалил. Сама возможность
дописывать accesskey и дополнительный текст к тултипу, разумеется, никуда не делась.
Если рассудил неверно, тогда упс.
Вобщем вот. Но выложенный код далеко не убирай, в том смысле, что если
я там чего-то испортил, и надо будет откатить, и понадеяться на тот, что на форуме,
то придётся реплейсить " & > — которые там прописались, как-то.
data:text/plain;charset=utf-8;base64,// Быстрое переключение параметров about:config
(async (name, id, func) => {
	if (name == "Object") return CustomizableUI.createWidget(func());
	var win = name == "Window", g = Components.utils.import("resource://gre/modules/Services.jsm", {});
	if (g[id]) {if (win) return;} else g[id] = func();
	if (win) return CustomizableUI.createWidget(g[id]);
	addDestructor(r => r[5] == "e" && delete g[id]);
	g[id].onCreated(this);
})(this.constructor.name, "QuickToggleAboutConfigSettings", () => {

	var {prefs} = Services, db = prefs.getDefaultBranch("");
	var pv = parseInt(Services.appinfo.platformVersion);
	var xul_ns = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";

//=====================================================================================

	// refresh:
	//	false - reload current tab
	//	true - reload current tab skip cache
	//
	// restart:
	//	false - restart browser
	//	true - restart browser with confirm

	var primary = [{

			pref: ["network.proxy.type", "Настройки прокси"],
			userChoice: 5, userAlt: 1, refresh: true,
			values: [
				[0, "Не проксировать", "0"], [5, "Системные (из IE)", "5"], [2, "Авто (pacfile)", "2"],
				[1, "Прописанные", "1"], [4, "Автоопределение", "4"]
	]},
			null,
	{
			pref: ["permissions.default.image", "Загружать графику"],
			userChoice: 1, refresh: true,
			values: [[1, "Да"], [3, "С сайта"], [2, "Нет"]]
	},{
			pref: ["browser.display.use_document_fonts", "Загружать web-шрифты"],
			userChoice: 1, refresh: true,
			values: [[1, "Да"], [0, "Нет"]]
	},{
			pref: ["javascript.enabled", "Выполнять скрипты Java"],
			userChoice: true, refresh: true,
			values: [[true, "Да"], [false, "Нет"]]
	},{
			pref: ["media.autoplay.default", "Автозапуск медиа"],
			userChoice: 5, refresh: true,
			values: [
				[5, "Блокировать все", "5"],
				[1, "Блокировать не приглушенное", "1"],
				[0, "Разрешить все", "0"]
	]},{
			pref: ["media.autoplay.blocking_policy", "Автозапуск (политика)"],
			userChoice: 1, userAlt: 2, refresh: true,
			values: [
				[1, "Временная", "1"],
				[2, "По действию", "2"],
				[0, "Постоянная", "0"]
	]},{
			pref: ["network.cookie.cookieBehavior", "Cookies"],
			userChoice: 1, userAlt: 3, refresh: false,
			values: [
				[1, "Не принимать сторонние"], [3, "Не принимать с не посещенных"], [4, "Не принимать от трекеров"],
				[2, "Не принимать со всех"], [0, "Принимать со всех"]
	]},
			null,
	{
			pref: ["dom.storage.enabled", "Локальное хранилище"],
			userChoice: true
	}
];

//=====================================================================================

	var secondary = [{

			pref: ["dom.serviceWorkers.enabled", "Видео dom.serviceWorkers"],
			userChoice: false
	},{
			pref: ["dom.enable_performance", "Статус загрузки страницы"],
			userChoice: false
	},
			null,
	{
			pref: ["browser.cache.memory.enable", "Кэш в оперативной памяти"],
			userChoice: true
	},
			null,
	{
			pref: ["intl.accept_languages", "Язык для веб-страниц"],
			userChoice: "en-US, en",
			values: [["en-US, en", "en-US, en"], ["en-US, en, ru-RU, ru", "en-US, en, ru-RU, ru"]]
	},{
			pref: ["browser.display.document_color_use", "Использовать цвета сайтов"],
			userChoice: 0,
			values: [[0, "Авто", "0"], [1, "Всегда", "1"], [2, "Никогда", "2"]]
	},
			null,
	{
			pref: ["network.http.sendRefererHeader", "Referer - для чего"],
			userChoice: 1,
			values: [[0, "Ни для чего", "0"], [1, "Только ссылки", "1"], [2, "Ссылки и изобр.", "2"]]
	},{
			pref: ["network.http.referer.trimmingPolicy", "Referer - что"],
			userChoice: 0,
			values: [[0, "Полный URL", "0"], [1, "scheme+host+port+path", "1"], [2, "scheme+host+port", "2"]]
	},{
			pref: ["network.http.referer.XOriginPolicy", "RefererXO - когда"],
			userChoice: 0,
			values: [[0, "В любом случае", "0"], [1, "При совп. баз. домена", "1"], [2, "При совпадении адреса", "2"]]
	},{
			pref: ["network.http.referer.XOriginTrimmingPolicy", "RefererXO - что"],
			userChoice: 0,
			values: [[0, "Полный URL", "0"], [1, "scheme+host+port+path", "1"], [2, "scheme+host+port", "2"]]
	},{
			pref: ["network.http.referer.spoofSource", "Referer - корень сайта"],
			userChoice: false
	},
			null,
	{
			pref: ["media.peerconnection.enabled", "WebRTC утечка IP"],
			userChoice: false
	}
	];

	return {
		label: "Quick toggle",
		id: "QuickToggleAboutConfigSettings",
		localized: false,
		image: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAFPSURBVDhPnVPNSsNAEJ5tamnBk+A7BMEcikohp/gi9i18nRafQoT00kK16tnn8CJo4vp9m5m41eSgH0zmd3d+diJ/gfdedvl0QW7k1PcLwem+3TjowZagK+Nn6yc3gNCJ+LCCh9LzzTMdqeoNbkQSFZmpJLGCW5Gx2hZaQahMbZ52VuDSfHpnAVJXOb4FKzianbzCXiLrHLYl5C3t5NTRwtxaKJxUTZ/JcOTr6q2Rxwf0aVb2/hLs4KqL3Isckz/mmSdRRksjyg+z7IM6LzDCBfuvAEOJmIKBOqAWmo1YodxLlQNsyO0rsIV/AZM+JGc2a4GgzhZ2p9mFmrpboCMqNYBDdMlwompoTWN6F2nFT5hBXb0PEpmo/Gk+oH+RrqNZINPeIlnZViV1jQuLFJQuWGB0gJdwgShvqdPXZv6J+F/QS7oXqQ9xBTFZ5oa8fAGs9fed5YhPSwAAAABJRU5ErkJggg==",
		onCreated(btn) {
			btn.setAttribute("image", this.image);
			var doc = btn.ownerDocument;

			btn.btn = true;
			btn.domParent = null;
			btn.popups = new btn.ownerGlobal.Array();
			this.createPopup(doc, btn, "primary", primary);
			this.createPopup(doc, btn, "secondary", secondary);

			btn.linkedObject = this;
			for(var type of ["command", "contextmenu"])
				btn.setAttribute("on" + type, `linkedObject.${type}(event)`);
		},
		createPopup(doc, btn, name, data) {
			var popup = doc.createElementNS(xul_ns, "menupopup");
			var prop = name + "Popup";
			btn.popups.push(btn[prop] = popup);
			popup.id = this.id + "-" + prop;
			for (var type of ["popupshowing", "click"])
				popup.setAttribute("on" + type, `parentNode.linkedObject.${type}(event)`);
			for(var obj of data) popup.append(this.createElement(doc, obj));
			btn.append(popup);
		},
		createElement(doc, obj) {
			if (!obj) return doc.createElementNS(xul_ns, "menuseparator");
			var pref = doc.ownerGlobal.Object.create(null), node, img, bool;
			for(var [key, val] of Object.entries(obj)) {
				if (key == "pref") {
					var [apref, lab, akey, ttt] = val;
					pref.pref = apref; pref.lab = lab || apref;
					if (ttt) pref.ttt = ttt;
				}
				else if (key == "image") img = val, pref.img = true;
				else if (key != "values") pref[key] = val;
				else pref.hasVals = true;
			}
			var type = prefs.getPrefType(pref.pref), str, bool;
			var pint = type == prefs.PREF_INT, inv = type == prefs.PREF_INVALID;
			if (inv && obj.values || pint || type == prefs.PREF_STRING)
				str = pint || (inv && typeof obj.values[0][0] == "number") ? "Int" : "String";
			else
				str = "Bool", bool = true;
			pref.get = prefs[`get${str}Pref`];
			pref.set = prefs[`set${str}Pref`];

			node = doc.createElementNS(xul_ns, "menu");
			node.className = "menu-iconic";
			node.setAttribute("closemenu", "none");
			img && node.setAttribute("image", img);
			akey && node.setAttribute("accesskey", akey);
			(node.pref = pref).vals = doc.ownerGlobal.Object.create(null);
			this.createRadios(doc,
				bool && !pref.hasVals ? [[true, "true"], [false, "false"]] : obj.values,
				node.appendChild(doc.createElementNS(xul_ns, "menupopup"))
			);
			if ("userChoice" in obj) pref.noAlt = !("userAlt" in obj);
			return node;
		},
		UserChoiceImg: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACRUlEQVR42qWTS2gTQRjH/zPJZo2BDfgkB62vYg5CjQdvRUE8WB8HKyahYotIQQOeFKGCF7EgvQlBqOCjWFKlevBRDyL4uHloLXhYqWJroaEqlETSptndGb+ZPDH0ojN8O8vA98vv+/Itw38u1nSRYRE6UhKyg4FF1R292/Q+xhhLi4TIrgjgIzwe4MbgmR2nrZMtxxEN63zYORuPZp7gzpehvCu9XjfuPmwCUHIiEtw4fK89zVvD2+FHAD56quXBpV3CVO4ret6nxHzxZxdBRuqADCKmz7RHD9y2tlmbYbBVfwE8DXBkEd/y39H5+mze8ZyoSIpsFXC9u/VE3/ldpyg5AAMEoJMTQi2hANKBgyKdy0h/eoChqcf9MimvVAHjd/fdiG0NRwgQpDAp1agBpDZwdHKJLKZzc+h5e3kCSexh5c5j8fmhW0HGpNZXFkqfsQpAKgcFKGmAlAxHXp5bkkmsrhosjh4cCIJ5pG5qfQXgtMslqO1qgEMWTPrR+eriEhoA4wN7L8TWh0KVXzfgY2RQAUjVBenWLH4UCrj04Wa9BNXEo5va+w63xCiJ6WRVP2MVgNQIgjgEA17MTODZ7Lt+AtSaGDG4377a1m2FAkKrc11/dUwIID1dSKHEcW3yft4RbpR6kK1PYgaJtaY1nNp5jIeMsjKNbs1AWRUcjvTnp2Kh9LtLJGTDINUhcTIZ3L+hzdq9ZgvWmSF9/Wu5gI8L03gzP5l3hddL/3/zKDdA9MdE0UERrdzaFGMUaap75Y/pX9Yfap76EQYaCeEAAAAASUVORK5CYII=",
		notUserChoiceImg: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACNUlEQVR42qXTT2jTUBwH8G9e0ta2ks5ND0Ww/qX1oGyevEwQnOBA6EHXwLwoUh3eRGRs501RPE6xgp6KrexQECZY8DBPXmbRS4Z/K0gRtmmDa2yb5PlLX7p2Gwpi4EfCy8sn3/dPwn9e0saGnCRF6XYFnA9DkhKtRs51ep6jmtEcp/JHIM9Yivl9mf0XNDV2Ngn1YLzVxdB1lJ8U8P5hzuCWnR6xrPwmIM8kLRjdkT326BaLHNgLKAFA9ouXdhOw6qi++4j589cd89vSaMqyc2sAPUXlgF8fyt9WI3t2Av4w4NsCMJ8AHAKav6hqqH76iuLINcNuNhOawyttYCp+7tRE/6XT9GFQAG4CpniATQkIaNQIMVG6/xSL2WfTGueTbWDh5L2rA9t29QGBrQQEPUDuAuoEmFSr+P5lGc8v33mtAUfaQO3M4/GgDOoYCIkUin8D0BDDoBQ2ZMxqN0wCQh0gMxaUnYb4uzt+dwIZ8wBHTGQbYAHMpu+uAxaGxpMDvSr3AHcFaAIlD+BtoN6ag5WqhOLNwrohTMWPH5roP7pdLIwb3wW6EziWGAZhpVdLWHzxdpqAyc4yKrJ+4uKg2oMV+lgRK9CdwAVsG1WpF8UH84Zt2QkCKmsbiRAtFAllB5OHWQ9fpr81OwlcgBL9kPrwsvDGMQ1zNMV5ZyN1ISlZYZl9/TE1tjuMSFgA1VWO8uef+FAqG47tpFMcm7dyFyIOEzBMlfCadao5qhk39l9P479evwFbiOcRSXKueAAAAABJRU5ErkJggg==",
		UserAltImg: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAIfSURBVDhPpVM9b9NQFD02cSKKFVGKIFFREFsmFiCA1AEQQrQjC0gssLCkC0jJj0iFWFoGdiT4Ay0SEmJBhPI5FBokikL4SCC0Te2UxInt13Mdu1CVrVc6713fe86x/T6w09DCeTNKJ7Q0pzyUmoCmZYOiUhXms8RMcd6vB7UwthhM5fTLWix2L3PmYjKdOwdz9DDFGtr1KuovnqD29JGlfO9Goew9DCV/DaZy2pXE8PD9Y/mbupk6xE6cVWPQhEujHo2+4fXdO76ztnq1UPYfSCcwKB1HWjeMyunCraSZOsDqbkBPsBOTNoMGvkOTLto/f+F56bble262OK/qesiYzIydTJojFLjrgEe4bcIOEdXWYe6LITN2KgmFSRFGBuPpo0eAngX0SexTJHD5LOgLpEZTcgKuUuMijAyy5l7+r0NSb21g5LQI5gGYSy2AjYBLjQyRAUniTnJXsBoayBzm3RBBndwwIoOK3fgNdIREQYCV/0DqLdiNZqCRITKYayx85RfQuUNiZ3k7IgO+vfGeXGpkiAyma++WLHtZhZ8o5H/FEVpos1V7u8TFwLQId8nw+Afa51Oq2vyycmlk9KCW0GXFZeu63IUO8z88Bx5s28SbuUXfddzrxVcobxpI0GTh7H734/dPzQt9d0/CGBpCPB7nbhmwbAPVxR4+PPtsuU7/WuElth/lKORUcsoTE8TgMg0WbJaY4Zu3XKYdBrABARUk+ls4DfQAAAAASUVORK5CYII=",
		upd(node) {
			var {pref} = node, def = false, user = false, val;
			if (prefs.getPrefType(pref.pref) != prefs.PREF_INVALID) {
				var pn = pref.pref;
				try {val = pref.defVal = db[pref.get.name](pn); def = true}
				catch(ex) {def = false;}
				var user = prefs.prefHasUserValue(pn);
				if (user) try {val = pref.get(pn, undefined);} catch(ex) {}
			}
			if (val == pref.val && def == pref.def && user == pref.user) return;
			pref.val = val; pref.def = def; pref.user = user;
			var exists = def || user;

			var ttt = exists ? val : "Этого префа не существует";
			if (ttt === "") ttt = "[ empty_string ]";
			ttt += "\n" + pref.pref;
			if (pref.ttt) ttt += "\n" + pref.ttt;
			node.tooltipText = ttt;

			var img, alt = "userAlt" in pref && val == pref.userAlt;
			if (alt) img = this.UserAltImg;
			if ("userChoice" in pref)
				if (val == pref.userChoice)
					node.style.removeProperty("color"),
					img = this.UserChoiceImg;
				else {
					node.style.setProperty("color", "maroon", "important");
					if (!alt) img = this.notUserChoiceImg;
				}
			if (!pref.img) img
				? node.setAttribute("image", img)
				: node.removeAttribute("image");
			user
				? node.style.setProperty("font-style", "italic", "important")
				: node.style.removeProperty("font-style");

			var {lab} = pref;
			if (exists && pref.hasVals) {
				if (val in pref.vals) var sfx = pref.vals[val] || val;
				else var sfx = user ? "Другое" : "По умолчанию";
				lab += ` — "${sfx}"`;
			}
			node.setAttribute("label", lab);
		},
		createRadios(doc, vals, popup) {
			for(var arr of vals) {
				if (!arr) {
					popup.append(doc.createElementNS(xul_ns, "menuseparator"));
					continue;
				}
				var [val, lab, key, ttt] = arr;
				var menuitem = doc.createElementNS(xul_ns, "menuitem");
				menuitem.setAttribute("type", "radio");
				menuitem.setAttribute("closemenu", "none");
				menuitem.style.setProperty("font-style", "italic", "important"),
				menuitem.setAttribute("label", popup.parentNode.pref.vals[val] = lab);
				key && menuitem.setAttribute("accesskey", key);
				var tip = menuitem.val = val;
				if (ttt) tip += "\n" + ttt;
				menuitem.tooltipText = tip;
				popup.append(menuitem);
			}
		},
		openPopup(popup) {
			var btn = popup.parentNode;
			if (btn.domParent != btn.parentNode) {
				btn.domParent = btn.parentNode;
				var pos;
				if (btn.matches(".widget-overflow-list > :scope"))
					pos = "after_start";
				else var win = btn.ownerGlobal, {width, height, top, bottom, left, right} =
					btn.closest("toolbar").getBoundingClientRect(), pos = width > height
						? `${win.innerHeight - bottom > top ? "after" : "before"}_start`
						: `${win.innerWidth - right > left ? "end" : "start"}_before`;
				for(var p of btn.popups) p.setAttribute("position", pos);
			}
			popup.openPopup(btn);
		},
		maybeRestart(node, conf) {
			var msgRest = "Перезапустить браузер?", msgAbort = "Запрос на выход отменен.";
			if (pv >= 77) {
				var title = node.closest("toolbarbutton").label;
				var pp = domWin => Services.prompt.wrappedJSObject.pickPrompter({
					domWin, modalType: Ci.nsIPrompt.MODAL_TYPE_WINDOW
				});
				var confirm = win => pp(win).confirm(title, msgRest);
				var alert = win => pp(win).alert(title, msgAbort);
			} else {
				var confirm = win => win.confirm(msgRest);
				var alert = win => win.alert(msgAbort);
			}
			return (this.mayBeRestart = (node, conf) => {
				var win = node.ownerGlobal;
				if (conf && !confirm(win)) return;
				if (win.BrowserUtils.restartApplication() === false) alert(win);
				else return true;
			})(node, conf);
		},
		regexpRefresh: /^(?:view-source:)?(?:https?|ftp)/,
		maybeRe(node, fe) {
			var {pref} = node;
			if ("restart" in pref) {
				if (this.maybeRestart(node, pref.restart)) return;
			}
			else this.popupshowing(fe, node.parentNode);
			if ("refresh" in pref) {
				var win = node.ownerGlobal;
				if (this.regexpRefresh.test(win.gBrowser.currentURI.spec)) pref.refresh
					? win.BrowserReloadSkipCache() : win.BrowserReload();
			}
		},
		command(e) {
			var trg = e.target;
			if (trg.btn) return this.openPopup(trg.primaryPopup);

			var menu = trg.closest("menu"), newVal = trg.val;
			if (newVal != menu.pref.val)
				menu.pref.set(menu.pref.pref, newVal),
				this.maybeRe(menu, true);
		},
		popupshowing(e, trg = e.target) {
			if (trg.id) {
				for(var node of trg.children) {
					if (node.nodeName.endsWith("r")) continue;
					this.upd(node);
					!e && node.open && this.popupshowing(null, node.querySelector("menupopup"));
				}
				return;
			}
			var {pref} = trg.closest("menu"), findChecked = true;

			var findDef = "defVal" in pref;
			var checked = trg.querySelector("[checked]");
			if (checked) {
				if (checked.val == pref.val) {
					if (findDef) findChecked = false;
					else return;
				}
				else checked.removeAttribute("checked");
			}
			if (findDef) {
				var def = trg.querySelector("menuitem:not([style*=font-style]");
				if (def)
					if (def.val == pref.defVal) {
						if (findChecked) findDef = false;
						else return;
					}
					else def.style.setProperty("font-style", "italic", "important");
			}
			for(var node of trg.children) if ("val" in node) {
				if (findChecked && node.val == pref.val) {
					node.setAttribute("checked", true);
					if (findDef) findChecked = false;
					else break;
				}
				if (findDef && node.val == pref.defVal) {
					node.style.removeProperty("font-style");
					if (findChecked) findDef = false;
					else break;
				}
			}
		},
		contextmenu(e) {
			var trg = e.target;
			if (trg.btn) {
				if (e.ctrlKey || e.shiftKey) return;
				if (e.detail == 2) return trg.secondaryPopup.hidePopup();
				this.openPopup(trg.secondaryPopup);
			}
			else if ("pref" in trg && trg.pref.user)
				prefs.clearUserPref(trg.pref.pref),
				this.maybeRe(trg);
			e.preventDefault();
		},
		click(e) {
			if (e.button) return;
			var trg = e.target, {pref} = trg;
			if (!pref || !("noAlt" in pref)) return;

			if (pref.val == pref.userChoice)
				if (pref.noAlt) return;
				else  pref.set(pref.pref, pref.userAlt);
			else
				pref.set(pref.pref, pref.userChoice);
			this.maybeRe(trg);
		}
	};
});
Отсутствует
Dumby ещё просьба по QuickSettings - сделать константу в коде, которая управляет скрытием меню после клика.
Например const HideMenu = 1; // автоскрывать после клика…
Может ещё константу для отключения default иконок сделаете ?
Чтобы иконки были как раньше, только на пунктах меню, в которых прописано userChoice: значение по-умолчанию.
Отредактировано Dobrov (15-08-2020 15:33:46)
Отсутствует
Dumby
Сделал так: если userChoice отсутствует, то клик не делает ничего,
иначе — переключает на userChoice, а если текущее значение
уже совпадает с userChoice, тогда переключает на userAlt, если таковой имеется
Все верно, я почти так и писал. Переключение на userChoice из любого отличного значения это уже вами добавленный бонус.
Заметил, что в каждом(!) pref'е, в конце,
продублировано имя настройки, чтобы оно отображалось в тултипе.
Показалось похожим на концепцию, поэтому поудалял их все, а отображение
имени настройки в тултипе добавил в код.
Тоже верно. Тултип никому не мешает, но зато всегда можно увидеть реальный преф скрытый под заголовком.
Ну и оставшиеся пустые "" под accesskey тоже удалил.
Я ими не пользуюсь. Возможно они тут работают, но я уже привык, что, как правило, они полностью неработоспособны, то ли с неанглийскими, то ли с измененными раскладками.
" & > — которые там прописались, как-то
Из-за перевода? Как их увидеть? Если об этом речь.
node.style.removeProperty("color"), и node.style.setProperty("color", "maroon", "important");
Отредактировано _zt (15-08-2020 17:25:31)
Отсутствует
Dobrov
Просто удалите иконки, приведите соответствующие строки к такому виду:
Отредактировано _zt (15-08-2020 18:27:24)
Отсутствует
Добрый день. Посмотрите пожалуйста код для FF71 "Удалить скролл" . Работал нормально , но после моих переделок, перестал. Какой код конфликтует не нашел. Возможно изменить этот или добавить код , устанавливающий приоритет над всеми другими кодами.
/* Scrollbar hidden / hide scrollbars totally */ //@-moz-document url("https://yandex.ru/"), url("https://www.yandex.ru/") { /* Disable scrollbar Firefox только yandex.ru*/ @-moz-document url-prefix() { /* Disable scrollbar Firefox */ html{ scrollbar-width: none; } } body { margin: 0; /* remove default margin */ scrollbar-width: none; /* Also needed to disable scrollbar Firefox */ overflow-y: scroll; }
Вопрос снимается ,разобрался.
Отредактировано Duche (16-08-2020 13:22:43)
Отсутствует
_zt
Из-за перевода? Как их увидеть? Если об этом речь.
Не-не, увидеть очень просто: если открыть во вкладке base64 адрес с кодом
из этого поста, и приглядеться к содержимому, то видно, что оно словно
пропущено через подготовщик для XML. Самое заметное — " вместо кавычек.
Можно удалить, ничего не поломав?
Думаю да, на эти строки ничего не завязано, закомментировал.
Согласен с Dobrov, быстрое переключение "скрывать"/"не скрывать" меню после клика, тут не помешало бы. Для тестов удобнее как сейчас, а при постоянном использовании нескрываемое меню будет отвлекать.
А интересно, нельзя ли эту опцию во вторичное меню вывести? ) Например, добавлением этой настройки в about:config и последующем ее чтением.
OK, попробую добавить во вторичное меню.
Сам этот пункт, при переключении, закрытие меню не вызывает (фишка такая).
Предусмотрено не закрывать, если зажат Ctrl.
... return { label: "Quick toggle", id: "QuickToggleAboutConfigSettings", localized: false, image: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAFPSURBVDhPnVPNSsNAEJ5tamnBk+A7BMEcikohp/gi9i18nRafQoT00kK16tnn8CJo4vp9m5m41eSgH0zmd3d+diJ/gfdedvl0QW7k1PcLwem+3TjowZagK+Nn6yc3gNCJ+LCCh9LzzTMdqeoNbkQSFZmpJLGCW5Gx2hZaQahMbZ52VuDSfHpnAVJXOb4FKzianbzCXiLrHLYl5C3t5NTRwtxaKJxUTZ/JcOTr6q2Rxwf0aVb2/hLs4KqL3Isckz/mmSdRRksjyg+z7IM6LzDCBfuvAEOJmIKBOqAWmo1YodxLlQNsyO0rsIV/AZM+JGc2a4GgzhZ2p9mFmrpboCMqNYBDdMlwompoTWN6F2nFT5hBXb0PEpmo/Gk+oH+RrqNZINPeIlnZViV1jQuLFJQuWGB0gJdwgShvqdPXZv6J+F/QS7oXqQ9xBTFZ5oa8fAGs9fed5YhPSwAAAABJRU5ErkJggg==", onCreated(btn) { btn.setAttribute("image", this.image); var doc = btn.ownerDocument; btn.btn = true; btn.domParent = null; btn.popups = new btn.ownerGlobal.Array(); this.createPopup(doc, btn, "primary", primary); this.createPopup(doc, btn, "secondary", secondary); this.createCloseMenusOption(doc, btn); btn.linkedObject = this; for(var type of ["command", "contextmenu"]) btn.setAttribute("on" + type, `linkedObject.${type}(event)`); }, createPopup(doc, btn, name, data) { var popup = doc.createElementNS(xul_ns, "menupopup"); var prop = name + "Popup"; btn.popups.push(btn[prop] = popup); popup.id = this.id + "-" + prop; for (var type of ["popupshowing", "click"]) popup.setAttribute("on" + type, `parentNode.linkedObject.${type}(event)`); for(var obj of data) popup.append(this.createElement(doc, obj)); btn.append(popup); }, map: {b: "Bool", n: "Int", s: "String"}, createElement(doc, obj) { if (!obj) return doc.createElementNS(xul_ns, "menuseparator"); var pref = doc.ownerGlobal.Object.create(null), node, img, bool; for(var [key, val] of Object.entries(obj)) { if (key == "pref") { var [apref, lab, akey, ttt] = val; pref.pref = apref; pref.lab = lab || apref; if (ttt) pref.ttt = ttt; } else if (key == "image") img = val, pref.img = true; else if (key != "values") pref[key] = val; else pref.hasVals = true; } var type = prefs.getPrefType(pref.pref); var str = this.map[type == prefs.PREF_INVALID ? obj.values ? (typeof obj.values[0][0])[0] : "b" : type == prefs.PREF_BOOL ? "b" : type == prefs.PREF_INT ? "n" : "s" ]; pref.get = prefs[`get${str}Pref`]; pref.set = prefs[`set${str}Pref`]; node = doc.createElementNS(xul_ns, "menu"); node.className = "menu-iconic"; node.setAttribute("closemenu", "none"); img && node.setAttribute("image", img); akey && node.setAttribute("accesskey", akey); (node.pref = pref).vals = doc.ownerGlobal.Object.create(null); this.createRadios(doc, str.startsWith("B") && !pref.hasVals ? [[true, "true"], [false, "false"]] : obj.values, node.appendChild(doc.createElementNS(xul_ns, "menupopup")) ); if ("userChoice" in obj) pref.noAlt = !("userAlt" in obj); return node; }, createCloseMenusOption(doc, btn) { var pn = this.closePref = "QuickToggleAboutConfigSettings.closeMenus"; var data = [null, { pref: [pn, "Закрывать меню этой кнопки"], values: [[true, "Да"], [false, "Нет"]] }]; var setCloseMenus = e => { e.stopPropagation(); var trg = e.target, {pref, val} = trg, updPopup = true, clear; switch(e.type) { case "command": pref = (trg = trg.closest("menu")).pref; updPopup = false; break; case "click": if (e.button) return; break; case "contextmenu": e.preventDefault(); clear = pref; } if (!pref) return; if (clear) prefs.clearUserPref(pn); else if (!updPopup && val === pref.val) return; else pref.set(pn, val !== undefined ? val : !pref.val); this.upd(trg); updPopup && this.popupshowing(null, trg.querySelector("menupopup")); } (this.createCloseMenusOption = (doc, btn) => { for(var obj of data) btn.secondaryPopup.append(this.createElement(doc, obj)); var m = btn.secondaryPopup.lastChild; m.style.cssText = "fill: lightblue !important; list-style-image: url(chrome://browser/skin/menu.svg) !important;"; m.setAttribute("oncommand", "setCloseMenus(event)"); m.onclick = m.oncontextmenu = m.setCloseMenus = setCloseMenus; })(doc, btn); }, UserChoiceImg: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACRUlEQVR42qWTS2gTQRjH/zPJZo2BDfgkB62vYg5CjQdvRUE8WB8HKyahYotIQQOeFKGCF7EgvQlBqOCjWFKlevBRDyL4uHloLXhYqWJroaEqlETSptndGb+ZPDH0ojN8O8vA98vv+/Itw38u1nSRYRE6UhKyg4FF1R292/Q+xhhLi4TIrgjgIzwe4MbgmR2nrZMtxxEN63zYORuPZp7gzpehvCu9XjfuPmwCUHIiEtw4fK89zVvD2+FHAD56quXBpV3CVO4ret6nxHzxZxdBRuqADCKmz7RHD9y2tlmbYbBVfwE8DXBkEd/y39H5+mze8ZyoSIpsFXC9u/VE3/ldpyg5AAMEoJMTQi2hANKBgyKdy0h/eoChqcf9MimvVAHjd/fdiG0NRwgQpDAp1agBpDZwdHKJLKZzc+h5e3kCSexh5c5j8fmhW0HGpNZXFkqfsQpAKgcFKGmAlAxHXp5bkkmsrhosjh4cCIJ5pG5qfQXgtMslqO1qgEMWTPrR+eriEhoA4wN7L8TWh0KVXzfgY2RQAUjVBenWLH4UCrj04Wa9BNXEo5va+w63xCiJ6WRVP2MVgNQIgjgEA17MTODZ7Lt+AtSaGDG4377a1m2FAkKrc11/dUwIID1dSKHEcW3yft4RbpR6kK1PYgaJtaY1nNp5jIeMsjKNbs1AWRUcjvTnp2Kh9LtLJGTDINUhcTIZ3L+hzdq9ZgvWmSF9/Wu5gI8L03gzP5l3hddL/3/zKDdA9MdE0UERrdzaFGMUaap75Y/pX9Yfap76EQYaCeEAAAAASUVORK5CYII=", notUserChoiceImg: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACNUlEQVR42qXTT2jTUBwH8G9e0ta2ks5ND0Ww/qX1oGyevEwQnOBA6EHXwLwoUh3eRGRs501RPE6xgp6KrexQECZY8DBPXmbRS4Z/K0gRtmmDa2yb5PlLX7p2Gwpi4EfCy8sn3/dPwn9e0saGnCRF6XYFnA9DkhKtRs51ep6jmtEcp/JHIM9Yivl9mf0XNDV2Ngn1YLzVxdB1lJ8U8P5hzuCWnR6xrPwmIM8kLRjdkT326BaLHNgLKAFA9ouXdhOw6qi++4j589cd89vSaMqyc2sAPUXlgF8fyt9WI3t2Av4w4NsCMJ8AHAKav6hqqH76iuLINcNuNhOawyttYCp+7tRE/6XT9GFQAG4CpniATQkIaNQIMVG6/xSL2WfTGueTbWDh5L2rA9t29QGBrQQEPUDuAuoEmFSr+P5lGc8v33mtAUfaQO3M4/GgDOoYCIkUin8D0BDDoBQ2ZMxqN0wCQh0gMxaUnYb4uzt+dwIZ8wBHTGQbYAHMpu+uAxaGxpMDvSr3AHcFaAIlD+BtoN6ag5WqhOLNwrohTMWPH5roP7pdLIwb3wW6EziWGAZhpVdLWHzxdpqAyc4yKrJ+4uKg2oMV+lgRK9CdwAVsG1WpF8UH84Zt2QkCKmsbiRAtFAllB5OHWQ9fpr81OwlcgBL9kPrwsvDGMQ1zNMV5ZyN1ISlZYZl9/TE1tjuMSFgA1VWO8uef+FAqG47tpFMcm7dyFyIOEzBMlfCadao5qhk39l9P479evwFbiOcRSXKueAAAAABJRU5ErkJggg==", UserAltImg: "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAIfSURBVDhPpVM9b9NQFD02cSKKFVGKIFFREFsmFiCA1AEQQrQjC0gssLCkC0jJj0iFWFoGdiT4Ay0SEmJBhPI5FBokikL4SCC0Te2UxInt13Mdu1CVrVc6713fe86x/T6w09DCeTNKJ7Q0pzyUmoCmZYOiUhXms8RMcd6vB7UwthhM5fTLWix2L3PmYjKdOwdz9DDFGtr1KuovnqD29JGlfO9Goew9DCV/DaZy2pXE8PD9Y/mbupk6xE6cVWPQhEujHo2+4fXdO76ztnq1UPYfSCcwKB1HWjeMyunCraSZOsDqbkBPsBOTNoMGvkOTLto/f+F56bble262OK/qesiYzIydTJojFLjrgEe4bcIOEdXWYe6LITN2KgmFSRFGBuPpo0eAngX0SexTJHD5LOgLpEZTcgKuUuMijAyy5l7+r0NSb21g5LQI5gGYSy2AjYBLjQyRAUniTnJXsBoayBzm3RBBndwwIoOK3fgNdIREQYCV/0DqLdiNZqCRITKYayx85RfQuUNiZ3k7IgO+vfGeXGpkiAyma++WLHtZhZ8o5H/FEVpos1V7u8TFwLQId8nw+Afa51Oq2vyycmlk9KCW0GXFZeu63IUO8z88Bx5s28SbuUXfddzrxVcobxpI0GTh7H734/dPzQt9d0/CGBpCPB7nbhmwbAPVxR4+PPtsuU7/WuElth/lKORUcsoTE8TgMg0WbJaY4Zu3XKYdBrABARUk+ls4DfQAAAAASUVORK5CYII=", upd(node) { var {pref} = node, def = false, user = false, val; if (prefs.getPrefType(pref.pref) != prefs.PREF_INVALID) { var pn = pref.pref; try {val = pref.defVal = db[pref.get.name](pn); def = true} catch(ex) {def = false;} var user = prefs.prefHasUserValue(pn); if (user) try {val = pref.get(pn, undefined);} catch(ex) {} } if (val == pref.val && def == pref.def && user == pref.user) return; pref.val = val; pref.def = def; pref.user = user; var exists = def || user; var ttt = exists ? val : "Этого префа не существует"; if (ttt === "") ttt = "[ empty_string ]"; ttt += "\n" + pref.pref; if (pref.ttt) ttt += "\n" + pref.ttt; node.tooltipText = ttt; var img, alt = "userAlt" in pref && val == pref.userAlt; if (alt) img = this.UserAltImg; if ("userChoice" in pref) if (val == pref.userChoice) //node.style.removeProperty("color"), img = this.UserChoiceImg; else { //node.style.setProperty("color", "maroon", "important"); if (!alt) img = this.notUserChoiceImg; } if (!pref.img) img ? node.setAttribute("image", img) : node.removeAttribute("image"); user ? node.style.setProperty("font-style", "italic", "important") : node.style.removeProperty("font-style"); var {lab} = pref; if (exists && pref.hasVals) { if (val in pref.vals) var sfx = pref.vals[val] || val; else var sfx = user ? "Другое" : "По умолчанию"; lab += ` — "${sfx}"`; } node.setAttribute("label", lab); }, createRadios(doc, vals, popup) { for(var arr of vals) { if (!arr) { popup.append(doc.createElementNS(xul_ns, "menuseparator")); continue; } var [val, lab, key, ttt] = arr; var menuitem = doc.createElementNS(xul_ns, "menuitem"); menuitem.setAttribute("type", "radio"); menuitem.setAttribute("closemenu", "none"); menuitem.style.setProperty("font-style", "italic", "important"), menuitem.setAttribute("label", popup.parentNode.pref.vals[val] = lab); key && menuitem.setAttribute("accesskey", key); var tip = menuitem.val = val; if (ttt) tip += "\n" + ttt; menuitem.tooltipText = tip; popup.append(menuitem); } }, openPopup(popup) { var btn = popup.parentNode; if (btn.domParent != btn.parentNode) { btn.domParent = btn.parentNode; var pos; if (btn.matches(".widget-overflow-list > :scope")) pos = "after_start"; else var win = btn.ownerGlobal, {width, height, top, bottom, left, right} = btn.closest("toolbar").getBoundingClientRect(), pos = width > height ? `${win.innerHeight - bottom > top ? "after" : "before"}_start` : `${win.innerWidth - right > left ? "end" : "start"}_before`; for(var p of btn.popups) p.setAttribute("position", pos); } popup.openPopup(btn); }, maybeRestart(node, conf) { var msgRest = "Перезапустить браузер?", msgAbort = "Запрос на выход отменен."; if (pv >= 77) { var title = node.closest("toolbarbutton").label; var pp = domWin => Services.prompt.wrappedJSObject.pickPrompter({ domWin, modalType: Ci.nsIPrompt.MODAL_TYPE_WINDOW }); var confirm = win => pp(win).confirm(title, msgRest); var alert = win => pp(win).alert(title, msgAbort); } else { var confirm = win => win.confirm(msgRest); var alert = win => win.alert(msgAbort); } return (this.mayBeRestart = (node, conf) => { var win = node.ownerGlobal; if (conf && !confirm(win)) return; if (win.BrowserUtils.restartApplication() === false) alert(win); else return true; })(node, conf); }, regexpRefresh: /^(?:view-source:)?(?:https?|ftp)/, maybeRe(node, fe) { var {pref} = node; if ("restart" in pref) { if (this.maybeRestart(node, pref.restart)) return; } else this.popupshowing(fe, node.parentNode); if ("refresh" in pref) { var win = node.ownerGlobal; if (this.regexpRefresh.test(win.gBrowser.currentURI.spec)) pref.refresh ? win.BrowserReloadSkipCache() : win.BrowserReload(); } }, maybeClosePopup(e, trg) { !e.ctrlKey && prefs.getBoolPref(this.closePref, undefined) && trg.parentNode.hidePopup(); }, command(e) { var trg = e.target; if (trg.btn) return this.openPopup(trg.primaryPopup); var menu = trg.closest("menu"), newVal = trg.val; this.maybeClosePopup(e, menu); if (newVal != menu.pref.val) menu.pref.set(menu.pref.pref, newVal), this.maybeRe(menu, true); }, popupshowing(e, trg = e.target) { if (trg.state == "closed") return; if (trg.id) { for(var node of trg.children) { if (node.nodeName.endsWith("r")) continue; this.upd(node); !e && node.open && this.popupshowing(null, node.querySelector("menupopup")); } return; } var {pref} = trg.closest("menu"), findChecked = true; var findDef = "defVal" in pref; var checked = trg.querySelector("[checked]"); if (checked) { if (checked.val == pref.val) { if (findDef) findChecked = false; else return; } else checked.removeAttribute("checked"); } if (findDef) { var def = trg.querySelector("menuitem:not([style*=font-style]"); if (def) if (def.val == pref.defVal) { if (findChecked) findDef = false; else return; } else def.style.setProperty("font-style", "italic", "important"); } for(var node of trg.children) if ("val" in node) { if (findChecked && node.val == pref.val) { node.setAttribute("checked", true); if (findDef) findChecked = false; else break; } if (findDef && node.val == pref.defVal) { node.style.removeProperty("font-style"); if (findChecked) findDef = false; else break; } } }, contextmenu(e) { var trg = e.target; if (trg.btn) { if (e.ctrlKey || e.shiftKey) return; if (e.detail == 2) return trg.secondaryPopup.hidePopup(); this.openPopup(trg.secondaryPopup); } else if ("pref" in trg) { this.maybeClosePopup(e, trg); if (trg.pref.user) prefs.clearUserPref(trg.pref.pref), this.maybeRe(trg); } e.preventDefault(); }, click(e) { if (e.button) return; var trg = e.target, {pref} = trg; if (!pref) return; this.maybeClosePopup(e, trg); if (!("noAlt" in pref)) return; if (pref.val == pref.userChoice) if (pref.noAlt) return; else pref.set(pref.pref, pref.userAlt); else pref.set(pref.pref, pref.userChoice); this.maybeRe(trg); } }; });
Отсутствует
Dumby
" вместо кавычек
Делал первым попавшимся онлайн-сервисом, теперь нашел плагин для npp++, в нем такого нет. Учту на будущее.
Спасибо. Получилась отличное меню в едином стиле, с традиционной индикацией состояния параметров и всякими дополнительными плюшками. Так намного лучше воспринимается, чем все вместе в стилях шрифта.
Отсутствует
Dumby
Подправьте пожалуйста, если возможно, этот скрипт для 79+
javascript.options.asyncstack — true в 79+ теперь по умолчанию, но никак не влияет на нужное поведение скрипта.
Add: т.е. в любом случае, через звёздочку или меню страницы, сохранение идёт в меню закладок. Спасибо.
Отредактировано sandro79 (17-08-2020 00:33:50)
Отсутствует
Dumby QuickSettings не работает на Basilisk (старый код работал)
можно сделать отдельную версию или адаптировать новый код для совместимости с Basilisk ?
Vitaliy V. а как отлаживать код для user_chrome_files без перезагрузки браузера ?
или это уже есть ? То есть в [CB] я выбираю: Редактировать код и изменения сразу начинают работать.
Как сделать, чтобы после правки скриптов user_chrome_files изменённые скрипты перезагружались автоматически ?
Отсутствует
javascript.options.asyncstack — true в 79+ теперь по умолчанию
Да, вижу (bug). Поставили true и выключили (псевдопарадокс).
На сей раз предлагать включить обратно не стану,
поскольку запугали, что мол это не по-перформансу.
Может тогда попробуй ещё более грубое
try {((obj, key) => { var funcs = [eval(`(${obj[key]})`.replace(" ", " function ").replace("unfiledGuid", "menuGuid")), obj[key]]; obj[key] = (...args) => funcs[+Components.stack.formattedStack.includes("nsContextMenu")](...args); })(PlacesCommandHook, "bookmarkPage");} catch(ex) {Cu.reportError(ex);}
QuickSettings не работает на Basilisk
Ага, на Firefox 2.0.0.8 тоже не работает.
А вообще который? Я вот попробовал поставить (как кнопку) тот,
что последний для _zt на Basilisk 2020.08.05, и, не скажу чтобы
прямо всё проверил, но на первый взгляд выглядит вполне рабочей.
Но засчёт того, что там (в кнопке) нет строковых настроек.
Если «не работает» в этом смысле, тогда попробуй заменить свойство map
на такой геттер
... //map: {b: "Bool", n: "Int", s: "String"}, get map() { if (!prefs.getStringPref) { var ssi = Components.interfaces.nsISupportsString; var gs = (o, name) => o.getComplexValue(name, ssi).data; var ss = (o, name, val) => { var str = Components.classes["@mozilla.org/supports-string;1"].createInstance(ssi); str.data = val; o.setComplexValue(name, ssi, str); } var p = {...prefs}, d = {...db}; [[p, prefs], [d, db]].forEach(([o, b]) => { o.getStringPref = function getStringPref(name) {return gs(b, name);} o.setStringPref = function setStringPref(name, val) {ss(b, name, val);} }); prefs = p; db = d; } delete this.map; return this.map = {b: "Bool", n: "Int", s: "String"}; },
Отсутствует
Как сделать, чтобы после правки скриптов user_chrome_files изменённые скрипты перезагружались автоматически ?
Не к чему это усложнять, к тому же это нужно в каждом куске коде предусматривать методы для инициализации и разрушения,
почти на каждый листенер, обсервер, добавленный элемент и т.д. лепить addDestructor() как в CB... Кроме того есть инструменты браузера.
Отсутствует
Не к чему это усложнять … Кроме того есть инструменты браузера.
А как после правки user_chrome_files скриптов перезагрузить весь их код кнопкой или HotKey ?
То есть неавтоматически, но без перезапуска браузера ?
Отсутствует
Общий вопрос - нужна кнопка, которая откроет html-файл, расположенный например, в profile/chrome
Нужно открыть справку в новой вкладке, а по повторному клику закрыть (если во вкладке открыта именно htm-справка)
В html будет справка по горячим клавишам, использованию функций кнопок (в моём профиле их много)…
Отсутствует
Общий вопрос - нужна кнопка, которая откроет html-файл, расположенный например, в profile/chrome
Нужно открыть справку в новой вкладке, а по повторному клику закрыть (если во вкладке открыта именно htm-справка)
Звучит как-то так
this._handleClick = () => { var fileName = "help.html"; var params = { inBackground: false, relatedToCurrent: true }; var file = Services.dirsvc.get("UChrm", Ci.nsIFile); file.append(fileName); var uri = Services.io.newFileURI(file); var open = window.openTrustedLinkIn || openUILinkIn; (this._handleClick = () => uri.equals(gBrowser.currentURI) ? gBrowser.removeCurrentTab() : open(uri.spec, "tab", params) )(); }
Отсутствует