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

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

№1567613-06-2021 21:48:11

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

Re: Custom Buttons

ВВП пишет

Как бы убрать #toggle_PersonalToolbar там ,где мне надо? Там id="toolbar-context-menu" везде одинаковый. И дотянуть до Nav-bar ,к примеру, ну никак...

Видимо, зависит от того, где надо.
На скрине, как я понимаю, на виджетах, создаваемых для WebExtensions.
Можно, например, так попробовать

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

Выделить код

Код:

#toolbar-context-menu > menuitem.customize-context-manageExtension:not([hidden]) ~ #toggle_PersonalToolbar {
	display: none !important;
}


А «дотянуть до Nav-bar» в смысле поднять выше остальных?
Вроде -moz-box-ordinal-group: 0 !important; должен работать.


vv07
Хорошо бы писать, что кнопка делала.
Не столько вообще, а, скорее, что делала для тебя.
В том смысле, что важно, что не очень, а что совсем не использовалось.
Это так, примечание общего характера.


Вот, скажем, Open URLs, так там написано, что для Thunderbird
Допустим, запишу так, подойдёт?

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

Выделить код

Код:

/*Initialization Code*/

(n => {
	var inBackground = false;
	var data = [{
		lab: "Google",
		url: "https://www.google.com",
		img: ""
	}, {
		lab: "Форум Mozilla",
		url: "https://forum.mozilla-russia.org",
		img: ""
	},
	null,
	{
		lab: "Test",
		url: "data:text/plain;charset=utf-8,Params",
		img: "chrome://browser/content/robot.ico",
		params: {
			userContextId: 1,
			get index() {
				return gBrowser.selectedTab._tPos + 1
			},
			inBackground: !inBackground,
		}
	}];
	this.type = "menu";
	var popup = n("menupopup"), dummy = n("menuitem");
	dummy.render = () => {
		dummy.remove();
		data.forEach((o, ind) => {
			if (!o) return popup.append(n("menuseparator"));
			var menuitem = n("menuitem");
			menuitem.setAttribute("label", o.lab || o.url);
			if (o.img)
				menuitem.className = "menuitem-iconic",
				menuitem.setAttribute("image", o.img);
			menuitem.ind = ind;
			popup.append(menuitem);
		});
		popup.setAttribute("oncommand", "tab(event.target.ind);");
		popup.tab = ind => {
			var {url, params} = data[ind];
			var tab = gBrowser.addTrustedTab(url, params);
			if (!(params?.inBackground || inBackground)) gBrowser.selectedTab = tab;
		}
	}
	popup.append(dummy);
	this.prepend(popup);

	//this.onmouseover = () => this.open = true;

	this.removeAttribute("tooltiptext");
	var tt = this.appendChild(n("box")).appendChild(n("tooltip"));
	tt.setAttribute("onpopupshowing", "return !(parentNode.parentNode.open = true);");
	this.setAttribute("tooltip", tt.id = _id + "-tooltip");

})(nn => document.createXULElement(nn));


Compact Menu? Ну, у меня давненько уже
болтается что-то приблизительно похожее, но, опять же, подойдёт ли?
скрытый текст

Выделить код

Код:

/*Initialization Code*/

(this.type != "menu" && (this.type = "menu") && !this.hasAttribute("is")) || (move => {
	if (this.parentNode.nodeName == "toolbarpaletteitem") return;
	var bar = document.getElementById("main-menubar");
	var menupopup = document.createXULElement("menupopup");
	this.prepend(menupopup);
	move(bar, menupopup);
	addDestructor(() => move(menupopup, bar));
})((from, to) => Array.from(from.children).forEach(child => {
	var popup = child.querySelector(":scope > menupopup");
	popup.remove();
	child.textContent = child.renderedOnce = "";
	to.append(child);
	child.render();
	child.append(popup);
}));


На Extension List сейчас времени нет.
Может повожусь на следующей неделе, если ничего не будет отвлекать.

Отсутствует

 

№1567713-06-2021 22:58:07

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
Благодарю за идею ! Сразу со всех ненужных popup
#toolbar-context-menu > menuitem.customize-context-removeFromToolbar:not([hidden]) ~ #toggle_PersonalToolbar {
    display: none !important;
}

Отсутствует

 

№1567814-06-2021 00:51:47

vv07
Участник
 
Группа: Members
Зарегистрирован: 07-11-2007
Сообщений: 672
UA: Firefox 54.0

Re: Custom Buttons

Dumby
Спасибо Дружище!
Привередничать не буду :o
Хотя если честно, то в старой версии Compact Menu, с иконками симпатичнее было

скрытый текст
6a66816471dd.jpg

Open URLs отлично работает. Очень благодарен вам. Есть только вопрос. У меня ссылок много очень, нельзя ли как то сделать, что бы использовать две таких кнопки одновременно? Что бы работали параллельно и независимо, а то ставлю две кнопки, но одни дублируют друг, друга
Без обид, что надоедаю. Я старый человек. Очень тяжел на подьем. К чему привыкаю, сложно отказываться.
Если не будете против. Попрошу еще о некоторых кнопках, без которых не очень удобно.
PS Обнаружились проблемы при работе Compact Menu. В ряде случаев, происходит такое
скрытый текст
561f9f9de385.jpg

Раздел
Вид-Боковая панель
Вид-Стиль страницы, Полный экран, Добавить новую кнопку.
Закладки, Справка, Инструменты. Журнал

Отредактировано vv07 (14-06-2021 08:13:36)

Отсутствует

 

№1567914-06-2021 13:34:03

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
Как бы этот код в ini запихать? Раньше он просто в коде был , но кнопка прокси изменилась...Короче, отмена ни отмена, код то уже сработал...

скрытый текст
var id = "{acf99872-d701-4863-adc2-cdda1163aa34}";

AddonManager.getAddonByID(id).then(
    addon => addon[
        addon.userDisabled ? "enable" : "disable"
       
    ]({allowSystemAddons: true}),
    Cu.reportError
)

Сама кнопка прокси:

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

Выделить код

Код:

(ps => {   
                                                                      
     
   this._handleClick = () => Services.prompt.confirm(null, "ВНИМАНИЕ !", "Прокси серверы : Включить / Выключить ?") && ps.setIntPref(s, ps.getIntPref(s, 2) ? 0 : 2);

     var s = "network.proxy.type";
 
    var toggleImage = val => {
        this.icon.src = (val = ps.getIntPref(s, 2))
            ? this.image
            : "";             
        this.tooltipText = val ? 'Автопрокси' : 'Без прокси';



        BrowserReload();
 
        cbu.setPrefs("network.proxy.autoconfig_url", val ? "resource://chrome/proxy/proxy2.pac" : "resource://chrome/proxy/proxy2.pac");
        cbu.setPrefs("intl.accept_languages", val ? "fr" : "ru");
        cbu.setPrefs("browser.zoom.full", val ? true : true); 
       cbu.setPrefs("media.autoplay.default", val ? 5 : 5); 

        
    }
    toggleImage();
    ps.addObserver(s, toggleImage, false);
    addDestructor(() => ps.removeObserver(s, toggleImage));
})(Services.prefs);

this.oncontextmenu =e=> { e.button && !e.ctrlKey && e.preventDefault() };

Отредактировано ВВП (14-06-2021 13:36:16)

Отсутствует

 

№1568015-06-2021 12:28:54

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

Re: Custom Buttons

vv07 пишет

в старой версии Compact Menu, с иконками симпатичнее было

Вот об этом я и говорю, что описание должно быть персонализированным.
Оказывается, возможность исключить пункты не важна,
а возможность задать иконки — важна (хотя в кнопке по ссылке
ничего подобного нет, это чем-то другим у тебя было сделано).

Обнаружились проблемы при работе Compact Menu. В ряде случаев, происходит такое

Хмм, странно, не вижу такого.
Нужен STR (шаги по воспроизводству), типа делай раз, делай два, делай три..
Иначе говоря, описание некой последовательности действий, проделав которые у себя,
я смогу увидеть проблему, то есть получить то, что демонстрирует приведённый скриншот.


Но косяк (другой) есть. Всё-таки, после перемещения #bookmarksMenu,
без пересборки _placesView портится контекстное меню его закладок,
так и не понятно почему.
Новая версия. Заодно, объект {"id": "icon"}, куда вписывать свои иконки.

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

Выделить код

Код:

this.parentNode.nodeName == "toolbarpaletteitem" || (icons => {
	this.type = "menu";
	var bar = document.getElementById("main-menubar");
	var menupopup = document.createXULElement("menupopup");
	menupopup.toggleAttribute("context");
	this.prepend(menupopup);
	var move = (from, to, ico) => {
		from.querySelector("#bookmarksMenu")._placesView?.uninit();
		for(var menu of Array.from(from.children)) {
			var popup = menu.menupopup;
			popup.remove();
			menu.textContent = menu.renderedOnce = "";
			if (ico) {
				var img = icons[menu.id];
				if (img)
					menu.className = "menu-iconic",
					menu.setAttribute("image", img);
			} else
				menu.removeAttribute("class"),
				menu.removeAttribute("image");
			to.append(menu);
			menu.render();
			menu.append(popup);
		}
	}
	move(bar, menupopup, true);
	addDestructor(() => move(menupopup, bar));
})({
	    "file-menu": "chrome://browser/content/robot.ico",
	    "edit-menu": "chrome://browser/skin/preferences/face-smile.svg",
	    "view-menu": "chrome://browser/skin/preferences/face-sad.svg",
	 "history-menu": "chrome://browser/content/robot.ico",
	"bookmarksMenu": "chrome://browser/skin/preferences/face-smile.svg",
	   "tools-menu": "chrome://browser/skin/preferences/face-sad.svg",
	     "helpMenu": "chrome://browser/content/robot.ico"
});

ВВП пишет

Раньше он просто в коде был

Ну, я же не в курсе включить или выключить.
Раз был, смотри сам.

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

Выделить код

Код:

