Полезная информация

Хотите узнать больше о расширениях? Посмотрите ролики, рассказывающие о работе с расширениями Firefox.

№1492613-08-2020 15:03:00

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

Dumby
Я имел ввиду выделять userChoice, а не наоборот, у себя изменил.
   
По поводу пунктов true-false:
Текст значения после заголовка добавьте пожалуйста, что бы к единому стилю привести. Например после:

скрытый текст

Выделить код

Код:

pref: ["javascript.enabled", "Выполнять скрипты Java", "", "javascript.enabled"],
			userChoice: true, refresh: true,
			values: [[true, "Да"], [false, "Нет"]]

"Выполнять скрипты Java" должно быть длинное тире и "Да"/"Нет" (по подобию с другими пунктами).
   
По поводу всех подменю и выделения в них дефолтных / не дефолтных значений:
Вот здесь же bold для user задается, или я чего то не понял.
скрытый текст

Выделить код

Код:

user
				? node.style.setProperty("font-weight", "bold", "important")
				: node.style.removeProperty("font-weight");

Т.е. получить значение по умолчанию, и выделить его (или отличное от него), можно.
Или как вы это вообще задумывали? То что пишет Dobrov для меня не соответствует действительности, у меня все что отлично от дефолта выделяется.
Заменил себе на курсив:
2020.1597319645.png
   
------------------
У меня ESR 78

Отредактировано _zt (13-08-2020 18:33:41)

Отсутствует

 

№1492713-08-2020 15:46:16

Vitaliy V.
Участник
 
Группа: Members
Зарегистрирован: 19-09-2014
Сообщений: 2186
UA: Firefox 80.0

Re: Custom Buttons

_zt пишет

Вот здесь же bold для user задается, или я чего то не понял.

это для текущего значения - prefHasUserValue, а чтобы проверить другие значения (в подменю) нужно определить его дефолтное значение,
не переключать же настройку и проверять каждое...

Отсутствует

 

№1492813-08-2020 16:10:02

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

Vitaliy V.
Теперь понятно. Тогда может на базе вашего кода решение найдется.
   
Dumby
maroon я у помянул так как он у меня не работает. Сами видите какое у меня меню, так что не удивительно, что цвета в нем не работают.
Вы все правильно сделали, только, на мой взгляд, логичнее UserChoice выделять, а не отличные от него.
Хотя, ради баловства, можно для каждого значения свою иконку назначить - UserChoice / notUserChoice.

Отредактировано _zt (13-08-2020 16:12:57)

Отсутствует

 

№1492913-08-2020 18:02:20

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

Dumby
Все таки добавил иконку на notUserChoice. Казалось, что будет сложнее.
2020.1597330070.png
Теперь вопрос появился, а нельзя ли на иконку (или весь пункт) клик ЛКМ повесить? (ПКМ там и так работает, сбрасывает значение на "по умолчанию" (я так понимаю)).
   
Дело в том, что подменю может появится и справа и слева, и все это приводит к бестолковым перемещениям мыши. А так можно будет только информацию снимать глазами с подменю, а переключать щелчком мыши.
   
Последовательное переключение неприемлемо (из-за refresh: true, restart: true), но можно добавить какой нибудь UserAlt и переключать по клику ЛКМ (UserChoice > UserAlt, UserAlt > UserChoice), если UserAlt назначен пункту.
   
В этом случае, UserAlt тоже можно будет свою иконку приделать.

Отредактировано _zt (13-08-2020 18:29:59)

Отсутствует

 

№1493014-08-2020 05:34:27

Kero
Участник
 
Группа: Members
Зарегистрирован: 09-11-2012
Сообщений: 221
UA: Firefox 52.0

Re: Custom Buttons

а есть ли где коллекция уже готовых кнопок в файле buttonsoverlay.xul ?
И как то можно отключить backup или задать только 1 backup ?

Отредактировано Kero (14-08-2020 05:36:45)

Отсутствует

 

№1493114-08-2020 13:34:51

Duche
Участник
 
Группа: Members
Зарегистрирован: 07-02-2016
Сообщений: 208
UA: unknown 0.0

Re: Custom Buttons

Добрый день. Посмотрите пожалуйста  код для 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)

Отсутствует

 

№1493214-08-2020 16:20:19

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2259
UA: Firefox 52.0

Re: Custom Buttons

firepox пишет

пробовал в начало кода добавлять это:
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)
);

Dobrov пишет

Хотелось бы совместить оба варианта - для _zt и для меня с методом londPress, они почему-то сильно различаются…

