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

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

№1715101-09-2024 22:16:40

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

Re: Custom Buttons

vv07 пишет

Обойдусь

линк был не про обойдусь, а на решение, которое почему-то работает по сей день


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

Отсутствует

 

№1715202-09-2024 01:44:57

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

Re: Custom Buttons

Farby пишет
vv07 пишет

Обойдусь

линк был не про обойдусь, а на решение, которое почему-то работает по сей день

У меня не сработало увы, дружище. Поэтому и написал так.

Отсутствует

 

№1715327-09-2024 15:27:23

manuk
Участник
 
Группа: Members
Зарегистрирован: 17-10-2010
Сообщений: 306
UA: Firefox 129.0

Re: Custom Buttons

Приветствую. Не совсем по теме, но т.к. в кнопках и дополнениях много общего спрошу здесь. В 131 FF кудесники наваяли-изменили так, что перестали работать звуковые уведомления в приложении "Мои уведомления", а разработчик аддона пропал. https://addons.mozilla.org/ru/firefox/a … ent=search
Farby, Dumby, не могли бы вы посмотреть, что изменилось в воспроизведении звуков уведомлений FF131 и скрипты аддона в XPI. Может у вас получится исправить.

Отредактировано manuk (27-09-2024 15:35:23)

Отсутствует

 

№1715428-09-2024 09:33:15

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

Re: Custom Buttons

manuk пишет

Dumby, не могли бы вы посмотреть, что изменилось в воспроизведении звуков уведомлений FF131 и скрипты аддона в XPI

FF130, скорее. Какой-то очень странный баг.
Лиса козлится, пишет какую-то дичь, типа
не поддерживается, не удалось загрузить медиаресурс,
не удалось воспроизвести, нет декодеров ...


Думал, может дело в .ogg — но замена на .mp3 хорошего результата не дала.
Думал, может дело в moz-extension:// — но с консоли браузера audio играется.


Тем не менее, по протоколам data: и blob: вроде работает, поэтому,
пока не найдено лучшего решения или не исправлен баг, можно так попробовать

popup.js

Выделить код

Код:

/*
	playSound(sound, volume) {
		if (sound !== null && sound !== 'off') {
			let audio = new Audio()
			audio.src = sound
			audio.autoplay = true
			audio.volume = volume
		}
	}
*/
	async playSound(sound, volume) {
		if (!sound || sound == "off") return;

		var resolver = Promise.withResolvers();
		var reader = new FileReader();
		reader.onload = resolver.resolve;
		reader.readAsDataURL(await (await fetch(sound)).blob());
		await resolver.promise;

		var audio = new Audio();
		audio.src = reader.result;
		audio.autoplay = true;
		audio.volume = volume;
	}


background.js

Выделить код

Код:

/*
function playSound(siteSettings){
	let options = JSON.parse(localStorage.getItem('options'))
	let audio = new Audio()

	audio.autoplay = true
	audio.src = siteSettings.sound ? siteSettings.sound : options.sound
	audio.volume = options.volume
}
*/
var db = Object.create(null);

async function playSound(siteSettings) {
	var options = JSON.parse(localStorage.getItem("options"));
	var audio = new Audio();
	audio.autoplay = true;
	audio.volume = options.volume;

	var sound = siteSettings.sound || options.sound;
	audio.src = db[sound] ??= URL.createObjectURL(await (await fetch(sound)).blob());
}

Отсутствует

 

№1715528-09-2024 21:04:47

manuk
Участник
 
Группа: Members
Зарегистрирован: 17-10-2010
Сообщений: 306
UA: Firefox 130.0

Re: Custom Buttons

Dumby, большое спасибо. Прекрасно работает. Догадываюсь, что не одному мне это нужно. Где бы поделиться ссылкой на ваш пост с исправлениями аддона (может в отзывах о "Мои уведомления" на addons.mozilla.org)? Или, кто ищет - тот всегда найдёт?

Отредактировано manuk (28-09-2024 21:23:39)

Отсутствует

 

№1715602-10-2024 09:49:49

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

Re: Custom Buttons

manuk пишет

Где бы поделиться ссылкой

Увы, я не знаю где.


Подломали Custom Buttons.


Bug 71895 - Remove Hidden Window from Linux and Windows builds (Firefox 132+)
Bug 1917745 - Remove dom.window.sizeToContent.enabled pref (Firefox 132+)


Custom Buttons 0.0.7.0.0.34

Отсутствует

 

№1715705-10-2024 18:49:26

Garalf
Участник
 
Группа: Members
Зарегистрирован: 19-09-2017
Сообщений: 324
UA: Firefox 129.0

Re: Custom Buttons

Dumby. перестала работать кнопка Экспорт в HTML файл в контекстном меню закладок

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

Выделить код

Код:

/*Initialization Code*/
//-------------------- Экспорт в HTML файл в контекстном меню закладок
(popup => addEventListener("popupshowing", {
    handleEvent() {
        if (this.shouldHide()) return;
        var before = document.getElementById("placesContext_openSeparator");
Можно поправить?
        var menuitem = popup.insertBefore(document.createXULElement("menuitem"), before);
        menuitem.setAttribute("label", "Экспорт папки в HTML");
        menuitem.setAttribute("oncommand", "exportFolder();");
        menuitem.exportFolder = this.pick.bind(this);
        addDestructor(() => menuitem.remove());
        (this.handleEvent = () => menuitem.hidden = menuitem.disabled = this.shouldHide())();
    },
    shouldHide() {
        var node = popup.triggerNode._placesNode;
        var hide = !node || node.type != node.RESULT_TYPE_FOLDER;
        if (!hide) this.guid = node.bookmarkGuid, this.title = node.title;
        return hide;
    },
    pick() {
        var fp = makeFilePicker();
        fp.init(window, PlacesUIUtils.getString("EnterExport"), fp.modeSave);
        fp.appendFilters(fp.filterHTML);
        fp.defaultString = (this.title ? DownloadPaths.sanitize(this.title) : "untitled") + ".html";
        fp.open(res => res == fp.returnCancel || this.export(fp.file.path));
    },
    async export(path) {
        var tree = await PlacesUtils.promiseBookmarksTree(
            this.guid, {includeItemIds: true}
        );
        var bookmarks = {children: [
            {root: "toolbarFolder"},
            {root: "unfiledBookmarksFolder"},
            {root: "bookmarksMenuFolder", children: [tree], guid: PlacesUtils.bookmarks.menuGuid}
        ]};
        new this.nsvo.BookmarkExporter(bookmarks).exportToFile(path);
    },
    get nsvo() {
        delete this.nsvo;
        return this.nsvo = Cu.import("resource://gre/modules/BookmarkHTMLUtils.jsm", {});
    }
}, false, popup))(document.getElementById("placesContext") || 1);


Можно поправить?

Отредактировано Garalf (05-10-2024 20:12:16)

Отсутствует

 

№1715806-10-2024 09:52:41

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

Re: Custom Buttons

Garalf пишет
Выделить код

Код:

Можно поправить?

А чего это «Можно поправить?» делает в коде?
Подлежит удалению.

перестала работать кнопка

Очень информативно. Хорошо, 129

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

Выделить код

Код:

/*
        fp.init(window, PlacesUIUtils.getString("EnterExport"), fp.modeSave);
*/
        fp.init(window.browsingContext, PlacesUIUtils.promptLocalization.formatValueSync("places-bookmarks-export"), fp.modeSave);

Отсутствует

 

№1715915-10-2024 03:07:51

kazarin
Участник
 
Группа: Members
Зарегистрирован: 23-11-2016
Сообщений: 87
UA: Firefox 126.0

Re: Custom Buttons

Dumby, большая просьба глянуть, если вам несложно. В браузере r3dfox 128 (и у 130) возник конфликт у последней CB (0.0.7.0.0.34-fx-paxmod) и utils от xiaoxiaoflood, конкретно с версией для скриптов и расширений. С версией только для скриптов конфликта нет, но мне расширения тоже нужны.
Выражается в том, что на странице дополнений становится невозможно посмотреть кнопки или детали расширений, всё перестаёт кликаться, вызывается только окно новой кнопки. В остальном всё работает, меню кнопок можно редактировать через меню правого клика. Проблема только на этой страничке.

скрытый текст
4SqeR9R.gif

Если удалить buttonsoverlay.xul, то проблема пропадает. Проверил на виртуалке, такая же штука происходит и в обычном [firefox] 131.
Хотел попробовать предыдущую версию CB, но ссылка умерла. Старые utils не работают.

И проблема с одной кнопкой (тоже в 128-130), перестала работать перезагрузка страницы в урлбаре, вот этот код

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

Выделить код

Код:

if (gURLBar.focused) {
var str = gURLBar.value;
  getBrowser (). selectedTab = getBrowser (). addTab (str);
  gURLBar.handleCommand();  
 }
 else  {
  BrowserReload();
 }


На всякий случай полный код кнопки
скрытый текст

Выделить код

Код:

self.image = "";


var urlbar=document.getElementById("page-action-buttons");
var button=document.getElementById(_id);
urlbar.insertBefore(button,urlbar.Child);

 this.onclick=e=>
{if(e.button==0)Rld();                                                                        //L
 if(e.button==1)gShowPopup(this);                                                             //M
 if(e.button==2 && !event.ctrlKey && !event.shiftKey && !event.altKey && !event.metaKey){e.preventDefault();CURL();}};//R
this.tooltipText="L: Reload\nR: CopyURL";

////////////////////////////////////////////////////////////////////
function Rld(){if (gURLBar.focused) {
var str = gURLBar.value;
  getBrowser (). selectedTab = getBrowser (). addTab (str);
  gURLBar.handleCommand();  
 }
 else  {
  BrowserReload();
 } };
////////////////////////////////////////////////////////////////////
function CURL(){gClipboard.write(gURLBar.value);
        var gub=gURLBar.value,line1='   Адрес скопирован в буфер !',line=" "+line1,i=0;
        function line_a(){
        if(i++<line.length){
        gURLBar.value=line.substring (1,i)}
        setTimeout(function(){line_a()},10)}
        line_a();
        setTimeout(function(){gURLBar.value = readFromClipboard()},1500)};
////////////////////////////////////////////////////////////////////

Отсутствует

 

№1716015-10-2024 12:13:08

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

Re: Custom Buttons

kazarin пишет

mega.nz

Серьёзно?! Ещё гэ-драйв предложи.


https://www.upload.ee/
Но даже он тут не обязателен, ибо utils невелик размером,
вполне поместится на форум в base64

такая же штука происходит и в обычном [firefox] 131

Ладно, попробовал взять utils у onemen'а.
Конфликт похожий вижу.
Помогло подправить BootstrapLoader.js

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

Выделить код

Код:

`/*
      if (addon.__AddonInternal__.optionsType == 1/*AddonManager.OPTIONS_TYPE_DIALOG*/)
/*`;
      if (addon.__AddonInternal__?.optionsType == 1) // AddonManager.OPTIONS_TYPE_DIALOG

Хотел попробовать предыдущую версию CB, но ссылка умерла.