…
	this._handleClick = () => {
		var direct = ps.getIntPref(s) == 0;
		var msg = `Прокси серверы : В${direct ? "" : "ы"}ключить ?`;

		if (!Services.prompt.confirm(null, "ВНИМАНИЕ !", msg)) return;

		ps.setIntPref(s, direct ? 2 : 0);
		var method = `${direct ? "dis" : "en"}able`;
		AddonManager.getAddonByID("{acf99872-d701-4863-adc2-cdda1163aa34}")
			.then(addon => addon[method](), Cu.reportError);
	}

Отсутствует

 

№1568115-06-2021 13:10:00

vv07
Участник
 
Группа: Members
Зарегистрирован: 07-11-2007
Сообщений: 672
UA: Firefox 54.0

Re: Custom Buttons

Dumby
Спасибо большое. Работает.
Научите пожалуйста, как делать тут?
"file-menu": "chrome://browser/content/robot.ico",
"edit-menu": "chrome://browser/skin/preferences/face-smile.svg",
"view-menu": "chrome://browser/skin/preferences/face-sad.svg",
"history-menu": "chrome://browser/content/robot.ico",
"bookmarksMenu": "chrome://browser/skin/preferences/face-smile.svg",
"tools-menu": "chrome://browser/skin/preferences/face-sad.svg",
"helpMenu": "chrome://browser/content/robot.ico"
Что бы свои иконки поставить. И да, иконки отображаются, но у меня таких нет на компе вообще))))
PS на старой кнопке вот так все

скрытый текст
ea9c30554193.jpg

Я сам ничего не делал, потому что элементарно не умею. Поэтому я сказал вам, но это же не критично было,
но очень приятно что сделали))))) Еще бы самому научится менять иконки

Отредактировано vv07 (15-06-2021 16:31:15)

Отсутствует

 

№1568215-06-2021 14:17:23

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby

Dumby пишет

Ну, я же не в курсе включить или выключить.

Все в елочку .dis и En -поменял .Класс! Аддон времени вкл вместе с прокси. Класс !
Последняя шняга. Чертов popup на кнопке Undo Close Tab. Стилем не могу убрать(убирается на всех кнопках) . В самой кнопке - стремно...
ep7s7cvo.png

Отредактировано ВВП (15-06-2021 14:18:23)

Отсутствует

 

№1568315-06-2021 19:39:06

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

Re: Custom Buttons

vv07 пишет

Научите пожалуйста, как делать тут?
Что бы свои иконки поставить.

Нужно вписать (заменить на свой) адрес ссылки на файл иконки.
Самое простое — по протоколу file:
Жмём Ctrl+O, выбираем нужное изображение,
копируем из адресной строки ссылку на открывшееся изображение,
и вписываем.


Можно по протоколу data: (всё своё ношу с собой)
Тут, для вэб-человеков, говорят, есть онлайн-кодировщики,
можно поискать.
Или прямо в CB-редакторе некоторые конвертировали
кнопками [Обзор…] и [⇒ base64] и копированием получившейся
ссылки с <menulist>'а (слева от них).
Или, вот у меня есть старая, крайне аскетичная кнопка конвертации,
нажал, выбрал, и всё, в буфере обмена base64, остаётся только вставить.

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

Выделить код

Код:

custombutton://%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%0D%0A%3Ccustombutton%20xmlns%3Acb%3D%22http%3A//xsms.nm.ru/custombuttons/%22%3E%0A%20%20%3Cname%3E64%3C/name%3E%0A%20%20%3Cimage%3E%3C%21%5BCDATA%5Bdata%3Aimage/x-icon%3Bbase64%2CAAABAAEAEBAAAAEAIABoBAAAFgAAACgAAAAQAAAAIAAAAAEAIAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAD/AIAQ/wCAEf8AgA//AIAR/wCAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP8AgBX/AIAVAAAAAAAAAAD/AIAo/wCA//8AgP//AID//wCA//8AgP//AIAoAAAAAAAAAAAAAAAAAAAAAP8AgBL/AID//wCA//8AgA3/AIAL/wCA//8AgP//AID//wCA//8AgP//AID//wCA//8AgBAAAAAAAAAAAAAAAAD/AIAR/wCA//8AgP//AIAK/wCACv8AgP//AID//wCAIf8AgAX/AIAh/wCA//8AgP//AIAQAAAAAAAAAAAAAAAA/wCACv8AgP//AID//wCAB/8AgAf/AID//wCA//8AgAUAAAAA/wCABf8AgP//AID//wCACgAAAAD/AIAQ/wCADP8AgCH/AID//wCA//8AgAf/AIAH/wCA//8AgP//AIAh/wCABf8AgCH/AID//wCA//8AgAv/AIAh/wCA//8AgP//AID//wCA//8AgP//AIAH/wCAB/8AgP//AID//wCA//8AgP//AID//wCA//8AgP//AIAg/wCA//8AgP//AID//wCA//8AgP//AID//wCAB/8AgAf/AID//wCA//8AgP//AID//wCA//8AgP//AIAh/wCAC/8AgP//AID//wCAHP8AgBz/AID//wCA//8AgAf/AIAH/wCA//8AgP//AIAh/wCACf8AgA7/AIAMAAAAAP8AgAj/AID//wCA//8AgAP/AIAD/wCA//8AgP//AIAH/wCAB/8AgP//AID//wCABQAAAAAAAAAA/wCADf8AgAr/AIAL/wCA//8AgP//AIAH/wCAB/8AgP//AID//wCAB/8AgAr/AID//wCA//8AgCH/AIAH/wCAJf8AgP//AID//wCAI/8AgP//AID//wCAB/8AgAf/AID//wCA//8AgAf/AIAL/wCA//8AgP//AID//wCA//8AgP//AID//wCA//8AgCT/AID//wCA//8AgAr/AIAK/wCA//8AgP//AIAKAAAAAP8AgCj/AID//wCA//8AgP//AID//wCA//8AgCP/AIAM/wCA//8AgP//AIAN/wCADf8AgP//AID//wCADQAAAAAAAAAA/wCAEP8AgBH/AIAP/wCAEf8AgBAAAAAAAAAAAP8AgBT/AIAVAAAAAAAAAAD/AIAV/wCAFQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA//+sQcH5rEGA8KxBAHCsQQBwrEEIQKxBAACsQQAArEEAAKxBAQCsQQwArEEAAKxBAACsQYAArEHBmaxB//+sQQ%3D%3D%5D%5D%3E%3C/image%3E%0A%20%20%3Cmode%3E0%3C/mode%3E%0A%20%20%3Cinitcode%3E%3C%21%5BCDATA%5B/*Initialization%20Code*/%5D%5D%3E%3C/initcode%3E%0A%20%20%3Ccode%3E%3C%21%5BCDATA%5Bvar%20reader%2C%20picker%20%3D%20makeFilePicker%28%29%3B%0Apicker.init%28window%2C%20%22%22%2C%20picker.modeOpen%29%3B%0Apicker.open%28async%20res%20%3D%3E%20res%20%3D%3D%20picker.returnOK%20%26%26%20%28%0A%09%28reader%20%3D%20new%20FileReader%28%29%29.onload%20%3D%20%28%29%20%3D%3E%20gClipboard.write%28reader.result%29%2C%0A%09reader.readAsDataURL%28await%20File.createFromNsIFile%28picker.file%29%29%0A%29%29%3B%5D%5D%3E%3C/code%3E%0A%20%20%3Caccelkey%3E%3C%21%5BCDATA%5B%5D%5D%3E%3C/accelkey%3E%0A%20%20%3Chelp%3E%3C%21%5BCDATA%5B%5D%5D%3E%3C/help%3E%0A%20%20%3Cattributes/%3E%0A%3C/custombutton%3E

Есть только вопрос. У меня ссылок много очень, нельзя ли как то сделать, что бы использовать две таких кнопки одновременно? Что бы работали параллельно и независимо, а то ставлю две кнопки, но одни дублируют друг, друга

Непонятно. Клонируешь кнопок сколько угодно,
и редактируешь в каждой массив объектов data.
Вот, например, поставь эту.
Работает ли параллельно и независимо?
Дублирует ли что-нибудь?

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

Выделить код

Код:

custombutton://%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%0D%0A%3Ccustombutton%20xmlns%3Acb%3D%22http%3A//xsms.nm.ru/custombuttons/%22%3E%0A%20%20%3Cname%3E%u0443%u0440%u043B%u044B%3C/name%3E%0A%20%20%3Cimage%3E%3C%21%5BCDATA%5Bdata%3Aimage/x-icon%3Bbase64%2CAAABAAEAEBAAAAEAIABoBAAAFgAAACgAAAAQAAAAIAAAAAEAIAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABmMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz//+ZAP//mQD//5kA//+ZAP//mQD//5kA//+ZAP//mQD//5kA//+ZAP//mQD//5kA//+ZAP//mQD/ZjMz/2YzM///mQD//+K0///itP//4rT//+K0///itP//4rT//+K0///itP//4rT//+K0///itP//4rT//5kA/2YzM/9mMzP//5kA///itP///////////////////////////////////////////////////////+K0//+ZAP9mMzP/ZjMz//+ZAP//4rT////////////////////////////////////////////////////////itP//mQD/ZjMz/2YzM///mQD//+K0////////////////////////////////////////////////////////4rT//5kA/2YzM/9mMzP//5kA///itP///////////////////////////////////////////////////////+K0//+ZAP9mMzP/ZjMz//+ZAP//4rT//////////////////////5lmZv+ZZmb////////////////////////itP//mQD/ZjMz/2YzM///mQD//+K0/////////////////5lmZv+ZZmb/mWZm/5lmZv//////////////////4rT//5kA/2YzM/9mMzP//5kA///itP///////////5lmZv+ZZmb/mWZm/5lmZv+ZZmb/mWZm/////////////+K0//+ZAP9mMzP/ZjMz//+ZAP//4rT////////////////////////////////////////////////////////itP//mQD/ZjMz/2YzM///mQD//+K0////////////////////////////////////////////////////////4rT//5kA/2YzM/9mMzP//5kA///itP///////////////////////////////////////////////////////+K0//+ZAP9mMzP/ZjMz//+ZAP//4rT//+K0///itP//4rT//+K0///itP//4rT//+K0///itP//4rT//+K0///itP//mQD/ZjMz/2YzM///mQD//5kA//+ZAP//mQD//5kA//+ZAP//mQD//5kA//+ZAP//mQD//5kA//+ZAP//mQD//5kA/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/ZjMz/2YzM/9mMzP/AACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQ%3D%3D%5D%5D%3E%3C/image%3E%0A%20%20%3Cmode%3E0%3C/mode%3E%0A%20%20%3Cinitcode%3E%3C%21%5BCDATA%5B%28n%20%3D%3E%20%7B%0A%09var%20inBackground%20%3D%20false%3B%0A%09var%20data%20%3D%20%5B%7B%0A%09%09lab%3A%20%22example.com%22%2C%0A%09%09url%3A%20%22https%3A//example.com%22%2C%0A%09%09img%3A%20%22data%3Aimage/x-icon%3Bbase64%2CAAABAAEAEBAAAAEAIABoBAAAFgAAACgAAAAQAAAAIAAAAAEAIAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAD/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////tv///wAAZv8AAAD/AAAA/wAAAP8AAAD/25A6/wBmtv+2ZgD/////////////////AGa2/7ZmAP///////////7b///86AGb//9uQ///////////////////////b////OjqQ///bkP//////kNv//2YAOv///7b///////////+2////OgBm///bkP///////////////////////////2a2//+QOgD/2//b/wA6kP//tmb/////////////////tv///zoAZv//25D/////////////////////////////////AGa2/2ZmZv+QOgD////b/////////////////7b///8AAGb/AAAA/wAAAP8AAAD/ZgAA////tv///////////7b///86AGb//9uQ//////////////////////+2////OgBm///bkP////////////////////////////////86kNv/Ojo6/7ZmAP//////////////////////tv///zoAZv//25D///////////////////////////+Q2///kDo6////2/8AZrb//7Zm/////////////////7b///86AGb//9uQ////////////////////////////AGa2//+2Zv//////kNv//5A6Ov///9v///////////+2////AABm/wAAAP8AAAD/AAAA/wAAAP//tmb/Zrb//5A6AP///9v///////////8AZrb/25A6//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQAArEEAAKxBAACsQQAArEEAAKxB//+sQQ%3D%3D%22%0A%09%7D%2C%0A%09null%2C%20%7B%0A%09%09lab%3A%20%22%u041E%20%u0441%u0431%u043E%u0440%u043E%u0447%u043D%u043E%u0439%20%u043A%u043E%u043D%u0444%u0438%u0433%u0443%u0440%u0430%u0446%u0438%u0438%22%2C%0A%09%09url%3A%20%22about%3Abuildconfig%22%2C%0A%09%09img%3A%20%22chrome%3A//devtools/skin/images/settings.svg%22%0A%09%7D%2C%20%7B%0A%09%09lab%3A%20%22%u0411%u0438%u0431%u043B%u0438%u043E%u0442%u0435%u043A%u0430%22%2C%0A%09%09url%3A%20%22chrome%3A//browser/content/places/places.xhtml%22%2C%0A%09%09img%3A%20%22data%3Aimage/png%3Bcbfc%3Bbase64%2CiVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACU0lEQVQ4jZWSX0hTYRjGn7lmSV120V1JIqHQza6ELgyyCKKIHEwI0gJrSqk7xjmUbt8kCVRcQRgjVNJYhBdRhF55q0aLSjLL/TmKc+qcO8eztUkRTxdi5J8MX/jBC9/7e3nhe4B/l+n7fVSnPbgCwLTD3PalN+FoxotXmQ681O7i8G59U7oFTn66McVA1ddUC27t6oq4QEHWizdcHSBXB5hpx2tN4Mh/RSGQozchX/dA5odrofUFfF85ZXhwW29CvhDI2SDpLpQZAo0rAp0rAv50G4YYqAwx4yeNO2tk/GSgMpRuw9CKgF8X6DAEGjUXTkJ34zFH7SEuelNM9/MPmkQm69fQJG54i7UbHLUHNTceIuxEoeZGL8fsERo95JJjZ4wecqwirLnRPduAAvissAQlFCVceMZR+wyTXeRc1fYku8iRS2rChT5VwjFRij0AYPJZYYnU4fhyM14w1v6T0Vpyunwjc/VkrOPHcjOeByUU+ayw/P21Jp8VlriMmxw+Nc2YIIPnNhJtJYdOhBdlXN8sAwBsgDmu4B7f2eOMOMjJMjJ0mQxWrPVqLTlSvrCoQNgA89YMADlLCvo5WZelqpAzgvx41eC4I8UZQUYkctyRiit4IrApCwAwUYMDiWYMU1XIsfJk1ovxBQW+eRnd2QemzwzYdYakX3EFg4NnsXfLgrAThQkP3mYfmSbmZfSqEs50liCvswR5ESfOz8t4mu3K/bLswYhav02sJ2pwKCqjVW3ABVGKfaIYuTbAbAPMohi5faexf7YRF6MyWr9V4+C69xvjXKoSQWwbTQAAAABJRU5ErkJggg%3D%3D%22%0A%09%7D%5D%3B%0A%0A%09this.type%20%3D%20%22menu%22%3B%0A%09var%20popup%20%3D%20n%28%22menupopup%22%29%2C%20dummy%20%3D%20n%28%22menuitem%22%29%3B%0A%09popup.toggleAttribute%28%22context%22%29%3B%0A%09dummy.render%20%3D%20%28%29%20%3D%3E%20%7B%0A%09%09dummy.remove%28%29%3B%0A%09%09data.forEach%28%28o%2C%20ind%29%20%3D%3E%20%7B%0A%09%09%09if%20%28%21o%29%20return%20popup.append%28n%28%22menuseparator%22%29%29%3B%0A%09%09%09var%20menuitem%20%3D%20n%28%22menuitem%22%29%3B%0A%09%09%09menuitem.setAttribute%28%22label%22%2C%20o.lab%20%7C%7C%20o.url%29%3B%0A%09%09%09if%20%28o.img%29%0A%09%09%09%09menuitem.className%20%3D%20%22menuitem-iconic%22%2C%0A%09%09%09%09menuitem.setAttribute%28%22image%22%2C%20o.img%29%3B%0A%09%09%09menuitem.ind%20%3D%20ind%3B%0A%09%09%09popup.append%28menuitem%29%3B%0A%09%09%7D%29%3B%0A%09%09popup.setAttribute%28%22oncommand%22%2C%20%22tab%28event.target.ind%29%3B%22%29%3B%0A%09%09popup.tab%20%3D%20ind%20%3D%3E%20%7B%0A%09%09%09var%20%7Burl%2C%20params%7D%20%3D%20data%5Bind%5D%3B%0A%09%09%09var%20tab%20%3D%20gBrowser.addTrustedTab%28url%2C%20params%29%3B%0A%09%09%09if%20%28%21%28params%3F.inBackground%20%7C%7C%20inBackground%29%29%20gBrowser.selectedTab%20%3D%20tab%3B%0A%09%09%7D%0A%09%7D%0A%09popup.append%28dummy%29%3B%0A%09this.prepend%28popup%29%3B%0A%0A%09//this.onmouseover%20%3D%20%28%29%20%3D%3E%20this.open%20%3D%20true%3B%0A%0A%09this.removeAttribute%28%22tooltiptext%22%29%3B%0A%09var%20tt%20%3D%20this.appendChild%28n%28%22box%22%29%29.appendChild%28n%28%22tooltip%22%29%29%3B%0A%09tt.setAttribute%28%22onpopupshowing%22%2C%20%22return%20%21%28parentNode.parentNode.open%20%3D%20true%29%3B%22%29%3B%0A%09this.setAttribute%28%22tooltip%22%2C%20tt.id%20%3D%20_id%20+%20%22-tooltip%22%29%3B%0A%0A%7D%29%28nn%20%3D%3E%20document.createXULElement%28nn%29%29%3B%5D%5D%3E%3C/initcode%3E%0A%20%20%3Ccode%3E%3C%21%5BCDATA%5B/*CODE*/%5D%5D%3E%3C/code%3E%0A%20%20%3Caccelkey%3E%3C%21%5BCDATA%5B%5D%5D%3E%3C/accelkey%3E%0A%20%20%3Chelp%3E%3C%21%5BCDATA%5B%5D%5D%3E%3C/help%3E%0A%20%20%3Cattributes/%3E%0A%3C/custombutton%3E

ВВП пишет

В самой кнопке - стремно...

Понимаю. Но не обязательно же лезть в сам код.
Можно просто добавить в конец что-то типа

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

Выделить код

Код:

addEventListener("contextmenu", e => !this.disabled || e.ctrlKey || e.shiftKey || e.preventDefault(), false, this);

Отсутствует

 

№1568415-06-2021 20:32:09

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby

Dumby пишет

Можно просто добавить в конец что-то типа

Класс! А я то эту шнягу добавлял:
this.oncontextmenu =e=> { e.button && !e.ctrlKey && e.preventDefault() };
Благодарю!

Отсутствует

 

№1568515-06-2021 20:34:28

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

Re: Custom Buttons

Dumby
А кнопка, та что для base64, ее очень сложно для ucf сделать, можете адаптировать?

Отсутствует

 

№1568615-06-2021 21:49:00

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

Re: Custom Buttons

_zt пишет

А кнопка, та что для base64, ее очень сложно для ucf сделать, можете адаптировать?

Да вроде не сложно, хотелось бы думать

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

Выделить код

Код:

(async self => CustomizableUI.createWidget(self = {
	label: "64",
	tooltiptext: "64",
	id: "ucf-fileToBase64",
	localized: false,
	onCreated(btn) {
		btn._handleClick = this.pick;
		btn.setAttribute("image", "");
	},
	get copy() {
		delete this.copy;
		return this.copy = Cc["@mozilla.org/widget/clipboardhelper;1"]
			.getService(Ci.nsIClipboardHelper).copyString;
	},
	load() {
		self.copy(this.result);
	},
	async pick() {
		var win = this.ownerGlobal;
		var picker = win.makeFilePicker();
		picker.init(win, self.label, picker.modeOpen);
		if (await new Promise(picker.open) != picker.returnOK) return;

		var reader = new win.FileReader();
		reader.onload = self.load;
		reader.readAsDataURL(picker.domFileOrDirectory);
	}
}))();

