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

Будьте в курсе последних изменений в мире Mozilla, следя за нашим микроблогом в Twitter.

№185117-01-2025 22:16:25

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

Re: UCF - ваши кнопки, скрипты…

О, значит будет в конце body


Ещё, у меня вторая серия этой истории с верхним тулбаром.


Bug 1935035 - Infobar has incorrect styling when new sidebar panel is enabled (Firefox 136+)


То есть, решили использовать для notifications специальный тулбар,
а раз верхний UCF-тулбар добавляется после последнего тулбара
(который теперь для нотификаций), то ситуация повторяется,
инфобар появляется над дополнительным и разрезает монолит подряд идущих.


Можно посмотреть, запустив с консоли, например
gNotificationBox.appendNotification("", {label: "Test"}, [{label: "Close", callback: () => false}]);


Надо на что-то заменить селектор «:scope > toolbar:last-of-type»
То ли на «#PersonalToolbar»
то ли на «:scope > :nth-last-child(1 of toolbar:not(#notifications-toolbar))»
то ли ещё на что-то.

Отсутствует

 

№185223-01-2025 12:41:47

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 631
UA: Firefox 131.0

Re: UCF - ваши кнопки, скрипты…

Возможно ли как-то прикрутить Attributes_Inspector к Thunderbird?
Или это совсем тухлая затея?
Просто Виталий говорил, что CustomizableUI там урезанный и поэтому кнопки таким способом в Thunderbird не добавить.


«The Truth Is Out There»

Отсутствует

 

№185324-01-2025 21:31:12

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

Re: UCF - ваши кнопки, скрипты…

unter_officer пишет

к Thunderbird

Что-то не слишком подробно.


Я совсем не комплиментарен TB,
но, ладно, поставил 135.0b2 и подключил распоследний UCF.


В CustomStylesScripts.mjs, в UcfStylesScripts.scriptschrome.load
прописал имя файла кода, добавляющего кнопку на #tabs-toolbar
с таким, примерно, содержимым

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

Выделить код

Код:

(async () => {
	var btn = document.createXULElement("toolbarbutton");

	var ai = "Attributes Inspector";

	//btn.setAttribute("label", ai);
	btn.setAttribute("tooltiptext", ai);
	btn.setAttribute("image", "");

	var id, focus = () => BrowsingContext.get(id)?.window?.focus();
	btn["onmouse" + (AppConstants.platform == "win" ? "over" : "move")]
		= () => id = Services.focus.activeBrowsingContext?.id;
	btn.onmousedown = e => e.button || focus();

	btn._handleClick = () => {
		var u = Services.io.newURI;
		var url = u(Components.stack.filename).resolve("attributesInspector.js");
		var code = Cu.readUTF8URI(u(url)).replace(/\n.+?$/, "})");

		var scope = Cu.getJSTestingFunctions().evalReturningScope(`var f = ${code}`, window);
		var func = scope.f.bind(btn);
		delete scope.f;
		(btn._handleClick = func)();
	}
	document.getElementById("tabs-toolbar").append(btn);
})();


Рядом с ним, файл кода Attributes Inspector'а
под именем attributesInspector.js


И, вроде, завелось. Выглядит как-то так

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

Выделить код

Код:




Кстати, special_widgets.js в UCF дефолтно подключён,
что, в Thunderbird, приводит к записи в консоли
«ReferenceError: gNavToolbox is not defined».
Такое вот наблюдение.

Отсутствует

 

№185424-01-2025 23:00:45

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 631
UA: Firefox 131.0

Re: UCF - ваши кнопки, скрипты…

Dumby
Просто супер! Отлично работает.
Огромное спасибо.


Можно я обнаглею и попрошу вас сделать ещё одну кнопочку для Thunderbird. Разумеется, если это будет возможно.
Вот эту: Reload user{Chrome, Content}


«The Truth Is Out There»

Отсутствует

 

№185525-01-2025 01:38:35

dinn
Участник
 
Группа: Members
Зарегистрирован: 28-09-2024
Сообщений: 46
UA: Firefox 134.0

Re: UCF - ваши кнопки, скрипты…

Dumby, убегает запятая, в стилях перечерк, что делать - я не знаю, может вы просветите чуток?

Выделить код

Код:

document.body.style.cssText = `-moz-context-properties: fill, fill-opacity;`

Отредактировано dinn (25-01-2025 01:40:08)

Отсутствует

 

№185625-01-2025 15:28:33

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

Re: UCF - ваши кнопки, скрипты…

unter_officer пишет

эту

Может extensions.user_chrome_files.custom_scripts_background
плюс такая правка, не уверен.

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

Выделить код

Код:

/*
	ChromeUtils.importESModule("resource:///modules/CustomizableUI.sys.mjs").CustomizableUI.createWidget({
		label: "Reload user{Chrome, Content}.css",
		tooltiptext: "L: Reload userChrome.css\nR: Reload userContent.css",

		id: "ucf-userContentReloader",
		localized: false,
		onCreated(btn) {
			btn._handleClick = this.click;
			btn.oncontextmenu = oncontextmenu;
			btn.setAttribute("image", "");
		},
		get click() {
			var {file, spec} = getURI("hrome");
			var chromeSheet = getSheet(Services.wm.getMostRecentWindow(null).document, spec);
			delete this.click;
			return this.click = !chromeSheet ? noop : function() {
				var win = this.ownerGlobal;
				if (win.event?.detail < 2 && file.exists())
					reload(chromeSheet),
					win.setTimeout(restyle, 50);
			}
		}
	});
*/
	var click = btn => {
		var {file, spec} = getURI("hrome");
		var chromeSheet = getSheet(Services.wm.getMostRecentWindow(null).document, spec);

		(click = !chromeSheet ? noop : btn =>  {
			var win = btn.ownerGlobal;
			if (win.event?.detail < 2 && file.exists())
				reload(chromeSheet),
				win.setTimeout(restyle, 50);
		})(btn);
	}
	var clickcall = function() {click(this);}

	var topic = "chrome-document-loaded";
	var obs = doc => {
		if (!doc.documentURI?.endsWith("/messenger.xhtml")) return;

		var btn = doc.createXULElement("toolbarbutton");

		btn._handleClick = clickcall;
		btn.oncontextmenu = oncontextmenu;
		btn.tooltipText = "L: Reload userChrome.css\nR: Reload userContent.css";
		btn.setAttribute("image", "");

		doc.getElementById("tabs-toolbar").append(btn);
	}
	Services.obs.addObserver(obs, topic);
	Services.obs.addObserver(function quit(s, t) {
		Services.obs.removeObserver(quit, t);
		Services.obs.removeObserver(obs, topic);
	}, "quit-application-granted");

dinn пишет

убегает запятая

Ух ты! Я вижу это!
Ожидалось что угодно, но только не такое

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

Выделить код

Код:




Видимо, какое-то повреждение в CSS-парсере, что ли.
У меня, сходу, не получилось ничего с этим сделать.

в стилях перечерк

А этот вот перечёрк, он вообще надёжен?
Просто я попробовал обрести в девтулс вот этот hbox...


Тут рад бы был привести скриншот,
но, из-за дата-урлов, тогда выпадаю из лимита в 64k, поэтому словами:


ПКМ по вкладке —> Пункт «Переместить вкладку» (#context_moveTabOptions)
—> его пред-последний дочерний элемент (hbox.menu-right),
ну, типа стрелка
[>]


И «перечёрк» присутствует.


Хотя я даже близко ничем никчему не прикасался,
скрин сделан практически с чистого профиля.

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

Выделить код

Код:



Отсутствует

 

№185725-01-2025 16:17:27

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 631
UA: Firefox 131.0

Re: UCF - ваши кнопки, скрипты…

Dumby пишет

Может extensions.user_chrome_files.custom_scripts_background
плюс такая правка, не уверен.

Спасибо большое! Всё работает.


«The Truth Is Out There»

Отсутствует

 

№185803-02-2025 12:15:24

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

Re: UCF - ваши кнопки, скрипты…

Dumby помоги ещё! :beer:


Как определить, что загруженный файл является графическим по его содержанию, а не по расширению ?
В коде завершения загрузки надо переименовывать лишь картинки, взяв имя из Clipboard


P.S. пробовал некоторые примеры из сети, но не получилось в твой скрипт приспособить...

Отредактировано Dobrov (03-02-2025 12:18:10)

Отсутствует

 

№185904-02-2025 04:51:52

xrun1
Участник
 
Группа: Members
Зарегистрирован: 12-12-2013
Сообщений: 1238
UA: Firefox 134.0

Re: UCF - ваши кнопки, скрипты…

Поставил крайнюю версию UCF. Отвалилось всё в custom_script_win.js
Очистка работает, но раньше при ESC возвращало всё взад, теперь пусто.

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

Выделить код

Код:

// Script For browser window document [ChromeOnly]
// https://forum.mozilla-russia.org/viewtopic.php?pid=788229#p788229
// Очистить панель адреса или поиска прокруткой колёсиком мыши на панели
(this.clearsearchurlbar = {
    init(that) {
        for (let el of (this.urlsearcbar = document.querySelectorAll("#urlbar,#searchbar")))
            el.addEventListener("wheel", this);
        that.unloadlisteners.push("clearsearchurlbar");
    },
    handleEvent(e) {
        e.target.value = "";
    },
    destructor() {
        for (let el of this.urlsearcbar)
            el.removeEventListener("wheel", this);
    },
}).init(this);

// Автоматически открывать папки закладок на панели закладок
(this.placesmenudndhandler = {
    delay: 350,
    init(that) {
        var PlacesToolbar = this.PlacesToolbar = document.querySelector("#PlacesToolbar");
        if (!PlacesToolbar) return;
        PlacesToolbar.addEventListener("mouseover", this);
        this.timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
        that.unloadlisteners.push("placesmenudndhandler");
    },
    isButtonMenu(node) {
        if (node.localName == "toolbarbutton" &&
            node.getAttribute("type") == "menu" &&
            node.menupopup?.hasAttribute("placespopup"))
            return true;
        return false;
    },
    isOpen(popup) {
        if (popup.state === "open")
            return true;
        return false;
    },
    handleEvent(event, target = event.target, popup) {
        if (!this.isButtonMenu(target) || this.isOpen(popup = target.menupopup))
            return;
        this.timer.cancel();
        this.timer.initWithCallback(() => {
            if (this.curpopup && this.isOpen(this.curpopup))
                this.curpopup.hidePopup();
            this.curpopup = popup;
            popup.openPopup();
        }, this.delay, Ci.nsITimer.TYPE_ONE_SHOT);
        target.addEventListener("mouseleave", () => {
            this.timer.cancel();
        }, { once: true });
    },
    destructor() {
        this.PlacesToolbar.removeEventListener("mouseover", this);
    },
}).init(this);

// Пункт для контекстного меню адресной строки, подставляющий модификаторы поиска
(this.searchmodifiers = {
    init(that) {
        var urlbar = this.urlbar = gURLBar.textbox;
        if (!urlbar) return;
        urlbar.addEventListener("popupshowing", this);
        that.unloadlisteners.push("searchmodifiers");
    },
    handleEvent(e) {
        if (e.originalTarget != this.popup) return;
        this.urlbar.removeEventListener("popupshowing", this);
        this.urlbar = null;
        var sep = this.popup.querySelector("menuseparator.menuseparator-add-engine") || this.popup.lastElementChild;
        var menu = this.append("menu", sep, {label: "Вставить ^ * + % ~ # @"}, "before");
        this.append("menuseparator", menu, null, "before");
        var popup = this.append("menupopup", menu, {oncommand: "insert(event);"}, "append");
        popup.addEventListener("popupshowing", this, { once: true });
        this.handleEvent = e => {
            var df = document.createDocumentFragment();
            for (let label of [
                "^ История",
                "* Закладки",
                "+ Страницы с метками",
                "% Текущие открытые вкладки",
                "~ Набранные",
                "# Названия",
                "@ Веб-адреса (URLs)"
            ])
                this.append("menuitem", df, {label}, "append");
            popup.append(df);
            var ed = gURLBar.inputField.editor
                .QueryInterface(Ci.nsIEditor || Ci.nsIPlaintextEditor);
            popup.insert = e => {
                var str = e.target.label[0] + " ";
                var val = gURLBar.inputField.value;
                if (val && !val.endsWith(" ")) str = " " + str;
                ed.endOfDocument();
                ed.insertText(str);
            };
        };
    },
    get popup() {
        delete this.popup;
        return this.popup = gURLBar.inputField.parentNode.menupopup;
    },
    append(name, parent, attrs, func) {
        var elm = document.createXULElement(name);
        if (attrs) for (let a in attrs) elm.setAttribute(a, attrs[a]);
        parent[func](elm);
        return elm;
    },
    destructor() {
        this.urlbar?.removeEventListener("popupshowing", this);
    },
}).init(this);

// Добавить подменю "Поиск изображения в" в контекстном меню изображений
(this.searchimagecontextmenu = {
    handleEvent(evt) {
        if (evt.target != this.contextMenu || !gContextMenu?.imageInfo?.currentSrc) return;
        var array = [
            ['Google', 'https://www.google.lv/favicon.ico', 'https://www.google.com/searchbyimage?&image_url='],
            ['Яндекс', 'https://yastatic.net/morda-logo/i/favicon_islands.ico', 'https://yandex.ru/images/search?rpt=imageview&url='],
            ['Bing', 'https://www.bing.com/s/a/bing_p.ico', 'https://www.bing.com/images/search?view=detailv2&iss=sbi&form=SBIHMP&sbisrc=UrlPaste&q=imgurl:'],
            ['Tineye', 'https://tineye.com/favicon.ico', 'https://tineye.com/search?pluginver=bookmark_1.0&url='],
        ];
        var menu = document.createXULElement("menu");
        menu.setAttribute("label", "Поиск изображения в ...");
        menu.setAttribute("class", "menu-iconic");
        menu.setAttribute("image", array[1][1]);
        menu.setAttribute("onclick", "_searcclick(event);");
        menu._searcclick = function(e) {
            if (e.target != this) return;
            gBrowser.selectedTab = gBrowser.addTrustedTab(this._searcharg[2] + encodeURIComponent(gContextMenu.imageInfo.currentSrc), { index: gBrowser.selectedTab._tPos + 1 } );
            this.parentNode.hidePopup();
        }
        menu._searcharg = array[0];
        var menuPopup = document.createXULElement("menupopup");
        menu.append(menuPopup);
        array.forEach(m=> {
            var mItem = document.createXULElement("menuitem");
            mItem.setAttribute("label", m[0]);
            mItem.setAttribute("image", m[1]);
            mItem.setAttribute("class", "menuitem-iconic");
            mItem.setAttribute("oncommand", "gBrowser.selectedTab = gBrowser.addTrustedTab(_searcharg[2] + encodeURIComponent(gContextMenu.imageInfo.currentSrc), { index: gBrowser.selectedTab._tPos + 1 } );");
            mItem._searcharg = m;
            menuPopup.append(mItem);
        });
        var mItem = document.createXULElement("menuitem");
        mItem.setAttribute("label", 'Искать во всех поисковиках');
        mItem.setAttribute("oncommand", "_searcharg.forEach(m => { gBrowser.selectedTab = gBrowser.addTrustedTab(m[2] + encodeURIComponent(gContextMenu.imageInfo.currentSrc), { index: gBrowser.selectedTab._tPos + 1 } );});");
        mItem._searcharg = array;
        menuPopup.append(mItem);
        this.contextMenu.querySelector("#context-copyimage-contents")?.before(menu);
        this.popupshowing = e => {
            if (e.target != this.contextMenu) return;
            menu.hidden = !gContextMenu?.imageInfo?.currentSrc;
        };
        this.popuphiding = e => {
            if (e.target != this.contextMenu) return;
            menu.hidden = true;
        };
        this.contextMenu.addEventListener("popuphiding", this);
        this.handleEvent = e => {
            this[e.type](e);
        };
    },
    init(that) {
        var contextMenu = this.contextMenu = document.querySelector("#contentAreaContextMenu");
        if (!contextMenu) return;
        contextMenu.addEventListener("popupshowing", this);
        that.unloadlisteners.push("searchimagecontextmenu");
    },
    destructor() {
        this.contextMenu.removeEventListener("popupshowing", this);
        this.contextMenu.removeEventListener("popuphiding", this);
    },
}).init(this);

// Очистить куки ПКМ на иконке в строке адреса https://forum.mozilla-russia.org/viewtopic.php?pid=783022#p783022
(this.clearsitedatawithrightclick = {
    get clearSiteData() {
        delete this.clearSiteData;
        try {
            return this.clearSiteData = eval(`(${gIdentityHandler.clearSiteData})`.replace(/^\((async\s)?.*?clearSiteData/, "($1function clearSiteData").replace(/this\s*\./g, "gIdentityHandler.").replace(/(?:let\s*hidden\s*=\s*new\s*Promise\s*\([\S\s]+await\s*hidden\s*;|PanelMultiView\.hidePopup.+?;|event\.stopPropagation.+?;)/g, ""));
        } catch (e) {}
        return this.clearSiteData = function() {}
    },
    init(that) {
    var identitybox = this.identitybox = document.querySelector("#identity-box");
    if (!identitybox)
            return;
        identitybox.addEventListener("contextmenu", this, true);
        identitybox.addEventListener("click", this, true);
        that.unloadlisteners.push("clearsitedatawithrightclick");
    },
    handleEvent(e) {
        if (e.button != 2)
            return;
        e.preventDefault();
        e.stopPropagation();
        e.stopImmediatePropagation();
        if (e.type != "click")
            return;
        this.clearSiteData(e);
    },
    destructor() {
        this.identitybox.removeEventListener("contextmenu", this, true);
        this.identitybox.removeEventListener("click", this, true);
    },
}).init(this);


Даже №1825  в скриптах исправил. :angry: Знать бы, что ещё отвалилось и когда я об этом узнаю.

Отсутствует

 

№186004-02-2025 09:31:12

Farby
Участник
 
Группа: Members
Зарегистрирован: 21-11-2012
Сообщений: 326
UA: Google 2.1

Re: UCF - ваши кнопки, скрипты…

xrun1
Посмотрите ещё №1836
Вы используете:

Выделить код

Код:

that.unloadlisteners.push

такого там больше нет.

Добавлено 04-02-2025 09:45:28
для примера

Выделить код

Код:

that.unloadlisteners?.push("clearsearchurlbar") || that.setUnloadMap?.("clearsearchurlbar", this.destructor, this);

правда не знаю работает ли через "||"... для меня стояла задача, чтоб не мешало

Отредактировано Farby (04-02-2025 09:45:28)


Жизнь иногда такое выкидывает, что хочется подобрать...

Отсутствует

 

№186104-02-2025 16:04:07

xrun1
Участник
 
Группа: Members
Зарегистрирован: 12-12-2013
Сообщений: 1238
UA: Firefox 134.0

Re: UCF - ваши кнопки, скрипты…

Farby пишет

that.unloadlisteners.push

такого там больше нет.

Проясните мне, пожалуйста, один момент. Такого нет в принципе, или только в новой версии UCF? Ведь на

version, date year-month-day: 2024-10-31

всё прекрасно работает, даже без правки №1836.
Это первое. И второе: сейчас попробую заменить и нужно ли в версию

version, date year-month-day: 2025-1-17

вносить правку №1836?
Спасибо за помощь.:beer:

Отредактировано xrun1 (04-02-2025 16:05:11)

Отсутствует

 

№186204-02-2025 16:57:37

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

Re: UCF - ваши кнопки, скрипты…

Dobrov пишет

Как определить, что загруженный файл является графическим по его содержанию, а не по расширению ?

Может что-то тупенькое подойдёт?
Типа попытки загрузить в <img>
Если удалось — значит графический.


Вот, погоняй

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

Выделить код

Код:

(async fx125 => {
	var arg = window;
	if (fx125) arg = arg.browsingContext;

	var fp = makeFilePicker();
	fp.init(arg, "", fp.modeOpen);
	var res = await new Promise(fp.open);
	if (res != fp.returnOK) return;

	var isGraf = url => new Promise(resolve => {
		var img = new Image();
		img.onload = img.onerror = e => resolve(e.type.startsWith("l"));
		img.src = url;
	});

	alert(await isGraf(fp.fileURL.spec));

})(parseInt(Services.appinfo.platformVersion) >= 125);

Отсутствует

 

№186304-02-2025 18:04:55

Farby
Участник
 
Группа: Members
Зарегистрирован: 21-11-2012
Сообщений: 326
UA: Google 2.1

Re: UCF - ваши кнопки, скрипты…

xrun1 пишет

Проясните мне, пожалуйста, один момент. Такого нет в принципе, или только в новой версии UCF?

unloadlisteners был удалён в этом комите, что касается правок это на вкус и цвет, я не пользую UCF...


Жизнь иногда такое выкидывает, что хочется подобрать...

Отсутствует

 

№186405-02-2025 15:50:41

xrun1
Участник
 
Группа: Members
Зарегистрирован: 12-12-2013
Сообщений: 1238
UA: Firefox 135.0

Re: UCF - ваши кнопки, скрипты…

Farby
Обновился на [firefox] 135, кнопки не отвалились.
Себе на будущее, 136 и т.д. сделал новый UCF и внёс правки, чтобы как только - так сразу. Спасибо, работают. :beer:
Кто поможет?
В 135-й отвалились значки поиска по выделенному.

картинка
6da12ef9b3cfc8f199c548ececdcdf41.png

contextsearch.js
    scriptschrome: { // For browser window document [ChromeOnly]
        domload: [ // By event "DOMContentLoaded"
            { path: "sidebar_tabs.js", ucfobj: true, },// Sidebar Tabs
        ],
        load: [ // By event "load"
            { path: "cs_win/contextsearch.js", ucfobj: true, },

Выделить код

Код:

(this.contextsearch = {
    topic: "browser-search-engine-modified",
    hide: "browser.search.hiddenOneOffs",
    defaultImg: "chrome://browser/skin/search-engine-placeholder.png",
    searchSelect: null,
    popup: null,
    init(that) {
        var searchSelect = this.searchSelect = document.querySelector("#context-searchselect");
        if (!searchSelect)
            return;
        var popup = this.popup = searchSelect.closest("menupopup");
        popup.addEventListener("popupshowing", this);
        that.unloadlisteners?.push("contextsearch");
    },
    destructor() {
        this.popup.removeEventListener("popupshowing", this);
        if (this.popupshowing == this.handler) {
            this.popup.removeEventListener("popuphidden", this);
            Services.obs.removeObserver(this, this.topic);
            Services.prefs.removeObserver(this.hide, this);
        }
    },
    handleEvent(e) {
        this[e.type](e);
    },
    popupshowing(e) {
        var popup = this.popup;
        var searchSelect = this.searchSelect;
        if (e.target != popup || searchSelect.hidden) return;

        var menu = document.createXULElement("menu");
        menu.className = "menu-iconic";
        var menupopup = document.createXULElement("menupopup");
        menu.append(menupopup);
        menu.ePopup = menupopup;
        searchSelect.style.setProperty("display", "none", "important");
        searchSelect.before(menu);
        menu.onclick = this.search.bind(this);
        this.handler = e => e.target != popup || (menu.hidden = searchSelect.hidden);
        this.handlerRebuild = e => this.handler(e) || this.rebuild(menu);
        this.popuphidden = ev => {
            if (ev.target != popup) return;
            menu.hidden = true;
        };
        this.popup.addEventListener("popuphidden", this);
        this.rebuild(menu);
    },
    getEngines() {
        var args = "hideOneOffButton" in Services.search.defaultEngine
            ? [e => !e.hideOneOffButton]
            : Object.defineProperty(
                [function(e) {return !this.includes(e.name);}], "1", {
                    get: () => Services.prefs.getStringPref(this.hide)?.split(",") || []
                }
            );
        return (this.getEngines = async () =>
            (await Services.search.getVisibleEngines()).filter(...args)
        )();
    },
    async rebuild(menu) {
        var de = Services.search.defaultEngine;
        de = de.wrappedJSObject || de;
        this.setAttrs(menu, de, `Искать в ${de.name} или в ...`);
        menu.ePopup.textContent = "";
        for(let engine of await this.getEngines()) {
            if (engine == de) continue;
            var menuitem = document.createXULElement("menuitem");
            menuitem.className = "menuitem-iconic";
            this.setAttrs(menuitem, engine);
            menu.ePopup.append(menuitem);
        }
        this.popupshowing = this.handler;
        Services.obs.addObserver(this, this.topic, false);
        Services.prefs.addObserver(this.hide, this);
    },
    setAttrs(node, engine, label = engine.name) {
        node.engine = engine;
        node.setAttribute("label", label);
        node.setAttribute("image", engine._iconURI ? engine._iconURI.spec : engine.iconURI ? engine.iconURI.spec : this.defaultImg);
    },
    observe() {
        this.popupshowing = this.handlerRebuild;
        Services.obs.removeObserver(this, this.topic);
        Services.prefs.removeObserver(this.hide, this);
    },
    search(e) {
        var {engine} = e.target;
        if (!engine) return;
        var searchSelect = this.searchSelect;
        var submission = engine.getSubmission(
            searchSelect.searchTerms, null, "contextmenu"
        );
        if (submission) {
            let tab = gBrowser.addTab(submission.uri.spec, {
                postData: submission.postData,
                index: (gBrowser.selectedTab._tPos + 1),
                triggeringPrincipal: searchSelect.principal
            });
            if (e.button == 0)
                gBrowser.selectedTab = tab;
        }
        var popup = this.popup;
        e.button != 1 && popup.state == "open" && popup.hidePopup();
    }
}).init(this);

Отредактировано xrun1 (05-02-2025 15:52:31)

Отсутствует

 

№186505-02-2025 17:12:15

Farby
Участник
 
Группа: Members
Зарегистрирован: 21-11-2012
Сообщений: 326
UA: Google 2.1

Re: UCF - ваши кнопки, скрипты…

xrun1 пишет

Кто поможет?
В 135-й отвалились значки поиска по выделенному.

На замену

browser-search-engine.uc.js

Выделить код

Код:

// ==UserScript==
// @name      Browser search engine
// @author    Vitaliy V.
// @include   main
// @shutdown  window.contextsearch.destructor();
// @note      https://forum.mozilla-russia.org/viewtopic.php?pid=780283#p780283
// ==/UserScript==

(this.contextsearch = {
    topic: "browser-search-engine-modified",
    hide: "browser.search.hiddenOneOffs",
    defaultImg: "chrome://browser/skin/search-engine-placeholder.png",
    searchSelect: null,
    popup: null,
    init(that) {
        var searchSelect = this.searchSelect = document.querySelector("#context-searchselect");
        if (!searchSelect)
            return;
        var popup = this.popup = searchSelect.closest("menupopup");
        popup.addEventListener("popupshowing", this);
      //  that.unloadlisteners?.push("contextsearch"); // OLD UCF
        that.setUnloadMap?.("contextsearch", this.destructor, this);
    },
    destructor() {
        this.popup.removeEventListener("popupshowing", this);
        if (this.popupshowing == this.handler) {
            this.popup.removeEventListener("popuphidden", this);
            Services.obs.removeObserver(this, this.topic);
            Services.prefs.removeObserver(this.hide, this);
        }
    },
    handleEvent(e) {
        this[e.type](e);
    },
    popupshowing(e) {
        var popup = this.popup;
        var searchSelect = this.searchSelect;
        if (e.target != popup || searchSelect.hidden) return;

        var menu = document.createXULElement("menu");
        menu.className = "menu-iconic";
        var menupopup = document.createXULElement("menupopup");
        menu.append(menupopup);
        menu.ePopup = menupopup;
        searchSelect.style.setProperty("display", "none", "important");
        searchSelect.before(menu);
        menu.onclick = this.search.bind(this);
        this.handler = e => e.target != popup || (menu.hidden = searchSelect.hidden);
        this.handlerRebuild = e => this.handler(e) || this.rebuild(menu);
        this.popuphidden = ev => {
            if (ev.target != popup) return;
            menu.hidden = true;
        };
        this.popup.addEventListener("popuphidden", this);
        this.rebuild(menu);
    },
    getEngines() {
        var args = "hideOneOffButton" in Services.search.defaultEngine
            ? [e => !e.hideOneOffButton]
            : Object.defineProperty(
                [function(e) {return !this.includes(e.name);}], "1", {
                    get: () => Services.prefs.getStringPref(this.hide, [])?.split(",") || []
                }
            );
        return (this.getEngines = async () =>
            (await Services.search.getVisibleEngines()).filter(...args)
        )();
    },
    async rebuild(menu) {
        var de = Services.search.defaultEngine;
        de = de.wrappedJSObject || de;
        this.setAttrs(menu, de, `Искать в ${de.name} или в ...`);
        menu.ePopup.textContent = "";
        for(let engine of await this.getEngines()) {
            if (engine == de) continue;
            var menuitem = document.createXULElement("menuitem");
            menuitem.className = "menuitem-iconic";
            this.setAttrs(menuitem, engine);
            menu.ePopup.append(menuitem);
        }
        this.popupshowing = this.handler;
        Services.obs.addObserver(this, this.topic, false);
        Services.prefs.addObserver(this.hide, this);
    },
    async setAttrs(node, engine, label = engine.name) {
        node.engine = engine;
        node.setAttribute("label", label);
        node.setAttribute("image", await engine.getIconURL?.() || engine.getIconURLBySize?.(16, 16) || this.defaultImg);
    },
    observe() {
        this.popupshowing = this.handlerRebuild;
        Services.obs.removeObserver(this, this.topic);
        Services.prefs.removeObserver(this.hide, this);
    },
    search(e) {
        var {engine} = e.target;
        if (!engine) return;
        var searchSelect = this.searchSelect;
        var submission = engine.getSubmission(
            searchSelect.searchTerms, null, "contextmenu"
        );
        if (submission) {
            let tab = gBrowser.addTab(submission.uri.spec, {
                postData: submission.postData,
                index: (gBrowser.selectedTab._tPos + 1),
                triggeringPrincipal: searchSelect.principal
            });
            if (e.button == 0)
                gBrowser.selectedTab = tab;
        }
        var popup = this.popup;
        e.button != 1 && popup.state == "open" && popup.hidePopup();
    }
}).init(this);


Жизнь иногда такое выкидывает, что хочется подобрать...

Отсутствует

 

№186606-02-2025 00:29:12

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

Re: UCF - ваши кнопки, скрипты…

Dumby пишет

Типа попытки загрузить в <img> Если удалось — значит графический.

Благодарю! пойдёт, за неимением лучшего, т.к. определять по расширению глупо.
Возвращает false для pcx tga tif и видимо, всех, не поддерживаемых на страницах HTML.
на Маке и Линукс тип можно получить через proc.runwAsync(file path) это слишком тупой способ…

Отсутствует

 

№186706-02-2025 01:44:13

xrun1
Участник
 
Группа: Members
Зарегистрирован: 12-12-2013
Сообщений: 1238
UA: Firefox 135.0

Re: UCF - ваши кнопки, скрипты…

Farby
Спасибо, работает.:beer:
Я сразу забыл уточнить, что использую правленый Вами вариант https://forum.mozilla-russia.org/viewto … 58#p808658.

Отсутствует

 

№186806-02-2025 01:58:19

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

Re: UCF - ваши кнопки, скрипты…

Dumby
Помоги! Не смог прикрутить проверку графического файла к скрипту отслеживания загрузок!
для isGraf await тоже не помог, сбой на строке var src = download.source.url;

onDownloadChanged

Выделить код

Код:

(async () => {
	var list = await Downloads.getList(Downloads.ALL);

	list.addView({
		onDownloadChanged(download) {
			if (!download.succeeded) return;

			var isGraf = url => new Promise(resolve => {
				var img = new Image();
				img.onload = img.onerror = e => resolve(e.type.startsWith("l"));
				img.src = url;
			});

			var src = download.source.url;
			// var s = await isGraf(src);
			console.log(src);
			// alert(await isGraf(src));

		}
	});
})();

Отсутствует

 

№186906-02-2025 02:12:21

xrun1
Участник
 
Группа: Members
Зарегистрирован: 12-12-2013
Сообщений: 1238
UA: Firefox 135.0

Re: UCF - ваши кнопки, скрипты…

Dobrov
По сути то же, что предложил Dumby, вдруг код пригодится. Я в своём скрипте использую. На экзотических форматах не проверял.

Отсутствует

 

№187006-02-2025 18:34:39

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

Re: UCF - ваши кнопки, скрипты…

Dobrov пишет

await тоже не помог

Ещё бы, onDownloadChanged() ведь не асинхронная функция.

download.source.url

Ещё раз по сети притащить? Нет, такое точно без меня.


Может как-то так

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

Выделить код

Код:

(async renamed => {

	var view = {
		onDownloadChanged(download) {
			if (
				!download.succeeded || download[renamed]
				|| this.skipRenameForDownload(download)
			)
				return;;
			
			var win = Services.wm.getMostRecentBrowserWindow();
			var img = new win.Image();

			img.onload = this.onImgLoad;	
			img.src = (img.u = Services.io.newFileURI(
				img.f = win.FileUtils.File((img.d = download).target.path)
			)).spec;
		},
		onImgLoad() {
			var uri = this.u.QueryInterface(Ci.nsIURL);
			var file = this.f, {parent} = file;
			var newName = view.getNewFileName() + "." + uri.fileExtension;

			var newFile = parent.clone();
			newFile.append(newName);
			newFile.createUnique(file.NORMAL_FILE_TYPE, file.permissions);

			file.renameTo(parent, newFile.leafName);

			var {d} = this;
			d.target.path = newFile.path;
			d[renamed] = true;

			d.onchange(d.succeeded = false);
			d.onchange(d.succeeded = true);
			view.dh.addDownloadToHistory(d);
		},
		get dh() {
			delete this.dh;
			return this.dh = ChromeUtils.importESModule(
				"resource://gre/modules/DownloadHistory.sys.mjs"
			).DownloadHistory;
		},

		skipRenameForDownload(download) {
			return false; // dummy
		},
		getNewFileName(arg) {
			return "renamed"; // dummy
		}
	};
	(await Downloads.getList(Downloads.ALL)).addView(view);

})(Symbol.for("Download renamed for some reason"));

Отсутствует

 

№187107-02-2025 01:52:14

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 631
UA: Firefox 135.0

Re: UCF - ваши кнопки, скрипты…

Решил "пощупать" 136 версию, бета 2.
Отвалилось с десяток скриптов.


У всех отвалившихся скриптов в консоли какая-то странная одинаковая ошибка:
Content-Security-Policy: Параметры страницы заблокировали выполнение обработчика события (script-src-attr), поскольку он нарушает следующую директиву: «script-src-attr 'none' 'report-sample'»


Это что за ошибка такая? И как её исправить?


add
Не уверен, но есть подозрение, что отвалились скрипты, где присутствует что-то типа такого:
btn.setAttribute("oncommand", "handleCommand(this)");
btn.setAttribute("oncommand", "BrowserCommands.reload()");
menuitem.setAttribute("oncommand", m[2]);
menuitem.setAttribute("onclick", "_searcclick(event);");
popup.setAttribute("onpopupshowing", "event.defaultPrevented || filler.fill(event)");

Отредактировано unter_officer (07-02-2025 03:56:32)


«The Truth Is Out There»

Отсутствует

 

№187207-02-2025 08:20:31

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

Re: UCF - ваши кнопки, скрипты…

unter_officer пишет

Это что за ошибка такая?

Не столько даже ошибка, скорее репортинг. Генезис таков:


Bug 1937080 - Mitigate breakage risks for parent-process script security


а точнее, именно для бетки,


Bug 1942991 - Block inline event handlers from browser.xhtml in Early Beta

И как её исправить?

Ну, пока что, можно просто настройку переключить.
Но, как у них заведено, настройка подлежит удалению.


Поэтому, лучше переделать коды с использования «on…» атрибутов
на использование метода addEventListener()


В некоторых случаях, можно использовать объектный on-хэндлер,
типа elm.onclick = e => {…}
но для каких-нибудь "oncommand", "onpopupshowing"
таковых нет, поскольку это чисто XUL'ский стафф.

Отредактировано Dumby (07-02-2025 08:22:07)

Отсутствует

 

№187307-02-2025 15:13:00

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

Re: UCF - ваши кнопки, скрипты…

Dumby - не все картинки переименовываются.


Наверное, с Overlay проблемы, но имена получены и меняются для всех.
ClickPicSave.jsm нормально сохраняет не в папку Загрузки, но не переименовывает с некоторых сайтов.
Может, для подобных расширений тоже не сработает. Через "Сохранить изображение как…" всегда переименует.

Mod onDownloadChanged

Выделить код

Код:

(async renamed => {

	var view = {
		onDownloadChanged(download) {
			if (!download.succeeded || download[renamed] || this.skipRenameForDownload(download))
				return;
			
			var win = Services.wm.getMostRecentBrowserWindow();
			var img = new win.Image();

			img.onload = this.onImgLoad;
			img.src = (img.u = Services.io.newFileURI(
				img.f = win.FileUtils.File((img.d = download).target.path)
			)).spec;
		},
		onImgLoad() {
			var uri = this.u.QueryInterface(Ci.nsIURL);
			var file = this.f, {parent} = file;

			var newName = view.getNewFileName(file, uri.fileExtension);
			var newFile = parent.clone();
			newFile.append(newName);
			newFile.createUnique(file.NORMAL_FILE_TYPE, file.permissions);

console.log(file.leafName, newName);

			file.renameTo(parent, newFile.leafName);

			var {d} = this;
			d.target.path = newFile.path;
			d[renamed] = true;

			d.onchange(d.succeeded = false); d.onchange(d.succeeded = true);
			view.dh.addDownloadToHistory(d);
		},
		get dh() {
			delete this.dh;
			return this.dh = ChromeUtils.importESModule("resource://gre/modules/DownloadHistory.sys.mjs").DownloadHistory;
		},
		getNewFileName(file, ext) { //имя будет из Clipboard…
			return (file.leafName.replace(new RegExp('.'+ ext +'$','i'),'') +"_ReName").slice(0,128) +"."+ ext;
		},
		skipRenameForDownload(download) {
			return false; // dummy
		}
	};
	(await Downloads.getList(Downloads.ALL)).addView(view);

})(Symbol.for("Download renamed"));

Отсутствует

 

№187407-02-2025 15:44:50

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 631
UA: Firefox 135.0

Re: UCF - ваши кнопки, скрипты…

Dumby пишет

Поэтому, лучше переделать коды с использования «on…» атрибутов
на использование метода addEventListener()

В некоторых случаях, можно использовать объектный on-хэндлер,
типа elm.onclick = e => {…}
но для каких-нибудь "oncommand", "onpopupshowing"
таковых нет, поскольку это чисто XUL'ский стафф.

Dumby, в общем что смог, то переделал. Не знаю насколько правильно, но вроде работает и в консоль не "гадит".
Однако остались скрипты, которые мне не осилить. Поэтому просьба помочь.

Восстановление удалённых закладок или папок

Выделить код

Код:

//
(async sep => {
    if (!sep) return;

    var key = "hasRemoveTransaction";
    var raws = UcfPrefs.dbg.ref("lazy", PlacesTransactions.undo).TransactionsHistory.proxifiedToRaw;
    raws[key] ??= entry => {
        for(var transaction of entry) {
            if (raws.get(transaction) instanceof PlacesTransactions.Remove)
                return true;
        }
    }
    var menuitem = document.createXULElement("menuitem");
    for(var args of Object.entries({
        closemenu: "single",
        class: "menuitem-iconic",
        id: "placesCmd_undoRemove",
        label: "Восстановить удалённое",
        oncommand: "PlacesTransactions.undo().catch(Cu.reportError);",
        image: "data:image/png;base64,.....",
    }))
        menuitem.setAttribute(...args);

    var desc = Object.getOwnPropertyDescriptor(XULElement.prototype, "hidden");
    var {set} = desc;
    desc.set = () => {
        var entry = PlacesTransactions.topUndoEntry;
        set.call(menuitem, !entry || !raws[key](entry));
    }
    Object.defineProperty(menuitem, "disabled", {});
    Object.defineProperty(menuitem, "hidden", desc);
    sep.before(menuitem);
})(document.getElementById("placesContext_deleteSeparator"));

Simple Session Manager

Выделить код

Код:

//
(async (pid, mp, self) => CustomizableUI.createWidget((self = {
    id: "ucf_SimpleSessionManager",
    label: "Simple Session Manager",
    tooltiptext: "Менеджер сессий",
    localized: false,
    init() {
        this.handleEvent = e => this[e.type](e);
        this.onTimeout = async () => await this.saveSession() || this.save();
        Services.obs.addObserver(this, "quit-application");
        var {openMenu} = this;
        this.render = function() {
            this.openMenu = openMenu;
            this.constructor.prototype.render.call(this);
        }
        delete this.init;
        return this;
    },
    image: "data:image/png;base64,.....",
    onCreated(btn) {
        btn.type = "menu";
        btn.phTimestamp = 0;
        btn.render = this.render;
        btn.onclick = this.click;
        btn.setAttribute("image", this.image);
        var popup = btn.ownerDocument.createXULElement("menupopup");
        popup.filler = this;
        popup.id = pid;
        popup.setAttribute("onpopupshowing", "event.defaultPrevented || filler.fill(event)");
        btn.prepend(popup);

        btn.addEventListener("mousedown", this);
        popup.addEventListener("command", this);
        btn.ownerGlobal.addEventListener("unload", () => {
            btn.removeEventListener("mousedown", this);
            popup.removeEventListener("command", this);
            if (popup.filler != this)
                // popup.removeEventListener("dragstart", this),
                popup.removeEventListener("popuphidden", this);
        }, {once: true});
    },
    openMenu(arg) {
        var pos;
        if (this.matches(".widget-overflow-list > :scope"))
            pos = "after_start";
        else var win = this.ownerGlobal, {width, height, top, bottom, left, right} =
            this.closest("toolbar").getBoundingClientRect(), pos = width > height
                ? `${win.innerHeight - bottom > top ? "after" : "before"}_start`
                : `${win.innerWidth - right > left ? "end" : "start"}_before`;
        this.firstChild.setAttribute("position", pos);
        delete this.openMenu;
        this.openMenu(arg);
    },
    mousedown(e) {
        if (e.button) return;
        var trg = e.target;
        if (trg.nodeName.startsWith("t"))
            trg.mdTimestamp = Cu.now(),
            trg.tid = e.view.setTimeout(this.onTimeout, 500),
            e.preventDefault();
    },
    boot(trg) {
        var popup = trg.parentNode;
        var old = popup.querySelector("[boot]");
        if (old != trg) old?.removeAttribute("boot");
        trg.toggleAttribute("boot");
        popup.bootChanged = true;
    },
    muTimestamp: 0,
    click(e) {
        var trg = e.target;
        if (trg.nodeName == "menu") {
            if (e.button > 1) self.boot(trg);
            else if (Cu.now() - self.muTimestamp > 50)
                e.view.closeMenus(trg.menupopup),
                self.restoreSession(trg.label, (e.button || e.ctrlKey) && e.view);
        }
        else if (trg != this || e.button) return;
        e.view.clearTimeout(this.tid);
        if (this.mdTimestamp - this.phTimestamp > 50) this.open = true;
    },
    async command(e) {
        var arg, trg = e.target, cmd = trg.value;
        if (cmd.startsWith("r"))
            arg = trg.parentNode.parentNode.label;
        await this[cmd](arg) || this.save();
    },
    dragstart(e) {
        var trg = e.target;
        if (trg.nodeName != "menu") return;

        var popup = trg.parentNode;
        this.dragData = {trg, mouse: true};
        trg.menupopup.hidePopup();

        var win = trg.ownerGlobal;
        win.setCursor("grabbing");
        var {width} = trg.getBoundingClientRect();
        trg.setAttribute("maxwidth", width);

        win.addEventListener("mouseup", this);
        popup.addEventListener("mousemove", this);
    },
    mousemove(e) {
        var trg = e.target, dtrg = this.dragData.trg;
        if (trg == dtrg || trg.nodeName != "menu") return;

        e.movementY > 0
            ? trg.nextSibling != dtrg && trg.after(dtrg)
            : trg.previousSibling != dtrg && trg.before(dtrg);
    },
    mouseup(arg) {
        if (arg.constructor.isInstance?.(arg)) {
            arg.preventDefault();
            var {trg} = this.dragData;
            this.dragData.mouse = false;
            this.muTimestamp = Cu.now();
        }
        else var trg = arg;

        trg.removeAttribute("maxwidth");
        trg.ownerGlobal.setCursor("auto");
        trg.ownerGlobal.removeEventListener("mouseup", this);
        trg.parentNode.removeEventListener("mousemove", this);
    },
    popuphidden(e) {
        if (!e.target.id) return;
        e.view.removeEventListener("keydown", this, true);
        var popup = e.target;
        popup.parentNode.phTimestamp = Cu.now();

        var save;
        if (this.dragData) {
            var {trg, mouse} = this.dragData;
            mouse && this.mouseup(trg);

            delete this.dragData;
            var order = Array.from(popup.getElementsByTagName("menu"), m => m.label);
            if (order.toString() != this.meta.order.toString()) {
                var hasBoot = this.meta.boot != null;
                if (hasBoot) var bootName = this.meta.order[this.meta.boot];
                this.meta.order = order;
                if (hasBoot) this.meta.boot = this.meta.order.indexOf(bootName);
                save = true;
            }
        }
        if (popup.bootChanged) {
            delete popup.bootChanged;
            var {boot} = this.meta;
            var bootNode = e.target.querySelector("[boot]");
            var ind = bootNode && this.meta.order.indexOf(bootNode.label);
            if (ind != boot)
                this.meta.boot = ind,
                save = true;
        }
        save && this.save(e.view);
    },

    sku: `#${pid} > menu[maxwidth]`,
    skd: `#${pid} > menu:is([maxwidth],[_moz-menuactive]):not([open])`,
    keydown(e) {
        if (e.repeat && e.key == "Shift" || !e.shiftKey && e.key != " ") return;
        var func = this.keyHandlers[e.key];
        if (!func) return;

        var menu = e.view.windowRoot
            .ownerGlobal.document.querySelector(this.skd);
        if (menu)
            e.stopImmediatePropagation(),
            func.call(this, menu, e);
    },
    keyup(e) {
        if (e.key != "Shift") return;
        var win = e.view.windowRoot.ownerGlobal;
        win.removeEventListener("keyup", this, true);
        win.document.querySelector(this.skd)?.removeAttribute("maxwidth");
    },
    keyHandlers: {
        Enter(menu) {
            this.boot(menu);
        }
    },
    arrow(menu) {
        if (menu.hasAttribute("maxwidth")) return;
        menu.setAttribute("maxwidth", menu.getBoundingClientRect().width);
        menu.ownerGlobal.addEventListener("keyup", this, true);
        this.dragData = {trg: menu};
    },
    fill(e) {
        var mxe = e.view.MozXULElement;

        var initFrag = mxe.parseXULToFragment(`
            <menuitem value="saveSession" class="menuitem-iconic" label="Сохранить сессию"/>
            <menuseparator/>
        `);
        this.menuFrag = mxe.parseXULToFragment(`<menu class="menu-iconic"><menupopup>
            <menuitem label="Переименовать"
                class="menuitem-iconic" value="renameSession"/>
            <menuitem label="Удалить"
                class="menuitem-iconic" value="removeSession"/>
        </menupopup></menu>`);

        this.regStyle();

        var filler = { fill: e => e.target.id
            ? e.view.addEventListener("keydown", this, true)
                || e.target.fillFlag || this.fillSessions(e.target)
                : this.dragData?.mouse && e.preventDefault()
        };

        (this.fill = e => {
            var trg = e.target;
            trg.setAttribute("context", "");
            trg.append(trg.ownerDocument.importNode(initFrag, true));
            (trg.filler = filler).fill(e);
            // trg.addEventListener("dragstart", this);
            trg.addEventListener("popuphidden", this);
        })(e);
    },
    fillSessions(popup) {
        while(popup.lastChild.nodeName == "menu") popup.lastChild.remove();
        var ind = 0, {boot} = this.meta;
        for(var name of this.meta.order) {
            var df = popup.ownerDocument.importNode(this.menuFrag, true);
            df.firstChild.setAttribute("label", name);
            if (ind++ == boot) df.firstChild.toggleAttribute("boot");
            popup.append(df);
        }
        popup.fillFlag = true;
    },
    regStyle() {
        delete this.regStyle;
        var subst = "ucf-ssm-style-resurl";
        Services.io.getProtocolHandler("resource").QueryInterface(Ci.nsIResProtocolHandler).setSubstitution(
            subst, Services.io.newURI("data:text/css;charset=utf-8," + encodeURIComponent(`
                @-moz-document url-prefix(chrome://browser/content/browser.xhtml) {
                    #${pid} > menu {
                        // list-style-image: url("${this.image}");
                        list-style-image: url("");
                    }
                    #${pid} > [value=saveSession] {
                        list-style-image: url("");
                    }
                    #${pid} [value=restoreSession] {
                        list-style-image: url("");
                    }
                    #${pid} [value=renameSession] {
                        list-style-image: url("");
                    }
                    #${pid} [value=removeSession] {
                        list-style-image: url("");
                    }
                    #${pid} > menuseparator:last-child,
                    #${pid} > menu[maxwidth] > .menu-right {
                        display: none;
                    }
                    #${pid} > menu[boot] {
                        color: #AA0000;
                        font-weight: bold;
                    }
                }`.replace(/;$/gm, " !important;"))));
        var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
        sss.loadAndRegisterSheet(Services.io.newURI("resource://" + subst), sss.USER_SHEET);
    },
    get gs() {
        delete this.gs;
        return this.gs = Cu.import("resource:///modules/sessionstore/SessionStore.jsm", {});
    },
    splice(name, newName) {
        var ind = this.meta.order.indexOf(name);
        if (ind == -1) return;
        var args = [ind, 1];

        if (1 in arguments) args.push(newName);
        else {
            if (ind == this.meta.boot) this.meta.boot = null;
            else if (ind < this.meta.boot) this.meta.boot--;
        }
        this.meta.order.splice(...args);
    },
    get meta() {
        var file = Services.dirsvc.get("UChrm", Ci.nsIFile);
        file.append("simple_session_manager.json");
        this.path = file.path;
        try {
            this.data = JSON.parse(Cu.readUTF8File(file));
        } catch {
            this.pp = file.parent.path;
            this.data = Object.create(null);
        }
        var meta = this.data[mp];
        if (!meta) {
            var order = Object.keys(this.data);
            meta = this.data[mp] = {order, boot: null};
        }
        delete this.meta;
        return this.meta = meta;
    },
    io: {
        get OS() {
            delete this.OS;
            Cu.import("resource://gre/modules/osfile.jsm", this);
            return this.OS;
        },
        makeDirectory(path) {
            return (this.makeDirectory = this.OS.File.makeDir)(path);
        },
        writeJSON(path, obj) {
            var wa = this.OS.File.writeAtomic;
            return (this.writeJSON = (path, obj) => wa(path, JSON.stringify(obj)))(path, obj);
        }
    },
    async save(excWin) {
        var io = Cu.getGlobalForObject(Cu).IOUtils || this.io;
        if (this.pp)
            await io.makeDirectory(this.pp), delete this.pp;
        (this.save = excWin => {
            this.meta.order.length
                ? io.writeJSON(this.path, this.data)
                : io.remove(this.path, {ignoreAbsent: true});
            for(var win of CustomizableUI.windows) {
                if (win == excWin) continue;
                var popup = win.document.getElementById(pid);
                if (popup) popup.fillFlag = false;
            }
        })(excWin);
    },
    get prompter() {
        var {prompt} = Services;
        var p = {}, args = [null, null, "UCF Simple Session Manager"];
        p.alert = prompt.alert.bind(...args);
        p.confirm = prompt.confirm.bind(...args);
        var pr = prompt.prompt.bind(...args);
        p.prompt = (msg, value) => {
            var res = {value};
            return pr(msg, res, null, {}) ? res.value : null;
        }
        delete this.prompter;
        return this.prompter = p;
    },
    observe(s, t, data) {
        Services.obs.removeObserver(this, "quit-application");
        if (data.includes("restart")) return;

        var {boot} = this.meta;
        if (boot == null) return;

        var state = this.data[this.meta.order[boot]];
        var ssi = this.gs.SessionStoreInternal;
        ssi.getCurrentState = () => state;
        Services.obs.removeObserver(ssi, "browser:purge-session-history");

        Services.prefs.setBoolPref("browser.sessionstore.resume_session_once", true);
    },
    get bwt() {
        delete this.bwt;
        var url = "resource:///modules/BrowserWindowTracker.jsm";
        return this.bwt = ChromeUtils.import(url).BrowserWindowTracker;
    },
    getTabLabel() {
        return `${
            this.bwt.getTopWindow().gBrowser.selectedTab.label.slice(0, 70)
        }`;
    },
    getName(state) {
        var tl = 0;
        for(var w of state.windows) tl += w.tabs.length;
        return `${tl}(B) [${
            new Date().toLocaleString("mn").replace(" ", " - ")
        }]`;
    },
    exists(name) {
        this.meta;
        return (this.exists = name => name in this.data &&
            !this.prompter.alert("Сессия с тем же именем уже существует!"))(name);
    },
    getState() {
        return JSON.parse(this.gs.SessionStore.getBrowserState());
    },
    get spref() {
        var pref = "browser.sessionstore.interval";
        var timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
        var wait = cb => timer.initWithCallback(cb, 1e3, timer.TYPE_ONE_SHOT);
        delete this.spref;
        return this.spref = async cb => {
            var val = Services.prefs.getIntPref(pref);
            Services.prefs.setIntPref(pref, 100);
            await new Promise(wait);
            Services.prefs.setIntPref(pref, val);
        }
    },
    async saveSession(name = this.getName(this.getState()), getTabLabel = this.getTabLabel()) {
        var saveName = this.prompter.prompt("Сохранить", getTabLabel );
        var name = (saveName + ", " + name );
        if (name == null || saveName == null) return true;

        if (this.exists(name)) return this.saveSession(name);

        await this.spref();
        this.data[name] = this.getState();

        this.meta.order.push(name);
        //this.meta.order.unshift(name);
        //if (this.meta.boot != null) this.meta.boot++;
    },
    restoreSession(name, win) {
        var ss = this.gs.SessionStore;
        var state = JSON.stringify(this.data[name]);
        win ? ss.setWindowState(win, state) : ss.setBrowserState(state);
    },
    renameSession(name, newName = name) {
        var newName = this.prompter.prompt(`Переименовать "${name}" в:`, newName);
        if (newName == null || newName == name) return true;

        if (this.exists(newName)) return this.renameSession(name, newName);

        var {data} = this;
        this.splice(name, newName);
        data[newName] = data[name];
        delete data[name];
    },
    removeSession(name) {
        if (!this.prompter.confirm(`Вы уверены, что хотите удалить: ${name} ?`))
            return true;
        delete this.data[name];
        this.splice(name);
    }
}).init()))("ucf-ssm-menupopup", "{07cae4f5-18b0-487b-80eb-973304af9528}");

Переключить текущий поисковик

Выделить код

Код:

//
try {CustomizableUI.createWidget({
    label: "Переключить текущий поисковик",
    id: "ucf_ToggleCurrentSearchEngine",
    localized: false,
    image: "data:image/png;base64,.....",
    excludeHiddenOneOffs: true,
    gn: () => Services.search.defaultEngine,
    gp: () => Services.search.defaultPrivateEngine,
    sn: val => Services.search.defaultEngine = val,
    sp: val => Services.search.defaultPrivateEngine = val,
    onCreated(btn) {
        btn.type = "menu";
        btn.owner = this;
        btn.setAttribute("image", this.image);

        var win = btn.ownerGlobal;
        var popup = btn.appendChild(win.document.createXULElement("menupopup"));
        var pr = win.PrivateBrowsingUtils.isWindowPrivate(win);
        popup.getDefaultEngine = pr ? this.gp : this.gn;
        popup.setDefaultEngine = pr ? this.sp : this.sn;
        popup.setAttribute("oncommand", "setDefaultEngine(event.target.engine)");
        popup.setAttribute("onpopupshowing", "this.shouldRebuild && owner.rebuild(this, document)");

        this.autoOpenCloseFeature(win, btn);
        this.updButton(btn, win);
    },
    getEngines() {
        var ve = Services.search.getVisibleEngines;
        if (!this.excludeHiddenOneOffs) return (this.getEngines = ve)();

        var arr = [];
        var args = this.fx116
            ? [e => !e.hideOneOffButton]
            : Object.defineProperty(
                [function(e) {return !this.includes(e.name);}], "1", {get: () => {
                    var str = Services.prefs.getStringPref(this.pref);
                    return str ? str.split(",") : arr;
                }}
            );
            return (this.getEngines = async () => (await ve()).filter(...args))();
    },
    async rebuild(popup, doc) {
        popup.textContent = "";
        var df = doc.createDocumentFragment();
        var de = popup.getDefaultEngine().wrappedJSObject, jsde = this.json(de);
        var check = true;
        for(var engine of await this.getEngines()) {
            if (check && engine.name == de.name && this.json(engine) == jsde) {
                check = false; continue;
            }
            var menuitem = df.appendChild(doc.createXULElement("menuitem"));
            menuitem.engine = engine;
            menuitem.label = engine.name;
            menuitem.className = "menuitem-iconic";
            menuitem.image = await this.img(engine);
        }
        popup.append(df);
        delete popup.shouldRebuild;
    },
    async updButton(btn, win) {
        this.updButton = () => {};
        Services.search.isInitialized || await Services.search.init();
        this.fx116 = "hideOneOffButton" in Services.search.defaultEngine;

        var topics = ["browser-search-engine-modified", "quit-application-granted"];
        for(var topic of topics) Services.obs.addObserver(this, topic, false);
        this.observe = (s, topic) => this[topic[0]]();

        var remove = () => topics.forEach(
            topic => Services.obs.removeObserver(this, topic)
        );
        var {id} = this;
        var wins = callback => {
            for(var win of CustomizableUI.windows) {
                var btn = win.document.getElementById(id);
                btn && callback(btn, win);
            }
        }
        if (this.excludeHiddenOneOffs && !this.fx116) {
            var setRebuild = btn => btn.firstChild.shouldRebuild = true;
            var {pref} = this, obs = () => wins(setRebuild);
            Services.prefs.addObserver(pref, obs);
            this.q = () => remove(Services.prefs.removeObserver(pref, obs));
        }
        else this.q = remove;

        var updButton = (btn, win) => {
            var popup = btn.firstChild;
            var engine = popup.getDefaultEngine();
            /*btn.label =*/ btn.tooltipText = engine.name;
            popup.shouldRebuild = true;
            win.requestAnimationFrame(async () => btn.icon.src = await this.img(engine));
        }
        (this.b = () => wins(updButton))();
        this.updButton = updButton;
        btn.tooltipText || updButton(btn, win);
    },
    pref: "browser.search.hiddenOneOffs",
    json: e => JSON.stringify(e.toJSON()),
    img: async e => await e.getIconURL?.() || e.iconURI?.spec || "chrome://browser/skin/search-engine-placeholder.png",

    // https://github.com/Infocatcher/Custom_Buttons/blob/master/code_snippets/autoOpenCloseMenu.js
    // Automatically open menu on mouse over (and hide it on mouse out)
    autoOpenCloseFeature(win, btn, openDelay = 20000, closeDelay = 350) {
        var _openTimer = 0;
        var _closeTimer = 0;
        btn.onmouseover = function(e) {
            win.clearTimeout(_closeTimer);
            if(e.target == btn && closeOtherMenus()) {
                btn.open = true;
                return;
            }
            _openTimer = win.setTimeout(function() {
                btn.open = true;
            }, openDelay);
        };
        btn.onmouseout = function(e) {
            win.clearTimeout(_openTimer);
            _closeTimer = win.setTimeout(function() {
                btn.open = false;
            }, closeDelay);
        };
        function closeOtherMenus() {
            return win.Array.prototype.some.call(
                btn.parentNode.getElementsByTagName("*"),
                function(node) {
                    if(
                        node != btn
                        && win.XULElement.isInstance(node)
                        // 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
                    ) {
                        node.open = false;
                        return true;
                    }
                    return false;
                }
            );
        }
    }
});} catch(ex) {Cu.reportError(ex);}

Отредактировано unter_officer (08-02-2025 01:19:14)


«The Truth Is Out There»

Отсутствует

 

№187508-02-2025 01:17:36

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 631
UA: Firefox 135.0

Re: UCF - ваши кнопки, скрипты…

Dumby, по мере тестирования выплыло ещё пару кнопок, которы у меня исправить не получается.
Если не трудно, поправьте пожалуйста и эти кнопочки.

Сохранить как PNG

Выделить код

Код:

//
(async func => CustomizableUI.createWidget({
    id: "ucf_SaveAsPNG",
    label: "Сохранить как PNG",
    tooltiptext: "Сохранить как PNG",
    localized: false,
    // defaultArea: CustomizableUI.AREA_NAVBAR,
    onCreated(btn) {
        var win = btn.ownerGlobal;
        new win.Function("_id, xhtmlns, addDestructor", func.toString().slice(7, -1)).call(
            btn, this.id, "http://www.w3.org/1999/xhtml",
            destructor => win.addEventListener("unload", destructor, {once: true})
        );
        btn.setAttribute("image", "");
    }
}))(() => {


((main, parts) => this._handleClick = () => {
    var df = MozXULElement.parseXULToFragment(`
        <menupopup>
            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить всю страницу как PNG"
                value="all"/>

            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить видимую часть страницы как PNG"
                value="page"/>

            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить выбранный элемент страницы как PNG"
                value="click"/>

            <menuitem class="menuitem-iconic"
                image=""
                label="Сохранить выбранную область страницы как PNG"
                value="clipping"/>
        </menupopup>
    `);
    var popup = df.firstChild;
    popup.setAttribute("context", "");
    popup.setAttribute("oncommand", "handleCommand(event);");
    popup.handleCommand = e => {
        var name = _id + ":DataURLReady";
        main = main.replace("%MESSAGE_NAME%", name);

        var urls = {}, configurable = true, enumerable = true;
        Object.entries(parts).forEach(([key, part]) => Object.defineProperty(urls, key, {
            configurable, enumerable, get() {
                var value = `data:;charset=utf-8,({${
                    encodeURIComponent(main + part)
                }%0A}).init("${key}")`;
                Object.defineProperty(urls, key, {configurable, enumerable, value});
                return value;
        }}));

        // Получить название вкладки без не сохраняемых символов и лишних пробелов .....
        var getTabLabel = () => {
            var label = gBrowser.selectedTab.label;
            var label = label.replace(/[:+.\\\/<>?*|"]+/g, " ").replace(/\s\s+/g, " ");
            return label.substring(0, 50);
        }

        var listener = msg => {
            var fp = makeFilePicker();
            // fp.init(window, "Сохранить как…", fp.modeSave); // Farby: https://forum.mozilla-russia.org/viewtopic.php?pid=809686#p809686 .....
            fp.init(
                !("inIsolatedMozBrowser" in window.browsingContext.originAttributes)
                  ? window.browsingContext
                  : window
                , "Сохранить как…", fp.modeSave);
            fp.appendFilter("", "*.png");

            var fileName = getTabLabel();
            fileName = fileName.replace(/[:\\\/<>?*|"]+/g, '').replace(/\s+/g, '_').slice(0, 100).replace(/^\s+|\s+$/g, '');
            var fileDate = (function () {
              var d = new Date(), z = function(n){return (n < 10 ? '0' : '') + n};
              return '[' + z(d.getFullYear()) + '_' + z(d.getMonth()+1) + '_' + z(d.getDate()) + '\u2014' + z(d.getHours()) + '_' + z(d.getMinutes()) + '_' + z(d.getSeconds()) + ']';
            })();

            fp.defaultString = fileName + "_" + fileDate + ".png";
            fp.open(res => res == fp.returnCancel || !fp.file || makeWebBrowserPersist().saveURI(
                Services.io.newURI(msg.data), document.nodePrincipal,
                null, null, null, null, null, fp.file, null, null
            ));
        }
        messageManager.addMessageListener(name, listener);
        addDestructor(() => messageManager.removeMessageListener(name, listener));

        (popup.handleCommand = e => gBrowser.selectedBrowser.messageManager
            .loadFrameScript(urls[e.target.value], false)
        )(e);
    }
    this.append(df);
    (this._handleClick = () => popup.openPopup(this, "after_start"))();
})(`
    init(cmd) {
        cmd.startsWith("c")
            ? this[cmd].init(this[cmd].parent = this)
            : this[cmd]();
    },
    capture(win, x, y, width, height) {
        var canvas = win.document.createElementNS("${xhtmlns}", "canvas");
        canvas.width = width;
        canvas.height = height;
        var ctx = canvas.getContext("2d");
        var tryDraw = ind => {
            try {ctx.drawWindow(win, x, y, canvas.width, canvas.height, "white")}
            catch(ex) {canvas.height = ind * canvas.width; tryDraw(--ind);}
        }
        tryDraw(17);
        sendAsyncMessage("%MESSAGE_NAME%", canvas.toDataURL("image/png"));
    },
    `, {

    all: `all() {
        var win = content;
        this.capture(win, 0, 0, win.innerWidth + win.scrollMaxX, win.innerHeight + win.scrollMaxY);
    }`,
    page: `page() {
        var win = content, doc = win.document, body = doc.body, html = doc.documentElement;
        var scrX = (body.scrollLeft || html.scrollLeft) - html.clientLeft;
        var scrY = (body.scrollTop || html.scrollTop) - html.clientTop;
        this.capture(win, scrX, scrY, win.innerWidth, win.innerHeight);
    }`,
    clipping: `clipping: {
        handleEvent(e) {
            if (e.button) return false;
            e.preventDefault();
            e.stopPropagation();
            switch(e.type) {
                case "mousedown":
                    this.downX = e.pageX;
                    this.downY = e.pageY;
                    this.bs.left = this.downX + "px";
                    this.bs.top = this.downY + "px";
                    this.body.appendChild(this.box);
                    this.flag = true;
                    break;
                case "mousemove":
                    if (!this.flag) return;
                    this.moveX = e.pageX;
                    this.moveY = e.pageY;
                    if (this.downX > this.moveX) this.bs.left = this.moveX + "px";
                    if (this.downY > this.moveY) this.bs.top  = this.moveY + "px";
                    this.bs.width = Math.abs(this.moveX - this.downX) + "px";
                    this.bs.height = Math.abs(this.moveY - this.downY) + "px";
                    break;
                case "mouseup":
                    this.uninit();
                    break;
            }
        },
        init() {
            var win = {};
            Cc["@mozilla.org/focus-manager;1"].getService(Ci.nsIFocusManager)
                .getFocusedElementForWindow(content, true, win);
            this.win = win.value;

            this.doc = this.win.document;
            this.body = this.doc.body;
            if (!HTMLBodyElement.isInstance(this.body)) {
                Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService)
                    .showAlertNotification("${self.image}", ${JSON.stringify(self.label)}, "Не удается захватить!");
                return false;
            }
            this.flag = null;
            this.box = this.doc.createElement("div");
            this.bs = this.box.style;
            this.bs.border = "red dashed 1px";
            this.bs.position = "absolute";
            this.bs.zIndex = "2147483647";
            this.defaultCursor = this.win.getComputedStyle(this.body, "").cursor;
            this.body.style.cursor = "crosshair";
            ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.addEventListener(type, this, true));
        },
        uninit() {
            var pos = [this.win, parseInt(this.bs.left), parseInt(this.bs.top), parseInt(this.bs.width), parseInt(this.bs.height)];
            this.body.style.cursor = this.defaultCursor;
            this.body.removeChild(this.box);
            this.parent.capture.apply(this, pos);
            ["click", "mouseup", "mousemove", "mousedown"].forEach(type=> this.doc.removeEventListener(type, this, true));
        }
    }`,
    click: `click: {
        getPosition() {
            var html = this.doc.documentElement;
            var body = this.doc.body;
            var rect = this.target.getBoundingClientRect();
            return [
                this.win,
                Math.round(rect.left) + (body.scrollLeft || html.scrollLeft) - html.clientLeft,
                Math.round(rect.top) + (body.scrollTop || html.scrollTop) - html.clientTop,
                parseInt(rect.width),
                parseInt(rect.height)
            ];
        },
        highlight() {
            this.orgStyle = this.target.hasAttribute("style") ? this.target.style.cssText : false;
            this.target.style.cssText += "outline: red 1px solid; outline-offset: 1px; -moz-outline-radius: 2px;";
        },
        lowlight() {
            if (this.orgStyle) this.target.style.cssText = this.orgStyle;
            else this.target.removeAttribute("style");
        },
        handleEvent(e) {
            switch(e.type){
                case "click":
                    if (e.button) return;
                    e.preventDefault();
                    e.stopPropagation();
                    this.lowlight();
                    this.parent.capture.apply(this, this.getPosition());
                    this.uninit();
                    break;
                case "mouseover":
                    if (this.target) this.lowlight();
                    this.target = e.target;
                    this.highlight();
                    break;
            }
        },
        init() {
            this.win = content;
            this.doc = content.document;
            ["click", "mouseover"].forEach(type=> this.doc.addEventListener(type, this, true));
        },
        uninit() {
            this.target = false;
            ["click", "mouseover"].forEach(type=> this.doc.removeEventListener(type, this, true));
        }
    }`
});
});

Автоматически перезагружать вкладку

Выделить код

Код:

//
(async (id, popup, self) => (self = {

    clickInterval: 5*60,
    intervals: [
        10, 15, 30, 60, 3*60, 5*60, 10*60, 15*60, 30*60, 45*60, 60*60,
    ],
    async init() {
        this.addStyle();
        var dsp = e => this[e.type](e);
        var tc = document.getElementById("tabbrowser-tabs");

        var trgs = [popup, tc, tc, document.getElementById("tabbrowser-tabpanels")];
        var types = ["popupshowing", "TabClose", "SSTabRestored", "EndSwapDocShells"];

        (this.destructor = (meth = "removeEventListener") => types.forEach(
            (type, ind) => trgs[ind][meth](type, dsp, ind == 3)
        ))("addEventListener");

        ucf_custom_scripts_win[id] = this;
        ucf_custom_scripts_win.unloadlisteners.push(id);

        await SessionStore.promiseAllWindowsRestored;
        for(var tab of gBrowser.tabs)
            tab.linkedPanel || this.maybeInitTab(tab);
    },
    maybeInitTab(tab) {
        var sec = this.sec(tab);
        sec && this.initTab(tab, sec, true);
    },
    mousedown(e) {
        if (e.button) return;
        e.stopImmediatePropagation();
        self.destroyTab(this.closest("tab"));
    },
    initTab(tab, sec, skipSet) {
        skipSet || SessionStore.setCustomTabValue(tab, id, sec);
        var img = document.createXULElement("hbox");
        img.className = id;
        // img.onmousedown = this.mousedown; // клик по иконке отключает перезагрузку вкладки .....

        tab.throbber.before(img);
        tab.setAttribute(id, setInterval(this.reload, sec * 1e3, tab));
    },
    destroyTab(tab) {
        clearInterval(tab.getAttribute(id));
        SessionStore.deleteCustomTabValue(tab, id);
        tab.removeAttribute(id);
        tab.querySelector("." + id).remove();
    },
    addStyle() {
        var css = `
            tab.tabbrowser-tab[${id}] .${id} {
                width: 16px;
                height: 16px;
                position: relative;
                margin-top: -1px;
                margin-inline-start: -2px;
                margin-inline-end: -14px;
                background-position: top right;
                background-repeat: no-repeat;
                background-image: url("");
                z-index: 1000;
            }
            tab.tabbrowser-tab[${id}]:-moz-locale-dir(rtl) .${id} {
                background-position: top right;
            }
            tab.tabbrowser-tab[${id}] .tab-icon-image {
                display: -moz-box;
            }
            tab.tabbrowser-tab[${id}][pendingicon] .tab-icon-image {
                visibility: hidden;
            }
            #context_autoreloadTab[checked] > menupopup > :nth-child(2),
            #context_autoreloadTab:not([checked]) > menupopup > :first-child {
                display: none;
            }
            #context_autoreloadTab[checked] > .menu-iconic-left > image {
                fill: currentColor;
                -moz-context-properties: fill;
                list-style-image: url("chrome://global/skin/icons/check.svg");
            }
            /*
            tab.tabbrowser-tab[${id}] .tab-throbber,
            tab.tabbrowser-tab[${id}] .tab-icon-pending,
            tab.tabbrowser-tab[${id}]:not([pendingicon]) .tab-icon-image:not([src],[busy],[pinned],[crashed],[sharing]) {
                display: none;
            }
            */
        `.replace(/;\s*\n/g, " !important;\n");
        windowUtils.loadSheetUsingURIString(
            "data:text/css," + encodeURIComponent(css), windowUtils.USER_SHEET
        );
    },
    get tab() {
        return TabContextMenu.contextTab;
    },
    sec(tab) {
        return SessionStore.getCustomTabValue(tab, id);
    },
    click(menu) {
        var {tab} = this;
        var has = menu.toggleAttribute("checked");
        has
            ? this.initTab(tab, this.clickInterval)
            : this.destroyTab(tab);

        var w = menu.clientWidth;
        this.setLabel(has && self.clickInterval);

        if (this.menupopup.state == "open")
            this.updMenupopup(),
            menu.clientWidth != w && setTimeout(this.move, 50);
    },
    changeInterval(tab, sec) {
        clearInterval(tab.getAttribute(id)),
        SessionStore.setCustomTabValue(tab, id, sec),
        tab.setAttribute(id, setInterval(this.reload, sec * 1e3, tab));
    },
    cmd(e) {
        var {value} = e.target;
        if (value == this.currSec) return;

        var {tab} = this;
        this.setLabel(value);

        if (this.menu.hasAttribute("checked"))
            this.changeInterval(tab, value);
        else
            this.menu.toggleAttribute("checked"),
            this.initTab(tab, value);
    },
    reload(tab) {
        gBrowser.reloadTab(tab);
    },
    get shouldHide() {
        return !this.tab.linkedBrowser.currentURI.scheme.startsWith("http");
    },
    format(sec) {
        var map = new Map();
        // resource://gre/modules/PluralForm.jsm
        var f = n => n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2;
        var hh = ["", "а", "ов"], ms = ["а", "ы", ""];
        return (this.format = sec => {
            var res = map.get(sec = +sec);
            if (!res) {
                var num, arr = [];
                if ((num = Math.floor(sec / 3600)) > 0)
                    sec -= num * 3600,
                    arr.push(`${num} час${hh[f(num)]}`);
                if ((num = Math.floor(sec / 60)) > 0)
                    sec -= num * 60,
                    arr.push(`${num} минут${ms[f(num)]}`);
                sec > 0 && arr.push(`${sec} секунд${ms[f(sec)]}`);
                map.set(sec, res = arr.join(" "));
            }
            return res;
        })(sec);
    },
    async prompt(val) {
        var {tab} = this, sec = this.sec(tab);
        var res = await Services.prompt.asyncPrompt(
            null, Services.prompt.MODAL_TYPE_WINDOW,
            val ? "ЕЩЁ РАЗ:" : "Задать интервал обновления",
            "Введите число секунд авто-обновления",
            val || sec || this.clickInterval, null, null
        );
        if (!res.get("ok")) return;

        var val = res.get("value");
        if (!val) return;
        if (!isFinite(val)) return this.prompt(val);

        var val = String(Math.round(val) || 1);
        sec ? this.changeInterval(tab, val) : this.initTab(tab, val);
    },
    initShadowDOM() {
        delete this.initShadowDOM;
        this.initShadowDOM();

        var df = MozXULElement.parseXULToFragment(
            `<menuitem closemenu="single" label="Не перезагружать"
                oncommand="event.stopPropagation(); parentNode.parentNode.click();"/>
            <menuitem closemenu="single" value="${self.clickInterval}"
                label="${self.format(self.clickInterval)}" type="radio"/>
            <menuitem label="Другой…"
                oncommand="event.stopPropagation(); parentNode.parentNode.linkedObject.prompt();"/>
            <menuseparator/>`
        );
        var menuitem = df.children[1];

        for(var sec of self.intervals) {
            if (sec == self.clickInterval) continue;
            menuitem = menuitem.cloneNode(false);
            menuitem.setAttribute("value", sec);
            menuitem.setAttribute("label", self.format(sec));
            df.append(menuitem);
        }
        this.append(df);
    },
    setLabel(sec) {
        this.menu.setAttribute("label", (this.currSec = sec)
            ? `Интервал перезагрузки:   ${this.format(sec)}`
            : "Задать интервал перезагрузки"
        );
    },
    popupshowing(e) {
        if (this.shouldHide) return;
        var df = MozXULElement.parseXULToFragment(
            `<menu id="context_autoreloadTab"
                class="menu-iconic"
                onclick="if (event.target == this) linkedObject.click(this)"
            >
                <menupopup oncommand="parentNode.linkedObject.cmd(event)"/>
            </menu>`
        );
        var menu = this.menu = df.firstChild;
        menu.linkedObject = this;
        var menupopup = this.menupopup = menu.firstChild;
        menupopup.initShadowDOM = this.initShadowDOM;
        popup.querySelector("#context_duplicateTab").after(menu);

        this.clickInterval = String(this.clickInterval);
        this.move = () => menupopup.moveToAnchor(menu, "end_before");

        this.updMenupopup = () => {
            var old = menupopup.querySelector("[checked=true]");
            var cur = this.currSec && menupopup.querySelector(`[value="${this.currSec}"]`);
            if (old != cur)
                old?.removeAttribute("checked"),
                cur && cur.setAttribute("checked", true);
        }
        (this.popupshowing = e => {
            if (e.target == popup) {
                if (menu.hidden = this.shouldHide) return;

                var sec = this.sec(this.tab);
                var has = menu.hasAttribute("checked");
                if (Boolean(sec) ^ has)
                    has = !has,
                    menu.toggleAttribute("checked");

                var curr = has && sec;
                curr !== this.currSec && this.setLabel(curr);
            }
            else if (e.target == menupopup) this.updMenupopup();
        })(e);
    },
    TabClose(e) {
        var intervalId = e.target.getAttribute(id);
        if (!intervalId) return;
        clearInterval(intervalId);

        var tab = e.detail.adoptedBy;
        tab?.ownerGlobal.ucf_custom_scripts_win[id].initTab(tab, this.sec(e.target));
    },
    SSTabRestored(e) {
        var tab = e.target;
        tab.hasAttribute(id) || this.maybeInitTab(tab);
    },
    async EndSwapDocShells(e) {
        var br = e.detail, trg = e.target;
        await new Promise(requestAnimationFrame);

        var win = br.ownerGlobal;
        if (!win.closed) return;
        var tab = win.gBrowser.getTabForBrowser(br);
        if (!tab) return;

        var sec = this.sec(tab);
        if (sec)
            tab = gBrowser.getTabForBrowser(trg),
            tab.hasAttribute(id) || this.initTab(tab, sec);
    }
}).init())("ucf-tab-auto-reload", document.getElementById("tabContextMenu"));


«The Truth Is Out There»

Отсутствует

 

Board footer

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