Ну, ты выразил пожелание чтобы менюшки закрывались,
а _zt к этому пожеланию не присоединился, и направился
куда-то в сторону единостилия. Поэтому и различаются.


Но londPress достаточно автономен, там только здесь вписан "mousedown":
for(var type of ["command", "contextmenu", "mousedown"])
и, в конце, метод mousedown, обрабатывающий это событие,
и метод londPress, куда передаётся результат обработки.


Vitaliy V. пишет

может как то так

Да! Это работает! Спасибо!
И ведь что-то подобное в коде есть, но мысль проверить
что возвращает defaultBranch мою тупую голову не посетила.


_zt пишет

у себя изменил

Смотрю много измененений.
Скинь тогда свой код целиком, чтобы было с чем возиться.
С проставленными userAlt и userAltImg.


Резюмирую, скорее для себя, что попробовать сделать:
1. italic в субменюшках для недефолтных значений,
    учитывая наставление Виталия.
2. длинное тире + лейбл и для true-false <menu>
    при наличии "values".
3. Последовательное переключение по клику на <menu>
    если отсутствуют restart и refresh,


    иначе, переключение между userChoice и userAlt,
    если присутствуют оба. Плюс иконка userAlt.
    (Если текущее значение не userChoice и не userAlt
    переключать на userChoice или на userAlt ?)
Всё это займёт какое-то время, полагаю.


Kero пишет

как то можно отключить backup

Наверно да, %Profile%\extensions\custombuttons@xsms.org\components\CustomButtonsService.js
function backupProfile (phase) {if(true)return;


Duche пишет

Удалить куки текущего сайта в контекстном меню на странице

Может подойдёт в контекстном меню не на странице, а на кнопке?

Отсутствует

 

№1493314-08-2020 21:03:43

firepox
Участник
 
Группа: Members
Зарегистрирован: 17-11-2011
Сообщений: 358
UA: Firefox 56.0

Re: Custom Buttons

Dumby пишет

firepox пишетпробовал в начало кода добавлять это:gBrowser.selectedTab = gBrowser.addTab('http://site.ru');Не работает)Направление мысли верное, но следует дождаться пока загрузится.Попробуй разместить в Инициализации, и если «нужный сайт»прям после загрузки ещё не готов быть снапшотнутым, тогдаподними таймаут с нуля до какого-нибудь эмпирического значения.

код вставил в инициализацию -  ничего не сохраняется вообще

Отредактировано firepox (15-08-2020 11:40:37)

Отсутствует

 

№1493414-08-2020 21:24:56

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

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();
		}
	};
});


2020.1597428606.png

Отредактировано _zt (14-08-2020 21:39:11)

Отсутствует

 

№1493515-08-2020 08:53:18

Duche
Участник
 
Группа: Members
Зарегистрирован: 07-02-2016
Сообщений: 208
UA: unknown 0.0

Re: Custom Buttons

Dumby пишет

Может подойдёт в контекстном меню не на странице, а на кнопке?

Это не кнопка,это комбайн какой то . Нет нужно попроще, только из контекстного меню. Желательно по возможности ,реанимировать старый код. Просто у некоторых, руки работаю быстрее головы, жмут на всё подряд, а до контекстного меню не дотягиваются, тонкой тайм-аут. Может у кого получится починить код.

Отсутствует

 

№1493615-08-2020 13:04:43

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2259
UA: Firefox 52.0

Re: Custom Buttons

firepox пишет

код вставил инициализацию -  ничего не сохраняются вообще

Ну не знаю, может 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});
})(() => {

//======================
// Здесь расположить
// весь код вкладки Код.
//======================

});


_zt
скрытый текст

Что-то по клику на <menu> не готов сказать, что сложилась
у меня какая-то полная ясность.
Сделал так: если userChoice отсутствует, то клик не делает ничего,
иначе — переключает на userChoice, а если текущее значение
уже совпадает с userChoice, тогда переключает на userAlt, если таковой имеется.

Иконку UserAlt прописал и она работает, но вы проверьте за мной

Вот эта строка:
            else ("userAlt" in pref)
не делает ничего (значимого), просто по else вычисляет есть ли у этого объекта такое свойство,
вычисляет в пустоту. Ну а дальше там просто установка "image", если совпадение с userAlt.
Всё бы ничего, но бывает, перед этим, чуть выше по коду, ставится notUserChoiceImg,
и сразу UserAltImg. То есть двойная установка, а это, наверно, нехорошо. Переделал.