Конечно, такое долго не живёт.
Если кто-то не исключает вероятности,
что может понадобиться «попробовать предыдущую»,
или любую другую, версию, то ожидается,
что он возьмёт её, вовремя скачанную, у себя с диска.

перестала работать перезагрузка страницы

Обсуждалось уже.

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

Выделить код

Код:

/*
  BrowserReload();
*/
  BrowserCommands.reload();

На всякий случай полный код кнопки

Функция CURL() ужасна!
При каждом вызове она образует бесконечную петлю
непрекращающейся таймаут-долбёжки, каждый раз новую,
сто раз в секунду, двести, триста...
Интересно, когда браузеру с такого поплохеет.
Вот, надеюсь, такая получше

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

Выделить код

Код:

function CURL() {
	var msg = "   gURLBar.value скопировано в буфер !";
	var tick = r => setTimeout(r, 10);
	var field = gURLBar.inputField, ed = field.editor;
	var {input} = gURLBar.view;
	var revert = () => {
		input._suppressStartQuery = true;
		ed.undo();
		input._suppressStartQuery = false;
	}
	(CURL = async () => {
		gClipboard.write(gURLBar.value);
		ed.beginTransaction();
		field.value = "";
		for(char of msg) field.value += char, await new Promise(tick);
		ed.endTransaction();
		setTimeout(revert, 1500);
	})();
}

Отсутствует

 

№1716115-10-2024 23:08:05

kazarin
Участник
 
Группа: Members
Зарегистрирован: 23-11-2016
Сообщений: 87
UA: Firefox 126.0

Re: Custom Buttons

Dumby

Я извиняюсь, а что не так с мегой? Сто лет пользуюсь. Сперва не понял, потом как понял. Вы вот так не глядя, можете коды писать? Это что-то слишком запредельное, буквально сверхспособности:)

Dumby пишет

Помогло подправить BootstrapLoader.js

Да, работает! Один знак всего... Спасибо огромное!

Dumby пишет

что он возьмёт её, вовремя скачанную, у себя с диска.