Отсутствует

 

№1568715-06-2021 22:05:45

foxboy
Участник
 
Группа: Members
Зарегистрирован: 06-06-2015
Сообщений: 33
UA: Palemoon 24.0

Re: Custom Buttons

Есть у кого кнопка - переключиться на первую вкладку и закрыть все справа.

Отсутствует

 

№1568815-06-2021 23:11:32

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

Re: Custom Buttons

Dumby
Спасибо. Наконец то эта кнопочка вернулась, а то мучился с локальным конвертером, который какой-то мусор в код добавлял.
Иконка для темной темы -  … VORK5CYII=

Отсутствует

 

№1568916-06-2021 02:15:48

vv07
Участник
 
Группа: Members
Зарегистрирован: 07-11-2007
Сообщений: 672
UA: Firefox 54.0

Re: Custom Buttons

Dumby

Непонятно. Клонируешь кнопок сколько угодно,

Я сам уже понял это, хотя изначально почему то дублировалось содержимое.
Поэтому и убрал коммент, но вы успели прочитать.
По

проблемы при работе Compact Menu

причина ясна
Я ваш код вписал, а старый из секции код

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

/*CODE*/
const ids = [

    "file-menu",
    "edit-menu",
    "view-menu",
    "bookmarksMenu",
    "history-menu",
    "tools-menu",
    "helpMenu"
];
const id = "Compact-Menu-menupopup";
var popup = document.getElementById(id);
if (!popup) {
    popup = this.appendChild(document.createElement("menupopup"));
    const attrs = {
        id: id,
        position: "after_start",
        oncontextmenu: "event.preventDefault()",
        oncommand: "event.stopPropagation()",
    };
    for (var attr in attrs) popup.setAttribute(attr, attrs[attr]);

    ids.forEach(function(id) {
        var clonedMenu = document.getElementById(id).cloneNode(true);
        clonedMenu.hidden = false;
        popup.appendChild(clonedMenu);
    });
}
popup.openPopup(this);