И ещё, немного похозяйничал. Заметил, что в каждом(!) pref'е, в конце,
продублировано имя настройки, чтобы оно отображалось в тултипе.
Показалось похожим на концепцию, поэтому поудалял их все, а отображение
имени настройки в тултипе добавил в код.
Ну и оставшиеся пустые "" под accesskey тоже удалил. Сама возможность
дописывать accesskey и дополнительный текст к тултипу, разумеется, никуда не делась.
Если рассудил неверно, тогда упс.


Вобщем вот. Но выложенный код далеко не убирай, в том смысле, что если
я там чего-то испортил, и надо будет откатить, и понадеяться на тот, что на форуме,
то придётся реплейсить &quot; &amp; &gt; &mdash; которые там прописались, как-то.

Выделить код

Код:

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);
		}
	};
});

Отсутствует

 

№1493715-08-2020 15:27:52

Dobrov
Участник
 
Группа: Members
Зарегистрирован: 04-10-2011
Сообщений: 476
UA: Firefox 56.0

Re: Custom Buttons

Dumby ещё просьба по QuickSettings - сделать константу в коде, которая управляет скрытием меню после клика.
Например const HideMenu = 1; // автоскрывать после клика…


Может ещё константу для отключения default иконок сделаете ?
Чтобы иконки были как раньше, только на пунктах меню, в которых прописано userChoice: значение по-умолчанию.

Отредактировано Dobrov (15-08-2020 15:33:46)

Отсутствует

 

№1493815-08-2020 16:00:54

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

Dumby

скрытый текст

Сделал так: если userChoice отсутствует, то клик не делает ничего,
иначе — переключает на userChoice, а если текущее значение
уже совпадает с userChoice, тогда переключает на userAlt, если таковой имеется

Все верно, я почти так и писал. Переключение на userChoice из любого отличного значения это уже вами добавленный бонус.
   

Заметил, что в каждом(!) pref'е, в конце,
продублировано имя настройки, чтобы оно отображалось в тултипе.
Показалось похожим на концепцию, поэтому поудалял их все, а отображение
имени настройки в тултипе добавил в код.

Тоже верно. Тултип никому не мешает, но зато всегда можно увидеть реальный преф скрытый под заголовком.
   

Ну и оставшиеся пустые "" под accesskey тоже удалил.

Я ими не пользуюсь. Возможно они тут работают, но я уже привык, что, как правило, они полностью неработоспособны, то ли с неанглийскими, то ли с измененными раскладками.
   

&quot; &amp; &gt; &mdash; которые там прописались, как-то

Из-за перевода? Как их увидеть? Если об этом речь.
2020.1597495749.png
2020.1597497830.png

Спасибо. На первый взгляд все нормально, поставил на основной профиль, сейчас наполню префами и посмотрю за поведением.
   
Согласен с Dobrov, быстрое переключение "скрывать"/"не скрывать" меню после клика, тут не помешало бы. Для тестов удобнее как сейчас, а при постоянном использовании нескрываемое меню будет отвлекать.
А интересно, нельзя ли эту опцию во вторичное меню вывести? ) Например, добавлением этой настройки в about:config и последующем ее чтением.
   
Добавлено 15-08-2020 16:07:56
Dumby
скрытый текст

Выделить код

Код:

node.style.removeProperty("color"),
и
node.style.setProperty("color", "maroon", "important");

Можно удалить, ничего не поломав?

Отредактировано _zt (15-08-2020 17:25:31)

Отсутствует

 

№1493915-08-2020 18:26:22

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

Dobrov
Просто удалите иконки, приведите соответствующие строки к такому виду:

скрытый текст

Выделить код

Код:

notUserChoiceImg: "",
UserAltImg: "",

После этого у вас не останется индикации UserAlt.
   
userChoice это не значение по умолчанию, значение по умолчанию будет когда вы ПКМ по пункту щелкнете.

Отредактировано _zt (15-08-2020 18:27:24)

Отсутствует

 

№1494016-08-2020 09:59:57

Duche
Участник
 
Группа: Members
Зарегистрирован: 07-02-2016
Сообщений: 208
UA: unknown 0.0

Re: Custom Buttons

Добрый день. Посмотрите пожалуйста  код для 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)

Отсутствует

 

№1494116-08-2020 19:40:53

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2259
UA: Firefox 52.0

Re: Custom Buttons

_zt

скрытый текст

Из-за перевода? Как их увидеть? Если об этом речь.

Не-не, увидеть очень просто: если открыть во вкладке base64 адрес с кодом
из этого поста, и приглядеться к содержимому, то видно, что оно словно
пропущено через подготовщик для XML. Самое заметное — &quot; вместо кавычек.


Можно удалить, ничего не поломав?