Так не успел я вовремя:( какое-то время вообще на форум не заходил.

Dumby пишет

Обсуждалось уже.

Я искал, но не нашёл:(

Dumby пишет

Интересно, когда браузеру с такого поплохеет.

Функцию писал не я я даже так не могу, из этой темы брал, до сих пор проблем не было. И спасибо ещё раз, всё отлично работает!

Отредактировано kazarin (15-10-2024 23:43:49)

Отсутствует

 

№1716218-10-2024 00:03:10

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

Re: Custom Buttons

Dumby

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

Выделить код

Код:

// Дополнительные возможности для значка идентификации сайта в строке адреса .....
 (identBox => {
    var tip = "Л: Добавить закладку\nП: О странице";

    var icon = gIdentityHandler._identityIcon;
    addEventListener("mouseenter", () => icon.setAttribute("tooltiptext", tip), false, icon || 1);
    addDestructor(() =>
        icon.setAttribute("tooltiptext", gNavigatorBundle.getString("identity.icon.tooltip"))
    );
    var listener = {
        
        handleEvent(e) {
            e.ctrlKey || e.shiftKey || e.detail > 1 || this[e.type](e);
            
        },
        click(e) {
            if (e.button || !identBox.contains(e.target)) return;
            e.stopPropagation();
            e.preventDefault();
            this.bookmarkCurrentPage(gBrowser.selectedBrowser, true);
            identBox.setAttribute("style", "background: linear-gradient(#0080FF , blue) !important;border-radius: 2px !important; ");
            setTimeout(() => identBox.removeAttribute("style"), 1400);
        },
        contextmenu(e) {
            e.preventDefault();
            var url = gBrowser.currentURI.spec;
            BrowserPageInfo(url, url.startsWith("http") ? "permTab" : "permTab");
            identBox.setAttribute("style", "background: linear-gradient(#0080FF , red) !important;border-radius: 2px !important; ");
            setTimeout(() => identBox.removeAttribute("style"), 1400);
        },
        get bookmarkCurrentPage() {
            delete this.bookmarkCurrentPage;
            return this.bookmarkCurrentPage = eval(`(${
                PlacesCommandHook.bookmarkPage.toSource()
                    .replace("async", "$& function")
                    .replace("unfiledGuid", "menuGuid")
            })`).bind(PlacesCommandHook);
        }
    };
    addEventListener("click", listener, true, identBox.parentNode);
    addEventListener("contextmenu", listener, false, identBox);
})(document.getElementById("identity-box") || 1);


Как бы на среднюю reload вставить ?

Отредактировано ВВП (18-10-2024 00:07:53)

Отсутствует

 

№1716318-10-2024 03:18:32

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

Re: Custom Buttons

Уважаемый Dumby! Перезалейте пожалуйста Custom Buttons 0.0.7.0.0.34

Отредактировано vv07 (18-10-2024 03:18:56)

Отсутствует

 

№1716418-10-2024 08:13:15

vending_machine
Участник
 
Группа: Members
Зарегистрирован: 10-01-2020
Сообщений: 558
UA: Firefox 115.0

Re: Custom Buttons

vv07 пишет

Перезалейте пожалуйста Custom Buttons 0.0.7.0.0.34

>>><<<

Отсутствует

 

№1716518-10-2024 10:45:58

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

Re: Custom Buttons

ВВП пишет

Как бы на среднюю reload вставить ?

После click(e) {
добавить
if (e.button == 1) return window.BrowserReload ? BrowserReload() : BrowserCommands.reload();


vv07 пишет

Перезалейте пожалуйста Custom Buttons 0.0.7.0.0.34

Вот, с небольшими правками.

Отсутствует

 

№1716618-10-2024 12:09:56

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

Re: Custom Buttons

Dumby пишет

if (e.button == 1) return window.BrowserReload ? BrowserReload() : BrowserCommands.reload();

Класс !

Отсутствует

 

№1716718-10-2024 15:08:11

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

Re: Custom Buttons

Bug 1919853 Make DevTools work with CSSNestedDeclarations objects. [firefox] 132+
Больно ударил по DOMi, но есть такой вариант восстановить

chrome\inspector\content\inspector\viewers\styleRules\styleRules.js

Выделить код

Код:

/*
    this.mRules = viewer.DOMUtils.getCSSStyleRules(aObject);
*/
    this.mRules = "getMatchingCSSRules" in viewer.DOMUtils
      ? viewer.DOMUtils.getMatchingCSSRules?.(aObject)
      : viewer.DOMUtils.getCSSStyleRules(aObject);


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

Отсутствует

 

№1716830-10-2024 14:07:32

manuk
Участник
 
Группа: Members
Зарегистрирован: 17-10-2010
Сообщений: 306
UA: Firefox 132.0

Re: Custom Buttons

Приветствую. Дошли руки до 132.0. И что? Как обычно "отвалилась" кнопочка" findbar.

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

Выделить код

Код:

/*Initialization Code*/
((bar, button = true, insertAtTop = true, ctrlFcloseFinbar = false) => ({
    init(parent) {
        var has = bar = parent.querySelector("#appcontent > findbar");
        has || this.initFinbar(parent);
        var lo = bar.linkedObject;
        lo.listenCtrlF = ctrlFcloseFinbar
            ? listen => listen
                ? addEventListener("keydown", lo, true)
                : removeEventListener("keydown", lo, true)
            : () => {};
        has && !bar.hidden && lo.listenCtrlF(true);
        if (button) self._handleClick = () => bar.hidden
            ? bar.startFind(bar.FIND_NORMAL)
            : bar.collapsed || bar.close();
        addDestructor(lo.destroy, lo);
    },
    destroy(reason) {
        if (reason[5] != "e") return;
        bar.close();
        bar._browser = {};
        bar.remove();
        this.setProgressListener(false);
        if (!this.receiver) this.actorProto.receiveMessage = this.actorReceiveMessage;
        for(var key of ["gFindBar", "gFindBarInitialized"])
            key in this && Object.defineProperty(window, key, this[key]);
        for(key of this.gBrKeys) gBrowser[key] = this[key];
        Services.ppmm.removeDelayedProcessScript(this.url);
        Services.ppmm.loadProcessScript("data:," + encodeURIComponent(`
            (ai => ai.processType == ai.PROCESS_TYPE_DEFAULT || ai.processType == ai.PROCESS_TYPE_CONTENT)(
                Cc["@mozilla.org/xre/app-info;1"].getService(Ci.nsIXULRuntime)
            ) && (nsvo => {
                var proto = nsvo.Finder.prototype;
                if ("_requestMatchesCount" in proto) {
                    proto.requestMatchesCount = proto._requestMatchesCount;
                    delete proto._requestMatchesCount;
                }
            })(Cu.import("resource://gre/modules/Finder.jsm", {}));`
        ) , false);
    },
    initFinbar(parent) {
        for(var tab of gBrowser.tabs) {
            if (!tab._findBar) continue;
            tab._findBar.browser = null;
            tab._findBar._browser = {};
            tab._findBar.remove();
            delete tab._findBar;
            
        }
        bar = document.createXULElement("findbar");
        var p = new Proxy({}, {get: () => () => {}});
        bar._browser = {finder: p, messageManager: p};
        parent.insertBefore(bar, insertAtTop ? parent.firstChild : null);
        bar.linkedObject = this;

        ["gFindBar", "gFindBarInitialized"].forEach((key, ind) => {
            var desc = Object.getOwnPropertyDescriptor(window, key);
            if (!desc.configurable) return;
            this[key] = desc;
            delete window[key];
            window[key] = ind ? true : bar;
        });
        var key = "getCachedFindBar" in gBrowser ? "getCachedFindBar" : "getFindBar";
        this.gBrKeys = [key];
        key = "isFindBarInitialized";
        if (key in gBrowser) this.gBrKeys.push(key);
        this.gBrKeys.forEach((key, ind) => {
            this[key] = gBrowser[key];
            gBrowser[key] = ind ? () => true : () => bar;
        });
        var props = [
            "close", "startFind", "onMatchesCountResult",
            "_updateMatchesCount", "_onBrowserKeypress"
        ];
        if ((this.receiver = "receiveMessage" in bar))
            props.push("receiveMessage");
        else {
            this.actorProto = Object.getPrototypeOf(
                gBrowser.selectedBrowser.browsingContext.currentWindowGlobal.getActor("FindBar")
            );
            this.actorReceiveMessage = this.actorProto.receiveMessage;
            this.actorProto.receiveMessage = msg => {
                if (msg.name == "Findbar:Keypress") bar._onBrowserKeypress(msg.data);
                else if (msg.name == "Findbar:Mouseup") bar.onMouseUp();
            }
        }
        props.forEach((key, ind) => {
            var func = bar[key].bind(bar);
            bar[key] = ind
                ? (...args) => this[key](...args) || func(...args)
                : (...args) => func(...args) || this[key](...args);
        });
        this.url = "data:," + encodeURIComponent(`
            (ai => ai.processType == ai.PROCESS_TYPE_DEFAULT || ai.processType == ai.PROCESS_TYPE_CONTENT)(
                Cc["@mozilla.org/xre/app-info;1"].getService(Ci.nsIXULRuntime)
            ) && (nsvo => {
                var proto = nsvo.Finder.prototype;
                if ("_requestMatchesCount" in proto) return;
                proto._requestMatchesCount = proto.requestMatchesCount;
                proto.requestMatchesCount = ${
                    this.newRequestMatchesCount
                }
            })(Cu.import("resource://gre/modules/Finder.jsm", {}));`
        );
        delete this.newRequestMatchesCount;
        Services.ppmm.loadProcessScript(this.url, true);

        var desc = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value");
        var setValue = desc.set.bind(bar._findField);
        desc.set = val => Components.stack.formattedStack.includes(
            "set browser@chrome://global/content/elements/findbar.js"
        ) ? val : setValue(val);
        Object.defineProperty(bar._findField, "value", desc);
    },
    newRequestMatchesCount: async function requestMatchesCount(aWord, aLinksOnly, aUseSubFrames = true) {
        if (aLinksOnly && aLinksOnly.constructor.name == "Object") {
            var {linksOnly, data} = aLinksOnly;
            aLinksOnly = linksOnly;

            this.entireWord = data.entireWord;
            this.caseSensitive = data.caseSensitive;
            this.onModalHighlightChange(data.useModalHighlight);
            this.onHighlightAllChange(data.highlightAll);
            data.highlightAll && await this.highlighter.highlight(true, aWord, linksOnly);
            this._iterator && this._iterator.reset();
            var obj;
            Object.defineProperty(this, "_currentMatchesCountResult", {
                configurable: true, enumerable: true,
                get: (val = obj) => {
                    if (val && val._currentFound)
                        val.total = Math.floor(val.total) + .0001;
                    return obj = val;
                },
                set: val => {
                    if (val) {
                        val.total = 10000;
                        return obj = val;
                    }
                    delete this._currentMatchesCountResult;
                    return obj = this._currentMatchesCountResult = val;
                }
            });
        }
        return await this._requestMatchesCount(aWord, aLinksOnly, aUseSubFrames);
    },

    close() {
        bar.collaped = false;
        this.setProgressListener(false);
        this.setBrowser(null, null);
    },
    startFind() {
        if (this.maybeCollapse(gBrowser.selectedBrowser))
            return true;
        if (bar.hidden)
            this.setBrowser(300),
            this.setProgressListener(true);
        else if (!ctrlFcloseFinbar)
            setTimeout(() => this.updateMatchesCount(), 100);
    },
    get pf() {
        delete this.pf;
        return this.pf = bar.pluralForm || ChromeUtils.import(
            "resource://gre/modules/PluralForm.jsm"
        ).PluralForm;
    },
    onMatchesCountResult(res) {
        if (res.total <= 1000) return;

        var strTotal = String(res.total);
        var found = strTotal.includes(".");
        if (found) strTotal = strTotal.split(".")[0];
        if ((res.total = +strTotal.slice(-4)) >= 1000) {
            res.total = -1;
            return;
        }
        if (res.current > 1000)
            res.current = +String(res.current).slice(-4);

        if (res.current && found) return;

        bar._foundMatches.value = `${+res.total || "Нет"} совпадени${
            this.pf.get(res.total, "е;я;й")
        }.`;
        bar._foundMatches.hidden = false;
        return true;
    },
    _updateMatchesCount() {
        return true;
    },
    _onBrowserKeypress(e) {
        if (!bar.hidden) return;
        if (!e.charCode) return true;
        this.setBrowser(300);
        this.setProgressListener(true);
    },
    receiveMessage(msg) {
        msg.target = bar._browser;
    },

    progressListenerAdded: false,
    setProgressListener(add) {
        if (add) {
            if (this.progressListenerAdded) return;
            this.progressListenerAdded = true;
            gBrowser.addProgressListener(this);
            this.listenCtrlF(true);
        } else {
            if (!this.progressListenerAdded) return;
            this.progressListenerAdded = false;
            gBrowser.removeProgressListener(this);
            this.listenCtrlF(false);
        }
    },
    handleEvent(e) {
        if (
            e.ctrlKey && e.code == "KeyF"
            && !e.shiftKey && !e.altKey && !bar.collapsed
        )
            e.preventDefault(),
            e.stopPropagation(),
            bar.close();
    },
    updateMatchesCount() {
        var str = bar._findField.value;
        if (!str) return;
        var data = {
            entireWord: bar._entireWord,
            caseSensitive: bar._typeAheadCaseSensitive,
            highlightAll: bar._highlightAll,
            useModalHighlight: bar._useModalHighlight
        };
        bar.browser.finder.requestMatchesCount(
            bar._findField.value,
            {linksOnly: bar._findMode == bar.FIND_LINKS, data}
        );
    },
    maybeCollapse(br) {
        return br.isSyntheticDocument ||
            br.documentContentType == "application/vnd.mozilla.xul+xml";
    },
    setBrowser(updateDelay, br = gBrowser.selectedBrowser) {
        if (bar._browser != br) {
            var b = bar._browser;
            if (b) {
                this.receiver && b.messageManager
                    .removeMessageListener("Findbar:Mouseup", bar);
                b.finder.removeResultListener(bar);
                bar._highlightAll && b.finder.highlight(false);
            }
            if (br) {
                this.receiver && br.messageManager
                    .addMessageListener("Findbar:Mouseup", bar);
                bar._updateBrowserWithState();
            }
            bar._browser = br;
        }
        if (!br) return;
        bar._updateStatusUI();
        bar._foundMatches.value = "";
        br.finder.addResultListener(bar);
        if (
            !(bar.collapsed = this.maybeCollapse(br))
            && br.currentURI.spec != "about:blank"
            && updateDelay !== null
        )
            updateDelay
                ? setTimeout(this.updateMatchesCount, updateDelay)
                : this.updateMatchesCount();
    },
    onStateChange(wpr, req, state) {
        state & Ci.nsIWebProgressListener.STATE_STOP && this.setBrowser();
    },
    onLocationChange(wpr, req) {
        req || wpr.isLoadingDocument ||
            gBrowser.selectedTab.hasAttribute("pending") || this.setBrowser();
    }
}).init(document.getElementById("appcontent")))();

Хорошо, что только одна из 15-ти. Не особо правильная была, но получше стандартного поиска. Dumby, может исправите, если не в тягость.

Отредактировано manuk (30-10-2024 14:11:03)

Отсутствует

 

№1716930-10-2024 21:24:16

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

Re: Custom Buttons

manuk пишет

если не в тягость

Шутишь? Она была в тягость с самого начала.


Отвал на 132 — это Bug 1916098 - Remove appcontent box.
Ладно, чисто формально, немного подправил-почистил всякое старьё.
Копаться в этом, тестировать, желания нет, и впредь, лучше меня о ней не спрашивать.

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

Выделить код

Код:

((bar, button = true, insertAtTop = true, ctrlFcloseFinbar = false) => ({
	init(parent) {
		var has = bar = parent.querySelector(":scope > findbar");
		has || this.initFinbar(parent);
		var lo = bar.linkedObject;
		lo.listenCtrlF = ctrlFcloseFinbar
			? listen => listen
				? addEventListener("keydown", lo, true)
				: removeEventListener("keydown", lo, true)
			: () => {};
		has && !bar.hidden && lo.listenCtrlF(true);
		if (button) self._handleClick = () => bar.hidden
			? bar.startFind(bar.FIND_NORMAL)
			: bar.collapsed || bar.close();
		addDestructor(lo.destroy, lo);
	},
	destroy(reason) {
		if (reason[5] != "e") return;
		bar.close();
		bar._browser = {};
		bar.remove();
		this.setProgressListener(false);
		this.actorProto.receiveMessage = this.actorReceiveMessage;
		for(var key of ["gFindBar", "gFindBarInitialized"])
			key in this && Object.defineProperty(window, key, this[key]);
		for(key of this.gBrKeys) gBrowser[key] = this[key];
		Services.ppmm.removeDelayedProcessScript(this.url);

		Services.ppmm.loadProcessScript("data:," + encodeURIComponent(`(url => {
			var ai = Services.appinfo, pt = ai.processType;
			if (pt != ai.PROCESS_TYPE_DEFAULT && pt != ai.PROCESS_TYPE_CONTENT) return;
			var proto = ChromeUtils.importESModule(url).Finder.prototype;
			if ("_requestMatchesCount" in proto)
				proto.requestMatchesCount = proto._requestMatchesCount,
				delete proto._requestMatchesCount;
		})("resource://gre/modules/Finder.sys.mjs");`), false);
	},
	initFinbar(parent) {
		for(var tab of gBrowser.tabs) {
			if (!tab._findBar) continue;
			tab._findBar.browser = null;
			tab._findBar._browser = {};
			tab._findBar.remove();
			delete tab._findBar;
		}
		bar = document.createXULElement("findbar");
		var p = new Proxy({}, {get: () => () => {}});
		bar._browser = {finder: p, messageManager: p};
		parent.insertBefore(bar, insertAtTop ? parent.firstChild : null);
		bar.linkedObject = this;

		["gFindBar", "gFindBarInitialized"].forEach((key, ind) => {
			var desc = Object.getOwnPropertyDescriptor(window, key);
			if (!desc.configurable) return;
			this[key] = desc;
			delete window[key];
			window[key] = ind ? true : bar;
		});

		(this.gBrKeys = ["getCachedFindBar", "isFindBarInitialized"]).forEach((key, ind) => {
			this[key] = gBrowser[key];
			gBrowser[key] = ind ? () => true : () => bar;
		});
		[
			"close", "startFind", "onMatchesCountResult",
			"_updateMatchesCount", "_onBrowserKeypress"
		]
			.forEach((key, ind) => {
				var func = bar[key].bind(bar);
				bar[key] = ind
					? (...args) => this[key](...args) || func(...args)
					: (...args) => func(...args) || this[key](...args);
			});

		this.actorProto = Object.getPrototypeOf(
			gBrowser.selectedBrowser.browsingContext.currentWindowGlobal.getActor("FindBar")
		);
		this.actorReceiveMessage = this.actorProto.receiveMessage;
		this.actorProto.receiveMessage = msg => {
			if (msg.name == "Findbar:Keypress") bar._onBrowserKeypress(msg.data);
			else if (msg.name == "Findbar:Mouseup") bar.onMouseUp();
		}

		this.url = "data:," + encodeURIComponent(`(url => {
			var ai = Services.appinfo, pt = ai.processType;
			if (pt != ai.PROCESS_TYPE_DEFAULT && pt != ai.PROCESS_TYPE_CONTENT) return;
			var proto = ChromeUtils.importESModule(url).Finder.prototype;
			if ("_requestMatchesCount" in proto) return;
			proto._requestMatchesCount = proto.requestMatchesCount;
			proto.requestMatchesCount = ${this.newRequestMatchesCount}
		})("resource://gre/modules/Finder.sys.mjs");`);

		delete this.newRequestMatchesCount;
		Services.ppmm.loadProcessScript(this.url, true);

		var desc = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value");
		var setValue = desc.set.bind(bar._findField);
		desc.set = val => Components.stack.formattedStack.includes(
			"set browser@chrome://global/content/elements/findbar.js"
		) ? val : setValue(val);
		Object.defineProperty(bar._findField, "value", desc);
	},
	newRequestMatchesCount: async function requestMatchesCount(aWord, aLinksOnly, aUseSubFrames = true) {
		if (aLinksOnly && aLinksOnly.constructor.name == "Object") {
			var {linksOnly, data} = aLinksOnly;
			aLinksOnly = linksOnly;

			this.entireWord = data.entireWord;
			this.caseSensitive = data.caseSensitive;
			this.onModalHighlightChange(data.useModalHighlight);
			this.onHighlightAllChange(data.highlightAll);
			data.highlightAll && await this.highlighter.highlight(true, aWord, linksOnly);
			this._iterator && this._iterator.reset();
			var obj;
			Object.defineProperty(this, "_currentMatchesCountResult", {
				configurable: true, enumerable: true,
				get: (val = obj) => {
					if (val && val._currentFound)
						val.total = Math.floor(val.total) + .0001;
					return obj = val;
				},
				set: val => {
					if (val) {
						val.total = 10000;
						return obj = val;
					}
					delete this._currentMatchesCountResult;
					return obj = this._currentMatchesCountResult = val;
				}
			});
		}
		return await this._requestMatchesCount(aWord, aLinksOnly, aUseSubFrames);
	},

	close() {
		bar.collaped = false;
		this.setProgressListener(false);
		this.setBrowser(null, null);
	},
	startFind() {
		if (this.maybeCollapse(gBrowser.selectedBrowser))
			return true;
		if (bar.hidden)
			this.setBrowser(300),
			this.setProgressListener(true);
		else if (!ctrlFcloseFinbar)
			setTimeout(() => this.updateMatchesCount(), 100);
	},
	onMatchesCountResult(res) {
		if (res.total <= 1000) return;

		var strTotal = String(res.total);
		var found = strTotal.includes(".");
		if (found) strTotal = strTotal.split(".")[0];
		if ((res.total = +strTotal.slice(-4)) >= 1000) {
			res.total = -1;
			return;
		}
		if (res.current > 1000)
			res.current = +String(res.current).slice(-4);

		if (res.current && found) return;

		var n = res.total, fm = bar._foundMatches;
		fm.value = `${n || "Нет"} совпадени${"еяй"[
			n % 10 == 1 && n % 100 != 11 ? 0 : n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20) ? 1 : 2
		]}.`;
		fm.hidden = false;
		fm.removeAttribute("data-l10n-id");
		return true;
	},
	_updateMatchesCount() {
		return true;
	},
	_onBrowserKeypress(e) {
		if (!bar.hidden) return;
		if (!e.charCode) return true;
		this.setBrowser(300);
		this.setProgressListener(true);
	},

	progressListenerAdded: false,
	setProgressListener(add) {
		if (add) {
			if (this.progressListenerAdded) return;
			this.progressListenerAdded = true;
			gBrowser.addProgressListener(this);
			this.listenCtrlF(true);
		} else {
			if (!this.progressListenerAdded) return;
			this.progressListenerAdded = false;
			gBrowser.removeProgressListener(this);
			this.listenCtrlF(false);
		}
	},
	handleEvent(e) {
		if (
			e.ctrlKey && e.code == "KeyF"
			&& !e.shiftKey && !e.altKey && !bar.collapsed
		)
			e.preventDefault(),
			e.stopPropagation(),
			bar.close();
	},
	updateMatchesCount() {
		var str = bar._findField.value;
		if (!str) return;
		var data = {
			entireWord: bar._entireWord,
			caseSensitive: bar._typeAheadCaseSensitive,
			highlightAll: bar._highlightAll,
			useModalHighlight: bar._useModalHighlight
		};
		bar.browser.finder.requestMatchesCount(
			bar._findField.value,
			{linksOnly: bar._findMode == bar.FIND_LINKS, data}
		);
	},
	maybeCollapse: br => br.isSyntheticDocument,

	setBrowser(updateDelay, br = gBrowser.selectedBrowser) {
		if (bar._browser != br) {
			var b = bar._browser;
			if (b) {
				b.finder.removeResultListener(bar);
				bar._highlightAll && b.finder.highlight(false);
			}
			if (br) bar._updateBrowserWithState();

			bar._browser = br;
		}
		if (!br) return;
		bar._updateStatusUI();
		bar._foundMatches.value = "";
		br.finder.addResultListener(bar);
		if (
			!(bar.collapsed = this.maybeCollapse(br))
			&& br.currentURI.spec != "about:blank"
			&& updateDelay !== null
		)
			updateDelay
				? setTimeout(this.updateMatchesCount, updateDelay)
				: this.updateMatchesCount();
	},
	onStateChange(wpr, req, state) {
		state & Ci.nsIWebProgressListener.STATE_STOP && this.setBrowser();
	},
	onLocationChange(wpr, req) {
		req || wpr.isLoadingDocument ||
			gBrowser.selectedTab.hasAttribute("pending") || this.setBrowser();
	}
}).init(document.getElementById("tabbrowser-tabbox")))();

Отсутствует

 

№1717031-10-2024 08:33:27

manuk
Участник
 
Группа: Members
Зарегистрирован: 17-10-2010
Сообщений: 306
UA: Firefox 132.0

Re: Custom Buttons

Копаться в этом, тестировать, желания нет, и впредь, лучше меня о ней не спрашивать.

Dumby, понял. Спасибо.

Отсутствует

 

№1717131-10-2024 16:14:35

Alex_one
Участник
 
Группа: Members
Зарегистрирован: 27-09-2015
Сообщений: 151
UA: Firefox 115.0

Re: Custom Buttons

Всем привет!
Ребят, подскажите, что могло пойти не так, как исправить??
Хотел кнопку добавить, а тут такое.
[firefox] 115.17

скрытый текст
31-10-2024-160901.jpg

Отсутствует

 

№1717201-11-2024 10:30:36

fuchsfan
Участник
 
Группа: Members
Зарегистрирован: 07-08-2023
Сообщений: 131
UA: Firefox 128.0

Re: Custom Buttons

Alex_one пишет

Хотел кнопку добавить, а тут такое.

Добавляйте, на работоспособность не влияет, тоже с этим сталкивался.
cb.png

Отсутствует

 

№1717308-11-2024 14:03:44

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

Re: Custom Buttons

Dumby
На https://addons.mozilla.org/ru/firefox/ никак версию не обмануть ? Ковырянием в omni и настройках - не катит. Панчить исходник и DLL ?
Манифест рихтовать всякий раз приходится...

Отсутствует

 

№1717409-11-2024 00:06:05

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

Re: Custom Buttons

ВВП пишет

https://addons.mozilla.org

Дела контентские...
Увы, здесь я бесполезен.


Тем не менее, скрипты-то запрещены,
и я не вижу, чтобы AMO превратили в говнище.


Вот посетил попс, и смог скачать
как заявленную актуальную версию (1.60.0),
так и какую-то двадцать второго года (1.44.4).
Ну, в смысле, через ПКМ —> «Сохранить объект как…»

Отсутствует

 

№1717525-11-2024 20:21:44

Garalf
Участник
 
Группа: Members
Зарегистрирован: 19-09-2017
Сообщений: 324
UA: Firefox 132.0

Re: Custom Buttons

Dumby
Перестала работать кнопка Экспорт закладок в HTML-файл

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

Выделить код

Код:

(obj => {
    this._handleClick = () => obj.popup.openPopup(this, "after_start");
    this.onmouseenter = e => this.tooltipText = this.label +
        "\nЛ: Меню кнопки\nП: CB меню\n\nПапка для экспорта:\n" + (obj.path || "Не установлено.");
    addDestructor(reason => reason == "delete" && Services.prefs.clearUserPref(obj.pref));
})({
    get popup() {
        var popup = document.createElementNS(xulns,"menupopup");
        popup.setAttribute("oncommand", "linkedObject[event.target.value]();");
        popup.linkedObject = this;
        var keys = ["label", "value", "image"];
        for(var data of [
            ["Экспорт закладок в HTML-файл", "export", ""],
            ["Импорт закладок из HTML-файла", "import", ""],
            ,
            ["Открыть папку для экспорта закладок", "reveal", ""],
            ["Экспорт закладок в HTML без запроса", "silentExport", ""]
        ])
            if (data) {
                var menuitem = popup.appendChild(document.createElementNS(xulns,"menuitem"));
                menuitem.className = "menuitem-iconic";
                keys.forEach((key, ind) => menuitem.setAttribute(key, data[ind]));
            } else
                popup.appendChild(document.createElementNS(xulns,"menuseparator"));
        delete this.popup;
        return this.popup = self.appendChild(popup);
    },
    get BookmarkHTMLUtils() {
        delete this.BookmarkHTMLUtils;
        Cu.import("resource://gre/modules/BookmarkHTMLUtils.jsm", this);
        return this.BookmarkHTMLUtils;
    },
    pref: "CB.exportsBookmarksToHTMLFile.path",
    get path() {
        return Services.prefs.getStringPref(this.pref, null);
    },
    pick(title, modeOpen) {
        var fp = makeFilePicker();
        fp.init(window, title, modeOpen ? fp.modeOpen : fp.modeGetFolder);
        modeOpen && fp.appendFilters(fp.filterHTML);
        var {path} = this;
        if (path) fp.displayDirectory = FileUtils.File(path);
        return new Promise(resolve => fp.open(
            res => resolve(res == fp.returnOK && fp.file)
        ));
    },
    async import() {
        var file = await this.pick("Выберите HTML-файл для импорта закладок", true);
        file && this.BookmarkHTMLUtils.importFromFile(file.path).then(null, alert);
    },
    async export(justSetPath) {
        var dir = await this.pick("Укажите папку для экспорта закладок!");
        if (!dir) return;
        dir.path != this.path && Services.prefs.setStringPref(this.pref, dir.path);
        justSetPath || this.silentExport(dir);
    },
    silentExport(dir) {
        if (!dir && !(dir = this.checkPath(this.path))) return;
        dir.append("bookmarks-" + new Date().toLocaleDateString("mn") + ".html");
        this.BookmarkHTMLUtils.exportToFile(dir.path)
            .then(() => this.notify(dir.path), alert);
    },
    ns: "Папка для экспорта не установлена.\n\nУстановить папку?",
    nf: path => `Папка для экспорта не найдена.\n${path}\n\nВыбрать другую?`,
    checkPath(path, justSetPath) {
        if (!path)
            return void(confirm(this.ns) && this.export(justSetPath));
        var dir = FileUtils.File(path);
        if (dir.exists() && dir.isDirectory()) return dir;
        confirm(this.nf(path)) && this.export(justSetPath);
    },
    reveal() {
        var dir = this.checkPath(this.path, true);
        dir && dir.reveal();
    },
    get notify() {
        var as = Cc["@mozilla.org/alerts-service;1"].getService(Ci.nsIAlertsService);
        var func = path => setTimeout(as.closeAlert, 3500, as.showAlertNotification(
            self.image, self.label, "Экспортировал закладки как HTML в " + path
        ));
        delete this.notify; return this.notify = func;
    }
});

Посмотри пожалуйста...

Отсутствует

 

Board footer

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