не убрал.
Поэтому и возникала такая ситуация.
Не подскажете, как определять ID для других категорий меню и как их вписывать?
На старой кнопке, они видимо благодаря приведенному выше коду, сами подхватывались
скрытый текст
97e4fda1057e.jpg
, а сейчас самому шаманить придется :)
И задам вопрос тут походу уже, хотя похожая тема есть, но не совсем то что надо.
На старых версиях, работало отличное расширение Status-4-Evar,которое добавляло и индикатор загрузки сайтов.
Очень удобная вещь была. На новых увы, расширение не работает, а своего ничего нет.
Попробовал Load Progress Bar, но не то :( Тут вроде есть решение, но куда это пихать, что бы заработало, я не знаю.
Может кто подскажет? И будет ли это работать. Может есть альтернативное решение., что бы выглядело так
скрытый текст
53089e79b9ad.jpg

Отредактировано vv07 (16-06-2021 11:42:06)

Отсутствует

 

№1569016-06-2021 16:17:19

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
Не могу к этой панели dropmarker привязать . По ходу, это невозможно. хочу вправо сдвинуть..
65ychs6t.png
e9wb9dh2.png
В author : #editBMPanel_folderMenuList::part(dropmarker) {margin-right: -2px !important;} И ни фига...

Отредактировано ВВП (16-06-2021 16:31:03)

Отсутствует

 

№1569116-06-2021 20:55:54

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

Re: Custom Buttons

vv07 пишет

Не подскажете, как определять ID для других категорий меню и как их вписывать?

Не, для других категорий меню ничего не предусмотрено,
только те, что есть.
ID можно кнопкой Attributes Inspector посмотреть.
Или с помощью Browser Toolbox, но для такой задачи
это как из пушки по воробьям.
А раз есть стойкое предпочтение видеть пункты меню с иконками,
тогда можно специальный стиль составить (пример).


ВВП пишет

В author : #editBMPanel_folderMenuList::part(dropmarker) {margin-right: -2px !important;} И ни фига...

У меня это работает.
Ну, в два пикселя, конечно, вглядываться не захотел,
поставил сразу -18px и дропмаркер наглядно уехал вправо. Скриншот:

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

Выделить код

Код:



Отсутствует

 

№1569216-06-2021 21:39:33

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
Косяк был: влепил по умолчанию

скрытый текст
:host {
  -moz-appearance: -moz-menulist-arrow-button;
  width: 17px;
  height: 17px;
  filter: brightness(1200%);
  margin-right: 2px !important;
 
}

Вот он и замер намертво. На каждый ID надо отдельно лепить ::part(dropmarker) . Теперь , вроде, норм...

Отсутствует

 

№1569317-06-2021 05:08:40

Viatcheslav
Участник
 
Группа: Members
Откуда: г. Бобруйск, Беларусь
Зарегистрирован: 23-11-2016
Сообщений: 312
UA: Firefox 88.0

Re: Custom Buttons

Помогите, пожалуйста, создать кнопку для проматывания страницы сайта до конца вниз с необходимой периодичностью, например, раз в 10 мин. Для определённости, адрес страницы - https://babushkina.by/news/%d0%b2%d1%80%d0%b5%d0%bc%d1%8f-%d0%b2%d1%8b%d0%b1%d0%b8%d1%80%d0%b0%d1%82%d1%8c-%d0%bb%d1%83%d1%87%d1%88%d1%83%d1%8e/
Важно срабатывание без нажатия на кнопку.
Здесь есть варианты решения - https://techarks.ru/general/brauzeri/kak-avtomaticheski-prokrutit-veb-straniczu-vniz-ili-vverh-v-chrome-ili-firefox/
только я не очень в этом понимаю :blush: Будет ли работать приведенный вариант с консолью, и что означают числовые параметры:

Выделить код

Код:

«var scroll = setInterval (function () {window.scrollBy (0,1000);}, 2000);»

Помогите, пожалуйста :whiteflag: Спасибо :beer:

Отсутствует

 

№1569417-06-2021 09:54:48

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
А на этом скрине чем навели на drpomarker ? У меня atributes не показывает...
vdg37tbo.png
А нет кода - "открыть все папки в сидебаре" ? Историю , желательно...Т.е , открываю сидебар сразу пусть папки истории раскроются.

Отредактировано ВВП (17-06-2021 15:21:07)

Отсутствует

 

№1569517-06-2021 21:02:46

beggrr
Участник
 
Группа: Members
Зарегистрирован: 04-02-2014
Сообщений: 118
UA: Firefox 85.0

Re: Custom Buttons

Если popup окошко открыто с помощью window.open()
Можно каким то образом заставить ссылки в нем открываться в новых вкладках этого же окошка?
Сейчас я кликаю по ссылке и она открывается в текущей вкладке popup'а. А если выбираю в контекстном меню ссылки "Открыть в новой вкладке", то открывается во вкладке основного окна.

Отсутствует

 

№1569618-06-2021 08:36:37

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

Re: Custom Buttons

Viatcheslav пишет

Будет ли работать приведенный вариант с консолью, и что означают числовые параметры

Вот прямо так, в кавычках «», — не будет,
а без кавычек — будет, если скроллер это window.
Числовые параметры означают:


первый (1000) — на сколько пикселей прокрутить,
положительное число — вниз, отрицательное — вверх,
но лучше использовать window.scrollTo(0, window.scrollMaxY)


второй (2000) — с какой периодичностью в миллисекундах прокручивать,
в данном случае, каждые две секунды.
Десять минут это 10 * 60 * 1000 = 600000 миллисекунд.


ВВП пишет

А на этом скрине чем навели на drpomarker ? У меня atributes не показывает...

Сначала навёл на <menulist>, далее кнопками стрелок клавиатуры:


Ctrl+ArrowDown (Контрол и стрелка вниз) — переход к дочерним узлам,
первый будет, соответственно, <html:link>


Затем Ctrl+ArrowRight — переход к следующему соседнему узлу,
первый следующий будет <hbox>, а второй — уже тот самый <dropmarker>


Вообще, наверно, полезно будет перечитать имеющиеся
в кнопке сочетания клавиш (ссылки на страницы кнопки есть в начале кода).

Историю , желательно...Т.е , открываю сидебар сразу пусть папки истории раскроются.

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

Выделить код

Код:

(url => addEventListener("pageshow", e => {
	if (e.target.documentURI != url) return;

	var rn = e.target.getElementById("historyTree").view._rootNode;
	var ind = rn.childCount;
	while(ind--) {
		var node = rn.getChild(ind);
		if (node.containerOpen) continue;
		node.containerOpen = true;
		Services.xulStore.setValue(url, node.uri, "open", "true");
	}
}, false, document.getElementById("sidebar") || 1))(
	"chrome://browser/content/places/historySidebar.xhtml"
);

Отсутствует

 

№1569718-06-2021 10:35:41

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
Да, теперь вкурил . Кстати, этот дроп только так : (он и в  другом месте есть,типа, добавить закладку из контекста...) Короче, работает, но нет ли тут принципиального косяка?

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

Выделить код

Код:

vbox.panel-subview-body > vbox#editBookmarkPanelContent > vbox#editBookmarkPanelRows > vbox#editBMPanel_folderRow > hbox > menulist#editBMPanel_folderMenuList::part(dropmarker){margin-right: 4px !important;}

Отсутствует

 

№1569818-06-2021 12:16:52

Пострел
Участник
 
Группа: Members
Зарегистрирован: 08-04-2021
Сообщений: 51
UA: Firefox 89.0

Re: Custom Buttons

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

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

Выделить код

Код:

Services.search.addPolicyEngine({

    iconURL: "",
    chrome_settings_overrides: {
        search_provider: {
            name: "",
            search_url: "",
            search_form: "",
            search_url_get_params: "tt=mzl&q={searchTerms}",
            suggest_url: "",
            suggest_url_get_params: "type=list&q={searchTerms}"
        }
    }
});

вписать эти два поисковика,

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

https://cs.majento.ru/, и https://www.nigma.net.ru/

При выделении текста и поиске из контекстного меню, открываются страницы поисковиков,
без результатов поиска.

Отсутствует

 

№1569918-06-2021 15:13:22

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Пострел
В поисковике nigma так:

скрытый текст
"search_url": "https://www.nigma.net.ru/index.php?",
       "search_url_get_params": "query={searchTerms}"

Отсутствует

 

№1570018-06-2021 17:11:18

ВВП
Участник
 
Группа: Members
Зарегистрирован: 13-03-2021
Сообщений: 332
UA: Firefox 89.0

Re: Custom Buttons

Dumby
Последнее вроде.Хочу чтобы кнопка Undo не реагировала на закрытые или открытые окна...

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

Выделить код

Код:

// http://infocatcher.ucoz.net/js/cb/undoCloseTabs.js
// https://forum.mozilla-russia.org/viewtopic.php?id=56267
// https://github.com/Infocatcher/Custom_Buttons/tree/master/Undo_Close_Tabs

// Undo Close Tabs button for Custom Buttons
// (code for "initialization" section)

// (c) Infocatcher 2009-2015, 2017-2018
// version 0.3.3.1 - 2018-08-01

var options = {
    menuTemplate: [
        "closedWindows",
        "separator",
        "restoreClosedWindows",
        "clearClosedWindows",
        "separator",
        "closedTabs",
        "separator",
        "restoreClosedTabs",
        "clearClosedTabs",
        "separator",
        "clearAll",
        "separator",
        "restoreLastSession",
        "separator"
       
    ],
    showInTabContextMenu: false,
    /*
    menuTemplateTabContext: [ // like menuTemplate
        "closedTabs",
        "separator",
        "restoreClosedTabs",
        "clearClosedTabs"
    ],
    */
    windowItemTemplate: "(%count) %title",
    windowSelectedTabPrefix: "*",
    buttonTipTemplate: ["header", "title", "url", "closedAt"],
    itemTipTemplate: ["title", "url", "closedAt"],
    hideRestoreAllForSingleEntry: false,
    allowDeleteEntries: true,
    accesskeys: { // Empty string ("") to disable or string with possible values ("0123...", "abcd...")
        closedTabs: "",
        closedWindows: ""
    },
    accesskeyPostfix: " ", // <accesskey><postfix><label>
    openMenuOnMouseover: false,
    useMenu: false,
    rightClickToUndoCloseTab: false // Useful with "useMenu: true"
};

function _localize(sid) {
    var strings = {
        en: {
            restoreTab: "Restore the most recently closed tab",

            restoreAllTabs: "Restore all tabs",
            restoreAllTabsAccesskey: "t",
            clearTabsHistory: "Clear history of closed tabs",
            clearTabsHistoryAccesskey: "b",

            restoreAllWindows: "Restore all windows",
            restoreAllWindowsAccesskey: "w",
            clearWindowsHistory: "Clear history of closed windows",
            clearWindowsHistoryAccesskey: "d",

            clearAllHistory: "Clear all history",
            clearAllHistoryAccesskey: "C",

            restoreLastSession: "Restore last session",
            restoreLastSessionAccesskey: "s",

            deleteUndoEntry: "Delete",

            tabContextMenu: "Recently Closed Tabs",
            tabContextMenuAccesskey: "y",

            itemTip: "%ago ago, %date",
            day: "d"
        },
        ru: {
            restoreTab: "Восстановить последнюю закрытую вкладку",

            restoreAllTabs: "Восстановить все вкладки",
            restoreAllTabsAccesskey: "л",
            clearTabsHistory: "Очистить историю закрытых вкладок",
            clearTabsHistoryAccesskey: "д",

            restoreAllWindows: "Восстановить все окна",
            restoreAllWindowsAccesskey: "о",
            clearWindowsHistory: "Очистить историю закрытых окон",
            clearWindowsHistoryAccesskey: "н",

            clearAllHistory: "Очистить всю историю",
            clearAllHistoryAccesskey: "ч",

            restoreLastSession: "Восстановить последнюю сессию",
            restoreLastSessionAccesskey: "с",

            deleteUndoEntry: "Удалить",

            tabContextMenu: "Недавно закрытые вкладки",
            tabContextMenuAccesskey: "о",

            itemTip: "%ago назад, %date",
            day: "д"
        }
    };
    var locale = (function() {
        if("Services" in window && "locale" in Services) {
            var locales = Services.locale.requestedLocales // Firefox 64+
                || Services.locale.getRequestedLocales && Services.locale.getRequestedLocales();
            if(locales)
                return locales[0];
        }
        var prefs = "Services" in window && Services.prefs
            || Components.classes["@mozilla.org/preferences-service;1"]
                .getService(Components.interfaces.nsIPrefBranch);
        function pref(name, type) {
            return prefs.getPrefType(name) != prefs.PREF_INVALID ? prefs["get" + type + "Pref"](name) : undefined;
        }
        if(!pref("intl.locale.matchOS", "Bool")) { // Also see https://bugzilla.mozilla.org/show_bug.cgi?id=1414390
            var locale = pref("general.useragent.locale", "Char");
            if(locale && locale.substr(0, 9) != "chrome://")
                return locale;
        }
        return Components.classes["@mozilla.org/chrome/chrome-registry;1"]
            .getService(Components.interfaces.nsIXULChromeRegistry)
            .getSelectedLocale("global");
    })().match(/^[a-z]*/)[0];
    _localize = function(sid) {
        return strings[locale] && strings[locale][sid] || strings.en[sid] || sid;
    };
    return _localize.apply(this, arguments);
}

var JSON = "JSON" in window
    ? window.JSON
    : "nsIJSON" in Components.interfaces
        ? {
            parse: function(s) {
                return Components.classes["@mozilla.org/dom/json;1"]
                    .createInstance(Components.interfaces.nsIJSON)
                    .decode(s);
            }
        }
        : {
            parse: function(s) {
                return Components.utils.evalInSandbox("(" + s + ")", new Components.utils.Sandbox("about:blank"));
            }
        };

this.onclick = function(e) {
    if(e.target != this)
        return;
    if(e.button == 1 || e.button == 0 && (e.ctrlKey || e.shiftKey || e.altKey || e.metaKey))
        this.undoCloseTabsList.clearAllLists();
    else if(
        e.button == 0
        || e.button == 2 && !e.ctrlKey && !e.shiftKey && !e.altKey && !e.metaKey
            && this.undoCloseTabsList.options.rightClickToUndoCloseTab
    ) {
        if(
            e.button == 0 && !this.undoCloseTabsList.options.useMenu
            || e.button == 2 && this.undoCloseTabsList.options.rightClickToUndoCloseTab
        ) {
            if(this.undoCloseTabsList.closedTabCount)
                this.undoCloseTabsList.undoCloseTab();
            else
                this.undoCloseTabsList.drawUndoList() && this.undoCloseTabsList.showMenu(e);
        }
        // Allow use "command" section only from hotkey:
        e.preventDefault();
        e.stopPropagation();
    }
};
if(!this.hasOwnProperty("defaultContextId"))
    this.defaultContextId = this.getAttribute("context") || "custombuttons-contextpopup";
this.onmousedown = function(e) {
    if(e.target != this)
        return;
    if(this.undoCloseTabsList.options.useMenu) {
        if(e.button == 0)
            this.undoCloseTabsList.drawUndoList();
    }
    else if(e.button == 2) {
        var showCbMenu = e.ctrlKey || e.shiftKey || e.altKey || e.metaKey || !this.undoCloseTabsList.drawUndoList();
        this.setAttribute(
            "context",
            showCbMenu
                ? this.defaultContextId
                : this.undoCloseTabsList.mpId
        );
    }
};
this.onmouseover = function(e) {
    if(e.target != this)
        return;
    if(!this.disabled)
        this.undoCloseTabsList.updUI();
    this.undoCloseTabsList.options.useMenu && Array.prototype.some.call(
        this.parentNode.getElementsByTagName("*"),
        function(node) {
            if(
                node != this
                && node.namespaceURI == xulns
                // See https://github.com/Infocatcher/Custom_Buttons/issues/28
                //&& node.boxObject
                //&& node.boxObject instanceof Components.interfaces.nsIMenuBoxObject
                && "open" in node
                && node.open
                && node.getElementsByTagName("menupopup").length
                && this.undoCloseTabsList.drawUndoList()
            ) {
                node.open = false;
                this.open = true;
                return true;
            }
            return false;
        },
        this
    );
    if(
        this.undoCloseTabsList.options.openMenuOnMouseover
        && this.undoCloseTabsList.drawUndoList()
    )
        this.undoCloseTabsList.openMenu();
};

this.undoCloseTabsList = {
    button: this,
    options: options,
    mpId: this.id + "-context",
    cmId: this.id + "-contextSub",
    tcmId: this.id + "-tabContextMenu",
    tipId: this.id + "-tooltip",
    errPrefix: "[Custom Buttons :: Undo Close Tabs List]: ",
    get mp() {
        var btn = this.button;
        var mp = btn.getElementsByTagName("menupopup");
        mp = mp.length && mp[0];
        mp && mp.parentNode.removeChild(mp);
        mp = this.createElement("menupopup", {
            id: this.mpId,
            onclick: "this.parentNode.undoCloseTabsList.checkForMiddleClick(event);",
            onpopupshowing: "if(event.target == this) document.popupNode = this.parentNode;",
            onpopuphidden: "if(event.target == this) document.popupNode = null;"
        });
        if(this.cm)
            mp.setAttribute("context", this.cmId);
        var tb = btn.parentNode;
        if(
            this.options.useMenu
            && tb.getAttribute("orient") == "vertical"
        ) {
            // https://addons.mozilla.org/firefox/addon/vertical-toolbar/
            var isRight = tb.parentNode.getAttribute("placement") == "right";
            mp.setAttribute("position", isRight ? "start_before" : "end_before");
        }
        delete this.mp;
        return this.mp = btn.appendChild(mp);
    },
    get useCentextMenu() {
        delete this.useCentextMenu;
        return this.useCentextMenu = this.options.allowDeleteEntries
            && ("forgetClosedTab" in this.ss || "forgetClosedWindow" in this.ss);
    },
    get cm() {
        delete this.cm;
        if(!this.useCentextMenu)
            return this.cm = null;
        var cm = document.getElementById(this.cmId);
        cm && cm.parentNode.removeChild(cm);
        cm = this.createElement("menupopup", {
            id: this.cmId,
            onpopupshowing: "return this.undoCloseTabsList.canDeleteUndoEntry(this.triggerNode || document.popupNode);"
        });
        var mi = this.createElement("menuitem", {
            oncommand: "this.parentNode.undoCloseTabsList.deleteUndoEntry(this.parentNode.triggerNode || document.popupNode);",
            label: _localize("deleteUndoEntry"),
            closemenu: "single"
        });
        cm.appendChild(mi);
        cm.undoCloseTabsList = this;
        return this.cm = document.getElementById("mainPopupSet").appendChild(cm);
    },
    get cbMenu() {
        var cbPopup = document.getElementById(this.button.defaultContextId);
        if(!cbPopup) {
            Components.utils.reportError(this.errPrefix + "cb menu not found");
            return this.cbMenu = null;
        }
        cbPopup = cbPopup.cloneNode(true);
        var id = "-" + this.button.id.match(/\d*$/)[0] + "-cloned";
        cbPopup.id += id;
        Array.prototype.slice.call(cbPopup.getElementsByAttribute("id", "*")).forEach(function(node) {
            node.id += id;
        });
        
        menu.appendChild(cbPopup);
        cbPopup.setAttribute(
            "onpopupshowing",
            '\
            var btn = document.popupNode = this.parentNode.parentNode.parentNode\n\
                .undoCloseTabsList.button;\n\
            custombutton.setContextMenuVisibility(btn);'
        );
        delete this.cbMenu;
        return this.cbMenu = menu;
    },
    get ss() {
        delete this.ss;
        return this.ss = "nsISessionStore" in Components.interfaces
            ? (
                Components.classes["@mozilla.org/browser/sessionstore;1"]
                || Components.classes["@mozilla.org/suite/sessionstore;1"]
            ).getService(Components.interfaces.nsISessionStore)
            : SessionStore; // Firefox 61+ https://bugzilla.mozilla.org/show_bug.cgi?id=1450559
    },
    get appInfo() {
        delete this.appInfo;
        return this.appInfo = Components.classes["@mozilla.org/xre/app-info;1"]
            .getService(Components.interfaces.nsIXULAppInfo);
    },
    get appVersion() {
        delete this.appVersion;
        return this.appVersion = parseFloat(this.appInfo.version);
    },
    get appName() {
        delete this.appName;
        return this.appName = this.appInfo.name;
    },

    init: function() {
        window.addEventListener("TabClose",       this, false);
        window.addEventListener("SSTabRestoring", this, false);
        window.addEventListener("unload",         this, false);
        if(this.appName == "SeaMonkey") // No SSTab* events in SeaMonkey
            window.addEventListener("TabOpen", this, false);
        setTimeout(function(_this) {
            _this.mp.addEventListener("DOMMenuItemActive",   _this, false);
            _this.mp.addEventListener("DOMMenuItemInactive", _this, false);
            _this.initTooltip();
        }, 50, this);
        this.addPbExitObserver(true);
        this.updUIGlobal();
        if(this.options.showInTabContextMenu) setTimeout(function(_this) {
            _this.initTabContext();
        }, 100, this);
    },
    initTabContext: function() {
        var origMi = this.tabContextUndoClose;
        if(!origMi) {
            LOG("Can't find \"Undo Close Tab\" item in tab context menu");
            return;
        }
        var menu = document.getElementById(this.tcmId);
        menu && menu.parentNode.removeChild(menu); // For SeaMonkey
        menu = this.createElement("menu", {
            id: this.tcmId,
            label: _localize("tabContextMenu"),
            accesskey: _localize("tabContextMenuAccesskey"),
            tooltip: this.tipId,
            popupsinherittooltip: "true"
        });
        menu.undoCloseTabsList = this;
        menu.onclick = function(e) {
            if(e.target != this)
                return;
            if(e.button == 1 || e.button == 0 && (e.ctrlKey || e.shiftKey || e.altKey || e.metaKey)) {
                if(this.undoCloseTabsList.closedTabCount) {
                    this.undoCloseTabsList.undoCloseTab();
                    closeMenus(this);
                }
            }
        };
        var origMp = this.mp;
        var mp = origMp.cloneNode(true);
        mp.id = this.button.id + "-tabContext";
        var _this = this;
        function drawUndoList() {
            var ok = false;
            var opts = _this.options;
            var origTemplate = opts.menuTemplate;
            opts.menuTemplate = opts.menuTemplateTabContext || origTemplate;
            _this.mp = mp;
            try {
                ok = _this.drawUndoList();
            }
            catch(e) {
                Components.utils.reportError(e);
            }
            opts.menuTemplate = origTemplate;
            _this.mp = origMp;
            return ok;
        }
        function updMenu() {
            if(drawUndoList())
                menu.removeAttribute("disabled");
            else
                menu.setAttribute("disabled", "true");
        }
        mp._updatePopup = function(e) {
            if(e.target != this)
                return;
            document.popupNode = _this.button;
            drawUndoList();
        };
        mp.setAttribute("onpopupshowing", "this._updatePopup(event);");
        mp.onclick = function(e) {
            _this.checkForMiddleClick(e, updMenu);
        };
        menu.appendChild(mp);
        addEventListener("popupshown", function(e) {
            if(e.target == e.currentTarget)
                setTimeout(updMenu, 0); // Pseudo async
        }, false, origMi.parentNode);
        addEventListener("DOMMenuItemActive",   this, false, mp);
        addEventListener("DOMMenuItemInactive", this, false, mp);
        origMi.parentNode.insertBefore(menu, origMi.nextSibling);
        origMi.setAttribute("hidden", "true");
    },
    initTooltip: function() {
        var tip = document.getElementById(this.tipId);
        tip && tip.parentNode.removeChild(tip);
        tip = this.tip = this.createElement("tooltip", {
            id: this.tipId,
            orient: "vertical",
            onpopupshowing: "return this.undoCloseTabsList.updTooltip(this, document.tooltipNode);",
            onpopuphiding: "this.cancelUpdateTimer();",
            style: "padding: 0;"
        });
        tip.undoCloseTabsList = this;
        tip._updateTimer = 0;
        tip.initUpdateTimer = function(fn, context) {
            if(this._updateTimer)
                clearInterval(this._updateTimer);
            this._updateTimer = setInterval(function() {
                fn.call(context);
            }, 1000);
        };
        tip.cancelUpdateTimer = function() {
            if(this._updateTimer) {
                clearInterval(this._updateTimer);
                this._updateTimer = 0;
            }
        };
        var btn = this.button;
        btn.removeAttribute("tooltiptext");
        btn.setAttribute("tooltip", this.tipId);
        btn.setAttribute("popupsinherittooltip", "true");
        document.getElementById("mainPopupSet").appendChild(tip);
        if(this.appVersion >= 61 && "getAnonymousElementByAttribute" in document) {
            var label = document.getAnonymousElementByAttribute(tip, "class", "tooltip-label");
            label && label.remove();
        }
    },
    _hasPbExitObserver: false,
    addPbExitObserver: function(add) {
        if(add == this._hasPbExitObserver || !("Services" in window))
            return;
        this._hasPbExitObserver = add;
        if(add)
            Services.obs.addObserver(this, "last-pb-context-exited", false);
        else
            Services.obs.removeObserver(this, "last-pb-context-exited");
    },
    destroy: function() {
        window.removeEventListener("TabClose",       this, false);
        window.removeEventListener("SSTabRestoring", this, false);
        window.removeEventListener("unload",         this, false);
        if(this.appName == "SeaMonkey")
            window.removeEventListener("TabOpen", this, false);
        this.mp.removeEventListener("DOMMenuItemActive",   this, false);
        this.mp.removeEventListener("DOMMenuItemInactive", this, false);
        this.addPbExitObserver(false);
        var menu = document.getElementById(this.tcmId);
        if(menu) {
            menu.parentNode.removeChild(menu);
            this.tabContextUndoClose.removeAttribute("hidden");
        }
        var tip = this.tip;
        tip && tip.parentNode && tip.parentNode.removeChild(tip);
    },
    handleEvent: function(e) {
        switch(e.type) {
            case "TabClose":
            case "SSTabRestoring":
            case "TabOpen":
                setTimeout(function(_this) {
                    _this.updUI();
                }, 0, this);
            break;
            case "DOMMenuItemActive":
            case "DOMMenuItemInactive":
                if(!("XULBrowserWindow" in window))
                    break;
                XULBrowserWindow.setOverLink(
                    e.type == "DOMMenuItemActive"
                        ? (e.target.getAttribute("cb_urlDecoded") || "")
                            .replace(/ \n/g, ", ")
                        : "",
                    null
                );
            break;
            case "unload":
                this.updUIGlobal();
                this.destroy();
        }
    },
    observe: function(subject, topic, data) {
        if(topic == "last-pb-context-exited") {
            setTimeout(function(_this) {
                _this.updUI();
            }, 25, this);
        }
    },

    createElement: function(name, attrs) {
        var node = document.createElementNS(xulns, name);
        if(attrs) for(var attrName in attrs) if(attrs.hasOwnProperty(attrName))
            node.setAttribute(attrName, attrs[attrName]);
        return node;
    },
    get tabContextUndoClose() {
        return document.getElementById("context_undoCloseTab")
            || document.getElementById("tabContextUndoCloseTab") // Firefox 2.0
            || document.getAnonymousElementByAttribute(gBrowser, "tbattr", "tabbrowser-undoclosetab"); // SeaMonkey
    },
    get closedWindowCount() {
        if(!("getClosedWindowCount" in this.ss)) {
            delete this.closedWindowCount;
            return this.closedWindowCount = 0;
        }
        this.__defineGetter__("closedWindowCount", function() {
            return this.ss.getClosedWindowCount();
        });
        return this.closedWindowCount;
    },
    get closedTabCount() {
        return this.ss.getClosedTabCount(window);
    },
    undoCloseTab: function(i) {
        if("undoCloseTab" in window) // Firefox 2.0+
            undoCloseTab(i);
        else // SeaMonkey
            gBrowser.undoCloseTab(i);
    },
    clearUndoTabsList: function() {
        var closedTabCount = this.closedTabCount;
        if(!closedTabCount)
            return;
        if("forgetClosedTab" in this.ss) // Gecko 1.9.2+
            while(closedTabCount--)
                this.ss.forgetClosedTab(window, 0);
        else {
            // Doesn't work in SeaMonkey
            const pName = "browser.sessionstore.max_tabs_undo";
            let val = cbu.getPrefs(pName);
            cbu.setPrefs(pName, 0);
            cbu.setPrefs(pName, val);
        }
        this.updUIGlobal();
    },
    clearUndoWindowsList: function() {
        var closedWindowCount = this.closedWindowCount;
        if(!closedWindowCount)
            return;
        if("forgetClosedWindow" in this.ss) // Gecko 1.9.2+
            while(closedWindowCount--)
                this.ss.forgetClosedWindow(0);
        else
            this.ss.setWindowState(window, '{"windows":[{}],"_closedWindows":[]}', false);
        this.updUIGlobal();
    },
    clearAllLists: function() {
        this.clearUndoTabsList();
        this.clearUndoWindowsList();
    },
    canDeleteUndoEntry: function(mi) {
        switch(mi.getAttribute("cb_type")) {
            case "tab":    return "forgetClosedTab"    in this.ss;
            case "window": return "forgetClosedWindow" in this.ss;
        }
        return false;
    },
    deleteUndoEntry: function(mi) {
        var i = +mi.getAttribute("cb_index");
        if(mi.getAttribute("cb_type") == "window") {
            this.ss.forgetClosedWindow(i);
            this.updUIGlobal();
        }
        else {
            this.ss.forgetClosedTab(window, i);
            this.updUI();
        }
        this.drawUndoList();
    },
    showMenu: function(e, isContext, mp) {
        var btn = this.button;
        document.popupNode = btn.ownerDocument.popupNode = btn;
        if(!mp)
            mp = this.mp;
        if("openPopupAtScreen" in mp)
            mp.openPopupAtScreen(e.screenX, e.screenY, isContext);
        else
            mp.showPopup(btn, e.screenX, e.screenY, isContext ? "context" : "popup", null, null);
    },
    openMenu: function() {
        var mp = this.mp;
        if("openPopup" in mp)
            mp.openPopup(this.button, "after_start");
        else
            mp.showPopup(this.button, -1, -1, "popup", "bottomleft", "topleft");
    },
    drawUndoList: function() {
        var mp = this.mp;

        var wc = this.closedWindowCount;
        var tc = this.closedTabCount;
        var ss = this.ss;
        var canRestoreLastSession = "restoreLastSession" in ss && ss.canRestoreLastSession
        if(!wc && !tc && !canRestoreLastSession) {
            mp.textContent = "";
            mp.hidePopup();
            return false;
        }

        this._undoWindowItems = wc && JSON.parse(ss.getClosedWindowData());
        this._undoTabItems    = tc && JSON.parse(ss.getClosedTabData(window));
        var df = document.createDocumentFragment();

        this.options.menuTemplate.forEach(function(sid, indx, arr) {
            switch(sid) {
                case "closedWindows":
                    wc && this.addUndoWindowsList(df);
                break;
                case "restoreClosedWindows":
                    wc > this.options.hideRestoreAllForSingleEntry
                    && df.appendChild(this.createElement("menuitem", {
                        label: _localize("restoreAllWindows"),
                        accesskey: _localize("restoreAllWindowsAccesskey"),
                        oncommand: "for(var i = 0; i < " + this._undoWindowItems.length + "; ++i) undoCloseWindow();"
                    }));
                break;
                case "clearClosedWindows":
                    wc && df.appendChild(this.createElement("menuitem", {
                        label: _localize("clearWindowsHistory"),
                        accesskey: _localize("clearWindowsHistoryAccesskey"),
                        oncommand: "this.parentNode.parentNode.undoCloseTabsList.clearUndoWindowsList();"
                    }));
                break;
                case "closedTabs":
                    tc && this.addUndoTabsList(df);
                break;
                case "restoreClosedTabs":
                    tc > this.options.hideRestoreAllForSingleEntry
                    && df.appendChild(this.createElement("menuitem", {
                        label: _localize("restoreAllTabs"),
                        accesskey: _localize("restoreAllTabsAccesskey"),
                        oncommand: "for(var i = 0; i < " + this._undoTabItems.length + "; ++i) this.parentNode.parentNode.undoCloseTabsList.undoCloseTab();"
                    }));
                break;
                case "clearClosedTabs":
                    tc && df.appendChild(this.createElement("menuitem", {
                        label: _localize("clearTabsHistory"),
                        accesskey: _localize("clearTabsHistoryAccesskey"),
                        oncommand: "this.parentNode.parentNode.undoCloseTabsList.clearUndoTabsList();"
                    }));
                break;
                case "clearAll":
                    (
                        wc && tc
                        || wc && arr.indexOf("clearClosedWindows") == -1
                        || tc && arr.indexOf("clearClosedTabs") == -1
                    )
                    && df.appendChild(this.createElement("menuitem", {
                        label: _localize("clearAllHistory"),
                        accesskey: _localize("clearAllHistoryAccesskey"),
                        oncommand: "this.parentNode.parentNode.undoCloseTabsList.clearAllLists();"
                    }));
                break;
                case "restoreLastSession": // Gecko 2.0+
                    canRestoreLastSession && df.appendChild(this.createElement("menuitem", {
                        label: _localize("restoreLastSession"),
                        accesskey: _localize("restoreLastSessionAccesskey"),
                        oncommand: "this.parentNode.parentNode.undoCloseTabsList.ss.restoreLastSession();"
                    }));
                
                break;
                case "separator":
                    if(df.hasChildNodes() && df.lastChild.localName != "menuseparator")
                        df.appendChild(document.createElementNS(xulns, "menuseparator"));
                break;
                default:
                    Components.utils.reportError(this.errPrefix + 'Invalid template entry: "' + sid + '"');
            }
        }, this);

        while(df.hasChildNodes() && df.lastChild.localName == "menuseparator")
            df.removeChild(df.lastChild);

        this._undoWindowItems = this._undoTabItems = null;

        mp.textContent = "";
        if(!df.hasChildNodes()) {
            mp.hidePopup();
            return false;
        }
        mp.appendChild(df);
        return true;
    },
    addUndoWindowsList: function(undoPopup) {
        // Based on code from chrome://browser/content/browser.js
        // Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.9.3a1pre) Gecko/20090824 Minefield/3.7a1pre

        var keys = this.options.accesskeys.closedWindows;
        this._undoWindowItems.forEach(function(undoItem, i) {
            var tabs = undoItem.tabs;
            var [key, keyPrefix] = this.getKey(keys, i);
            var title = undoItem.title;
            var selectedTab = tabs[undoItem.selected - 1];
            var urls = [];
            tabs.forEach(function(tab) {
                if(!tab.entries || !tab.entries.length) // Can be [] for about:blank
                    return;
                var url = this.convertURI(tab.entries[tab.index - 1].url, 120);
                var selectedPrefix = tab == selectedTab && tabs.length > 1
                    ? this.options.windowSelectedTabPrefix
                    : "";
                urls.push(selectedPrefix + url);
            }, this);
            var url = urls.join(" \n");
            var mi = this.createElement("menuitem", {
                label: keyPrefix + this.options.windowItemTemplate
                    .replace("%title", title)
                    .replace("%count", tabs.length),
                accesskey: key,
                "class": "menuitem-iconic bookmark-item menuitem-with-favicon",
                oncommand: "undoCloseWindow(" + i + ");",
                cb_url: url,
                cb_urlDecoded: this.convertURI(url),
                cb_closedAt: undoItem.closedAt || 0,
                cb_index: i,
                cb_type: "window"
            });
            if(this.cm)
                mi.setAttribute("context", this.cmId);
            var icon = selectedTab.image || selectedTab.attributes && selectedTab.attributes.image;
            if(icon)
                mi.setAttribute("image", this.cachedIcon(icon));
            if(i == 0)
                mi.setAttribute("key", "key_undoCloseWindow");
            undoPopup.appendChild(mi);
        }, this);
    },
    addUndoTabsList: function(undoPopup) {
        // Based on code from chrome://browser/content/browser.js
        // Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.9.3a1pre) Gecko/20090824 Minefield/3.7a1pre

        var keys = this.options.accesskeys.closedTabs;
        this._undoTabItems.forEach(function(undoItem, i) {
            var state = undoItem.state;
            var [key, keyPrefix] = this.getKey(keys, i);
            var title = undoItem.title;
            var url = state && state.entries && state.entries[state.index - 1].url || "";
            var mi = this.createElement("menuitem", {
                label: keyPrefix + title,
                accesskey: key,
                class: "menuitem-iconic bookmark-item menuitem-with-favicon",
                oncommand: "this.parentNode.parentNode.undoCloseTabsList.undoCloseTab(" + i + ");",
                cb_url: url,
                cb_urlDecoded: this.convertURI(url),
                cb_closedAt: undoItem.closedAt || 0,
                cb_index: i,
                cb_type: "tab"
            });
            if(
                state
                && "attributes" in state
                && "privateTab-isPrivate" in state.attributes
            ) // https://addons.mozilla.org/addon/private-tab/
                mi.setAttribute("privateTab-isPrivate", "true");
            if(this.cm)
                mi.setAttribute("context", this.cmId);
            var image = undoItem.image // Firefox
                || state && state.attributes && state.attributes.image // SeaMonkey
                || state && state.xultab
                    && /(?:^| )image=(\S+)/.test(state.xultab)
                    && decodeURI(RegExp.$1); // Only Firefox 2.0 ?
            if(image)
                mi.setAttribute("image", this.cachedIcon(image));
            if(i == 0)
                mi.setAttribute("key", "key_undoCloseTab");
            undoPopup.appendChild(mi);
        }, this);
    },
    getKey: function(keys, i) {
        var key = keys && keys.charAt(i % keys.length);
        var keyPrefix = keys && (key + this.options.accesskeyPostfix);
        return [key, keyPrefix];
    },
    checkForMiddleClick: function(e, upd) {
        var mi = e.target;
        if(
            "doCommand" in mi
            && e.button == 1
            && mi.parentNode == e.currentTarget
        ) {
            mi.doCommand();
            if(upd)
                upd();
            else
                this.drawUndoList();
        }
    },
    crop: function(s, crop) {
        if(crop == undefined)
            crop = 500;
        if(s.length <= crop)
            return s;
        var start = Math.round(crop*0.6);
        return s.substr(0, start) + "…" + s.substr(start - crop);
    },
    convertURI: function(uri, crop) {
        if(!uri || uri.indexOf("\n") != -1)
            return uri;
        try {
            uri = "losslessDecodeURI" in window
                ? losslessDecodeURI(makeURI(uri))
                : decodeURI(uri);
        }
        catch(e) {
            Components.utils.reportError(e);
        }
        return this.crop(uri, crop);
    },
    cachedIcon: function(src) {
        src = src.replace(/[&#]-moz-resolution=\d+,\d+$/, ""); // Firefox 22+
        if(
            !/^https?:/.test(src)
            // IDN, see https://bugzilla.mozilla.org/show_bug.cgi?id=311045
            || /^https?:\/\/[^.:\/]+\.[^a-z0-9-]+(?:\/|$)/.test(src)
            || this.appName == "SeaMonkey" && this.appVersion <= 2
            || this.appName == "Firefox"   && this.appVersion <= 3.5
        )
            return src;
        return "moz-anno:favicon:" + src; // https://bugzilla.mozilla.org/show_bug.cgi?id=467828
    },
    updUI: function() {
        var tabsCount = this.closedTabCount;
        var dis = !tabsCount && !this.closedWindowCount;
        if(
            dis
            && this.options.useMenu
            && this.options.menuTemplate.indexOf("restoreLastSession") != -1
            && "restoreLastSession" in this.ss && this.ss.canRestoreLastSession
        )
            dis = false;
        this.button.disabled = dis;
    },
    updTooltip: function(tip, tn) {
        var template, header, title, url, closedAt;
        if(tn == this.button) {
            template = this.options.buttonTipTemplate;
            header = _localize("restoreTab");
            let undoTabItems = JSON.parse(this.ss.getClosedTabData(window));
            if(undoTabItems.length) {
                let lastItem = undoTabItems[0];
                title = lastItem.title;
                url = lastItem.state && lastItem.state.entries
                    && lastItem.state.entries[lastItem.state.index - 1].url;
                closedAt = lastItem.closedAt || 0;
            }
        }
        else if(tn.hasAttribute("cb_index")) {
            template = this.options.itemTipTemplate;
            title = tn.getAttribute("label");
            url = tn.getAttribute("cb_url");
            closedAt = +tn.getAttribute("cb_closedAt");
        }
        else {
            return false;
        }

        var tipData = this.getTooltipData(template, header, title, url, closedAt);
        tip.textContent = "";
        tip.appendChild(tipData);
        if(closedAt && template.indexOf("closedAt") != -1) {
            tip.initUpdateTimer(function() {
                var tipData = this.getTooltipData(template, header, title, url, closedAt);
                if(tipData.textContent != tip.textContent) {
                    tip.textContent = "";
                    tip.appendChild(tipData);
                }
            }, this);
        }
        return tip.hasChildNodes();
    },
    getTooltipData: function(template, header, title, url, closedAt) {
        var df = document.createDocumentFragment();
        var hasHeader = header && template.indexOf("header") != -1;
        function item(key, val) {
            var lbl = document.createElementNS(xulns, "label");
            lbl.className = "cb-" + key;
            //lbl.setAttribute("value", val);
            lbl.textContent = val;
            lbl.setAttribute("maxwidth", "450"); // Trick to restore right border for long lines
            if(key == "closedAt" || hasHeader && key != "header")
                lbl.style.color = "grayText";
            return df.appendChild(lbl);
        }
        template.forEach(function(key) {
            switch(key) {
                case "header":
                    if(header)
                        item(key, header);
                break;
                case "title":
                    if(title && title != url)
                        item(key, title);
                break;
                case "url":
                    if(url)
                        item(key, this.convertURI(url));
                break;
                case "closedAt":
                    if(!closedAt)
                        break;
                    let dt = Math.round(Math.max(0, Date.now() - closedAt)/1000);
                    let days = Math.floor(dt/24/3600);
                    dt -= days*24*3600;
                    let d = new Date((dt + new Date(dt).getTimezoneOffset()*60)*1000);
                    let m = d.getMinutes();
                    let ts = d.getHours() + ":" + (m > 9 ? m : "0" + m);
                    if(days)
                        ts = days + _localize("day") + " " + ts;
                    let tsTip = _localize("itemTip")
                        .replace("%ago", ts)
                        .replace("%date", new Date(closedAt).toLocaleString());
                    item(key, tsTip);
            }
        }, this);
        return df;
    },
    get wm() {
        delete this.wm;
        return this.wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
            .getService(Components.interfaces.nsIWindowMediator);
    },
    updUIGlobal: function() {
        var isSeaMonkey = this.appName == "SeaMonkey";
        var ws = this.wm.getEnumerator(isSeaMonkey ? null : "navigator:browser");
        const id = this.button.id;
        while(ws.hasMoreElements()) {
            let win = ws.getNext();
            if(!isSeaMonkey || this.isBrowserWindow(win)) {
                let btn = win.document.getElementById(id);
                if(btn && "undoCloseTabsList" in btn) {
                    let ucl = btn.undoCloseTabsList;
                    ucl.ensureSessionsInitialized(ucl.updUI, ucl);
                }
            }
        }
    },
    isBrowserWindow: function(win) {
        var loc = window.location.href;
        return loc == "chrome://browser/content/browser.xul"
            || loc == "chrome://navigator/content/navigator.xul";
    },
    ensureSessionsInitialized: function(callback, context) {
        var _this = this;
        var stopTime = Date.now() + 3e3;
        (function ensureInitialized() {
            try {
                _this.ss.getClosedTabCount(window);
                callback.call(context);
                return;
            }
            catch(e) {
                if(Date.now() > stopTime) {
                    Components.utils.reportError(
                        _this.errPrefix
                        + "Can't initialize: nsISessionStore.getClosedTabCount() failed"
                    );
                    Components.utils.reportError(e);
                    return;
                }
            }
            setTimeout(ensureInitialized, 50);
        })();
    }
};

if(!this.undoCloseTabsList.options.useMenu && this.undoCloseTabsList.useCentextMenu) {
    this.oncontextmenu = function(e) {
        if(
            e.target != this
            || e.ctrlKey || e.shiftKey || e.altKey || e.metaKey
            || !this.undoCloseTabsList.mp.hasChildNodes()
        )
            return;
        e.preventDefault();
        this.undoCloseTabsList.showMenu(e); // Show menu without "context" flag
    };
}
if(this.undoCloseTabsList.options.rightClickToUndoCloseTab) {
    this.oncontextmenu = function(e) {
        if(e.target == this && !e.ctrlKey && !e.shiftKey && !e.altKey && !e.metaKey)
            e.preventDefault();
    };
}

this.disabled = true;
setTimeout(function(_this) {
    _this.undoCloseTabsList.init();
}, 0, this);

//===================
// Styles
// Used icons from Undo Closed Tabs Button extension

// Styles can't override hardcoded icon
if( // Remove icon only if nsIStyleSheetService works on-the-fly (Firefox 3.0+)
    !Components.ID("{41d979dc-ea03-4235-86ff-1e3c090c5630}")
        .equals(Components.interfaces.nsIStyleSheetService)
) {
    let icon = this.icon
        || this.ownerDocument.getAnonymousElementByAttribute(this, "class", "toolbarbutton-icon");
    if(icon)
        icon.src = "";
    else
        this.image = "";
}

var cssStr = '\
    @namespace url("http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul");\n\
    @-moz-document url("%windowURL%") {\n\
        %button% {\n\
            list-style-image: url("") !important;\n\
            -moz-image-region: rect(0, 24px, 24px, 0) !important;\n\
        }\n\
        %button%:hover {\n\
            -moz-image-region: rect(0, 48px, 24px, 24px) !important;\n\
        }\n\
        %button%[disabled="true"] {\n\
            -moz-image-region: rect(0, 72px, 24px, 48px) !important;\n\
        }\n\
        toolbar[iconsize="small"] %button% {\n\
            -moz-image-region: rect(24px, 16px, 40px, 0) !important;\n\
        }\n\
        toolbar[iconsize="small"] %button%:hover {\n\
            -moz-image-region: rect(24px, 32px, 40px, 16px) !important;\n\
        }\n\
        toolbar[iconsize="small"] %button%[disabled="true"] {\n\
            -moz-image-region: rect(24px, 48px, 40px, 32px) !important;\n\
        }\n\
    }'
    .replace(/%windowURL%/g, window.location.href)
    .replace(/%button%/g, "#" + this.id);
var cssURI = this.cssURI = Components.classes["@mozilla.org/network/io-service;1"]
    .getService(Components.interfaces.nsIIOService)
    .newURI("data:text/css," + encodeURIComponent(cssStr), null, null);
var sss = this.sss = Components.classes["@mozilla.org/content/style-sheet-service;1"]
    .getService(Components.interfaces.nsIStyleSheetService);
if(!sss.sheetRegistered(cssURI, sss.USER_SHEET))
    sss.loadAndRegisterSheet(cssURI, sss.USER_SHEET);


this.onDestroy = function(reason) {
    this.undoCloseTabsList.destroy();
    if(reason == "update" || reason == "delete") {
        let sss = this.sss;
        let cssURI = this.cssURI;
        if(sss.sheetRegistered(cssURI, sss.USER_SHEET))
            sss.unregisterSheet(cssURI, sss.USER_SHEET);
    }
};
if(this.undoCloseTabsList.options.useMenu) {
    this.type = "menu";
    this.orient = "horizontal";
}

var style = custombutton.buttonGetHelp(self).replace(/id/g, _id);
var uri = makeURI('data:text/css,'+ encodeURIComponent(style));
var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
sss.loadAndRegisterSheet(uri, 0);
addEventListener("contextmenu", e => !this.disabled || e.ctrlKey || e.shiftKey || e.preventDefault(), false, this);

Отсутствует

 

Board footer

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