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

Заказывай стафф с атрибутикой Mozilla и... пусть все вокруг завидуют тебе! Быть уникальным - быть с Mozilla!

№1497613-08-2020 14:33:14

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

Re: Custom Buttons

Dumby пишет

Если я ничего не упускаю, то для этого просто нет API, то есть
нет чего-то типа getDefaultPref(prefName) или isDefaultPref(prefName, prefValue)

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

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

Выделить код

Код:

var name = "extensions.user_chrome_files.custom_script";
var getters_pref_type = {
  [Ci.nsIPrefBranch.PREF_BOOL]: "getBoolPref",
  [Ci.nsIPrefBranch.PREF_INT]: "getIntPref",
  [Ci.nsIPrefBranch.PREF_STRING]: "getStringPref",
};
var type = Services.prefs.getPrefType(name);
try {
   console.log("getDefaultPref: " + Services.prefs.getDefaultBranch("")[getters_pref_type[type]](name));
} catch (ex) {
   console.log("getDefaultPref: none");
}

Отсутствует

 

№1497713-08-2020 15:03:00

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 399
UA: unknown 0.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)

Отсутствует

 

№1497813-08-2020 15:46:16

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

Re: Custom Buttons

_zt пишет

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

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

Отсутствует

 

№1497913-08-2020 16:10:02

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

Re: Custom Buttons

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

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

Отсутствует

 

№1498013-08-2020 18:02:20

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 399
UA: unknown 0.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)

Отсутствует

 

№14981Вчера 05:34:27

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

Re: Custom Buttons

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

Отредактировано Kero (Вчера 05:36:45)

Отсутствует

 

№14982Вчера 13:34:51

Duche
Участник
 
Группа: Members
Зарегистрирован: 07-02-2016
Сообщений: 190
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 (Вчера 13:41:07)

Отсутствует

 

№14983Вчера 16:20:19

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 1207
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 пишет

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

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

Отсутствует

 

№14984Вчера 21:03:43

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

Re: Custom Buttons

Dumby пишет

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

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

Отсутствует

 

№14985Вчера 21:24:56

_zt
Участник
 
Группа: Members
Зарегистрирован: 10-11-2014
Сообщений: 399
UA: unknown 0.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 (Вчера 21:39:11)

Отсутствует

 

Board footer

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