Думаю да, на эти строки ничего не завязано, закомментировал.


Согласен с 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);
		}
	};
});

Отсутствует

 

№1494216-08-2020 21:40:33

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 1647
UA: Firefox 78.0

Re: Custom Buttons

Dumby

&quot; вместо кавычек

Делал первым попавшимся онлайн-сервисом, теперь нашел плагин для npp++, в нем такого нет. Учту на будущее.
   
Спасибо. Получилась отличное меню в едином стиле, с традиционной индикацией состояния параметров и всякими дополнительными плюшками. Так намного лучше воспринимается, чем все вместе в стилях шрифта.

Отсутствует

 

№1494316-08-2020 23:47:42

sandro79
Участник
 
Группа: Members
Зарегистрирован: 15-11-2017
Сообщений: 1750
UA: Firefox 78.0

Re: Custom Buttons

Dumby
Подправьте пожалуйста, если возможно, этот скрипт для 79+
javascript.options.asyncstack — true в 79+ теперь по умолчанию, но никак не влияет на нужное поведение скрипта.
Add: т.е. в любом случае, через звёздочку или меню страницы, сохранение идёт в меню закладок. Спасибо.

Отредактировано sandro79 (17-08-2020 00:33:50)

Отсутствует

 

№1494417-08-2020 05:01:09

Dobrov
Участник
 
Группа: Members
Зарегистрирован: 04-10-2011
Сообщений: 476
UA: Firefox 56.0

Re: Custom Buttons

Dumby QuickSettings не работает на Basilisk (старый код работал)
можно сделать отдельную версию или адаптировать новый код для совместимости с Basilisk ?


Vitaliy V. а как отлаживать код для user_chrome_files без перезагрузки браузера ?
или это уже есть ? То есть в [CB] я выбираю: Редактировать код и изменения сразу начинают работать.
Как сделать, чтобы после правки скриптов user_chrome_files изменённые скрипты перезагружались автоматически ?

Отсутствует

 

№1494517-08-2020 09:23:51

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2259
UA: Firefox 52.0

Re: Custom Buttons

sandro79 пишет

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);}

Dobrov пишет

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"};
		},

Отсутствует

 

№1494617-08-2020 12:43:07

sandro79
Участник
 
Группа: Members
Зарегистрирован: 15-11-2017
Сообщений: 1750
UA: Firefox 78.0

Re: Custom Buttons

Dumby пишет

Может тогда попробуй ещё более грубое

Благодарю! Работает! :beer:

Отсутствует

 

№1494717-08-2020 22:23:20

Vitaliy V.
Участник
 
Группа: Members
Зарегистрирован: 19-09-2014
Сообщений: 2186
UA: Firefox 80.0

Re: Custom Buttons

Dobrov пишет

Как сделать, чтобы после правки скриптов user_chrome_files изменённые скрипты перезагружались автоматически ?

Не к чему это усложнять, к тому же это нужно в каждом куске коде предусматривать методы для инициализации и разрушения,
почти на каждый листенер, обсервер, добавленный элемент и т.д. лепить addDestructor() как в CB... Кроме того есть инструменты браузера.

Отсутствует

 

№1494818-08-2020 00:52:39

Dobrov
Участник
 
Группа: Members
Зарегистрирован: 04-10-2011
Сообщений: 476
UA: Firefox 56.0

Re: Custom Buttons

Vitaliy V. пишет

Не к чему это усложнять …  Кроме того есть инструменты браузера.

А как после правки user_chrome_files скриптов перезагрузить весь их код кнопкой или HotKey ?
То есть неавтоматически, но без перезапуска браузера ?

Отсутствует

 

№1494918-08-2020 01:53:26

Dobrov
Участник
 
Группа: Members
Зарегистрирован: 04-10-2011
Сообщений: 476
UA: Firefox 56.0

Re: Custom Buttons

Общий вопрос - нужна кнопка, которая откроет html-файл, расположенный например, в profile/chrome
Нужно открыть справку в новой вкладке, а по повторному клику закрыть (если во вкладке открыта именно htm-справка)

В html будет справка по горячим клавишам, использованию функций кнопок (в моём профиле их много)…

Отсутствует

 

№1495018-08-2020 10:21:32

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2259
UA: Firefox 52.0

Re: Custom Buttons

Dobrov пишет

Общий вопрос - нужна кнопка, которая откроет 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)
	)();
}

Отсутствует

 

Board footer

Powered by PunBB
Modified by Mozilla Russia
Copyright © 2004–2020 Mozilla Russia GitHub mark
Язык отображения форума: [Русский] [English]