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

Список ответов на каверзные вопросы можно получить в FAQ-разделе форума.

№1395111-12-2019 17:27:14

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
В71 размер иконок СВ заманал... Как уменьшить?

скрытый текст
.card-heading-icon > .addon[type="custombuttons"] .icon (
  width: 16px !important;
  height: 16px !important;
)

Отсутствует

 

№1395211-12-2019 18:43:29

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

Re: Custom Buttons

solombala
Вот парочка вариантов, которые работают.
Может ещё как-нибудь можно. А вот как лучше — без понятия.

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

Выделить код

Код:

addon-card[addon-id^="custombutton://buttons/"] .addon-icon {
    width: 16px !important;
    height: 16px !important;
    /*
    padding-top: 3px !important;
    */
}
Выделить код

Код:

addon-list[type=custombuttons] .addon-icon {
    width: 16px !important;
    height: 16px !important;
    /*
    padding-top: 3px !important;
    */
}

Отсутствует

 

№1395311-12-2019 19:12:45

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
Если знать куда вставлять, то шикардос! Так, раз уж ты король кнопок , как тут выровнять? Значок могу шевелить , а текст то не опустится...Тако курвье будали ово мазиловци.
fzn79br5.png

Отредактировано solombala (12-12-2019 11:48:41)

Отсутствует

 

№1395412-12-2019 14:01:48

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

Re: Custom Buttons

solombala пишет

как тут выровнять? Значок могу шевелить , а текст то не опустится...

Попробуй так:

Выделить код

Код:

addon-list[type=custombuttons] .card-contents {
    padding-top: 3px !important;
}

Отредактировано unter_officer (12-12-2019 14:02:02)


«The Truth Is Out There»

Отсутствует

 

№1395512-12-2019 14:20:42

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

unter_officer
Шикарно! Жаль размер текста никак...

Отсутствует

 

№1395612-12-2019 15:25:12

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

Re: Custom Buttons

solombala пишет

Жаль размер текста никак...

addon-list[type=custombuttons] .addon-name {
    font-size: 22px !important;
}


«The Truth Is Out There»

Отсутствует

 

№1395712-12-2019 15:53:19

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

unter_officer

unter_officer пишет

addon-list[type=custombuttons] .addon-name {
    font-size: 22px !important;
}

Не срабатывает. И самих аддонов тоже не могу сделать.

Отсутствует

 

№1395812-12-2019 16:22:15

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

Re: Custom Buttons

solombala пишет

И самих аддонов тоже не могу сделать.

В этом коде font-size не для всех аддонов, а только для СВ.

solombala пишет

Не срабатывает.

Не срабатывает возможно из-за версии [firefox], я на 68 ESR проверял, там срабатывает. А [firefox] 71+ я пока не устанавливал, поэтому проверить не могу.


«The Truth Is Out There»

Отсутствует

 

№1395913-12-2019 08:37:08

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
В 71 отрихтованая Вами Flip-Close_Tab ....ПКМ -  на prev Tab - не пашет ( это я па правый переделал  , но сам код не трогал) В 70 - океу...

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

Выделить код

Код:

var {interfaces: Ci, utils: Cu} = Components;
Cu.import("resource://gre/modules/Services.jsm");

var flipclosetabStatic = {
    customjsm: false,
    initialized: false,
    get styleURI() {
        delete this.styleURI;
        return this.styleURI = Services.io.newURI("chrome://flip_close_tab/content/button.css", null, null);
    },
    get buttonStrings() {
        delete this.buttonStrings;
        return this.buttonStrings = Services.strings.createBundle("chrome://flip_close_tab/locale/button.properties");
    },
    get CustomizableUI() {
        delete this.CustomizableUI;
        var _CustomizableUI = null;
        try {
            _CustomizableUI = Cu.import("resource:///modules/CustomizableUI.jsm", {}).CustomizableUI;
        } catch(e) {
            try {
                _CustomizableUI = Cu.import("chrome://flip_close_tab/content/customizable.jsm", {}).CustomizableUI;
                this.customjsm = true;
            } catch(e) {}
        }
        return this.CustomizableUI = _CustomizableUI;
    },
    start: function(win) {
        var onLoad = () => {
            win.removeEventListener("load", onLoad);
            this.loadIntoWindow(win);
        };
        win.addEventListener("load", onLoad);
    },
    get loadIntoWindow() {
        delete this.loadIntoWindow;
        this.init();
        return this.loadIntoWindow = function(win) {
            var obj = new FlipCloseTab();
            obj.load(win);
            Object.defineProperty(win, "_Flip_Close_Tab_Prototype", {
                value: obj,
                writable: false,
                configurable: true,
                enumerable: false
            });
        };
    },
    init: function() {
        if (this.initialized || !this.CustomizableUI) return;
        this.initialized = true;
        var cWidget = {
            id: "f-flip-tabs",
            type: "custom",
            label: flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.label"),
            tooltiptext: flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.tooltip"),
            defaultArea: this.CustomizableUI.AREA_NAVBAR,
            onBuild: function(document) {
                var window = document.defaultView;
                flipclosetabStatic.stylifyButton(window, true);
                var toolbarbutton_0 = (document.createXULElement || document.createElement).call(document, "toolbarbutton");
                toolbarbutton_0.id = "f-flip-tabs";
                toolbarbutton_0.setAttribute("label", flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.label"));
                toolbarbutton_0.setAttribute("tooltiptext", flipclosetabStatic.buttonStrings.GetStringFromName("f-flip-tabs.tooltip"));
                toolbarbutton_0.setAttribute("context", false);
                toolbarbutton_0.classList.add("toolbarbutton-1");
                toolbarbutton_0.classList.add("chromeclass-toolbar-additional");
                toolbarbutton_0.addEventListener("click", function(event) {
                    var gtBrowser = window.gBrowser;
                    if (event.button == 2) {
                        var old = gtBrowser.selectedTab;
                        var tabs = Array.filter(old.parentNode.getElementsByAttribute("flipselectedID", "*"), tab => !tab.hidden && !tab.closing);
                        var last  = 2;
                        var prevTab = null;
                        tabs.forEach((tab)=> {
                            let s = +tab.getAttribute("flipselectedID");
                            if (s && s > last && old != tab) {
                                prevTab = tab;
                                last = s;
                            }
                        });
                        if (prevTab !== null) 
                        gtBrowser.selectedTab = prevTab;
                    } else if (event.button == 1) {
                        gtBrowser.removeAllTabsBut(gtBrowser.selectedTab);
                    } else if (event.button == 0) {
                        event.preventDefault();
                        event.stopPropagation();
                        gtBrowser.removeTab(gtBrowser.selectedTab);
                    }
                }, false);
                return toolbarbutton_0;
            }
        };
        if (this.customjsm) {
            this.CustomizableUI.addListener();
            cWidget.onloadStylePalette = function(window) {
                flipclosetabStatic.stylifyButton(window, true);
            };
        }
        try {
            this.CustomizableUI.createWidget(cWidget);
        } catch(e) {}
    },
    getWindowUtils: function(window) {
        var has = "windowUtils" in window &&
            window.windowUtils instanceof Ci.nsIDOMWindowUtils;
        return (this.getWindowUtils = has
            ? window => window.windowUtils
            : window => window.QueryInterface(Ci.nsIInterfaceRequestor)
                .getInterface(Ci.nsIDOMWindowUtils)
        )(window);
    },
    stylifyButton: function(window, stylify) {
        var method = stylify ? "loadSheet" : "removeSheet";
        try {
            this.getWindowUtils(window)[method](
                this.styleURI, Ci.nsIDOMWindowUtils.USER_SHEET
            );
        } catch(ex) {
            Cu.reportError(ex);
        }
    },
    uninit: function() {
        if (!this.initialized) return;
        try {
            this.CustomizableUI.destroyWidget("f-flip-tabs");
        } catch(e) {}
        if (this.customjsm) {
            this.CustomizableUI.removeListener();
            Cu.unload("chrome://flip_close_tab/content/customizable.jsm");
        }
        var windows = Services.wm.getEnumerator("navigator:browser");
        while (windows.hasMoreElements()) {
            var win = windows.getNext();
            if (win.QueryInterface) win = win.QueryInterface(Ci.nsIDOMWindow);
            try {
                if ("_Flip_Close_Tab_Prototype" in win) {
                    win._Flip_Close_Tab_Prototype.unload(true);
                    delete win._Flip_Close_Tab_Prototype;
                }
                this.stylifyButton(win);
            } catch(e) {}
        }
    }
};

var filterTabs = node => {
    var func = tab => !tab.hidden && !tab.closing;
    return (filterTabs = "from" in Array
        ? node => Array.from(node.getElementsByAttribute("flipselectedID", "*")).filter(func)
        : node => Array.filter(node.getElementsByAttribute("flipselectedID", "*"), func)
    )(node);
}

function FlipCloseTab() {
    this.gtBrowser = null;
    this.removeListeners = null;
    this.blurTab = null;
    this.eventTabClose = false;
}

FlipCloseTab.prototype = {
    load: function(win) {
        var gtBrowser = this.gtBrowser = win.gBrowser || win.getBrowser();
        var tabid = 1;
        var TabSelect = (event) => {
            var tab = event.target;
            tab.setAttribute("flipselectedID", tabid++);
        };
        var TabClose = (event) => {
            var old = event.target;
            if (!old.selected) return;
            var tabs = filterTabs(old.parentNode);
            var last  = 0;
            var prevTab = null;
            tabs.forEach((tab) => {
                var s = +tab.getAttribute("flipselectedID");
                if (s && s > last && old != tab) {
                    prevTab = tab;
                    last = s;
                }
            });
            if (prevTab !== null && !prevTab.closing)
                gtBrowser.selectedTab = prevTab;
        };
        gtBrowser.tabContainer.addEventListener("TabSelect", TabSelect, false);
        if ("_blurTab" in gtBrowser) {
            var blurTab = this.blurTab = gtBrowser._blurTab;
            gtBrowser._blurTab = function _blurTab() {
                var old = arguments[0];
                if (!old.selected) return;
                var tabs = filterTabs(old.parentNode);
                var last  = 0;
                var prevTab = null;
                tabs.forEach((tab) => {
                    var s = +tab.getAttribute("flipselectedID");
                    if (s && s > last && old != tab) {
                        prevTab = tab;
                        last = s;
                    }
                });
                if (prevTab !== null && !prevTab.closing) {
                    this.selectedTab = prevTab;
                    return;
                }
                return blurTab.apply(this, arguments);
            };
        } else {
            gtBrowser.tabContainer.addEventListener("TabClose", TabClose, true);
            this.eventTabClose = true;
        }
        this.removeListeners = function() {
            gtBrowser.tabContainer.removeEventListener("TabSelect", TabSelect, false);
            if (this.eventTabClose)
                gtBrowser.tabContainer.removeEventListener("TabClose", TabClose, true);
        };
        var sel = gtBrowser.selectedTab;
        if (sel && !sel.hidden)
            sel.setAttribute("flipselectedID", tabid++);
    },
    unload: function(shutdown = false) {
        this.removeListeners();
        if (shutdown) {
            if (this.blurTab !== null)
                this.gtBrowser._blurTab = this.blurTab;
            var tabs = filterTabs(this.gtBrowser.tabContainer);
            tabs.forEach((tab) => {
                tab.removeAttribute("flipselectedID");
            });
        }
    }
};

var WindowListener = {
    onOpenWindow: function(aWindow) {
        var win = aWindow.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindow);
        function onWindowLoad() {
            win.removeEventListener("load", onWindowLoad);
            if (win.document.documentElement.getAttribute("windowtype") == "navigator:browser")
                flipclosetabStatic.loadIntoWindow(win);
        }
        win.addEventListener("load", onWindowLoad);
    },
    onCloseWindow: function(aWindow) {
        var win = aWindow.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindow);
        if (win.document.documentElement.getAttribute("windowtype") == "navigator:browser" && "_Flip_Close_Tab_Prototype" in win)
            win._Flip_Close_Tab_Prototype.unload();
    },
    onWindowTitleChange: function(aWindow, newTitle) { }
};

function startup(data, reason) {
    var wm = Services.wm;
    var windows = wm.getEnumerator(null);
    while (windows.hasMoreElements()) {
        var win = windows.getNext();
        if (win.QueryInterface) win = win.QueryInterface(Ci.nsIDOMWindow);
        try {
            var type = win.document.documentElement.getAttribute("windowtype");
            if (type == "navigator:browser")
                flipclosetabStatic.loadIntoWindow(win);
            else if (type == "navigator:blank")
                flipclosetabStatic.start(win);
        } catch(e) {}
    }
    wm.addListener(WindowListener);
}

function shutdown(data, reason) {
    if (reason == APP_SHUTDOWN)
        return;
    flipclosetabStatic.uninit();
    Services.wm.removeListener(WindowListener);
}

function install(data, reason) { }

function uninstall(data, reason) { }

Отсутствует

 

№1396013-12-2019 13:59:57

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

Re: Custom Buttons

solombala пишет

ПКМ -  на prev Tab - не пашет ( это я па правый переделал  , но сам код не трогал)

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

Выделить код

Код:

toolbarbutton_0.addEventListener("click", function(event) {
                    var gtBrowser = window.gBrowser;
                    if (event.button == 2) {
                        event.preventDefault();
                        event.stopPropagation();
                        var old = gtBrowser.selectedTab;
                        var tabs = filterTabs(old.parentNode);
                        var last  = 0;
                        var prevTab = null;
                        tabs.forEach((tab)=> {
                            let s = +tab.getAttribute("flipselectedID");
                            if (s && s > last && old != tab) {
                                prevTab = tab;
                                last = s;
                            }
                        });
                        if (prevTab !== null) 
                        gtBrowser.selectedTab = prevTab;
                    } else if (event.button == 1) {
                        gtBrowser.removeAllTabsBut(gtBrowser.selectedTab);
                    } else if (event.button == 0) {
                        gtBrowser.removeTab(gtBrowser.selectedTab);
                    }
                }, false);

Отсутствует

 

№1396113-12-2019 14:31:03

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
Круто , очень ! Нема речи!

Отсутствует

 

№1396214-12-2019 16:22:49

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

Re: Custom Buttons

Есть кнопка Proxy. В ней есть функция:

Выделить код

Код:

// Функция открывает настройки прокси в окне ...................  
function openConnections() {
   self.win && self.win.close();
   self.win = openDialog("chrome://browser/content/preferences/connection.xul", "Proxy", "centerscreen");
   
   // добавить атрибут "prefwindow"
   self.win.addEventListener("load", function f(e) {
       this.removeEventListener("load", f, true); 
       e.target.documentElement.setAttribute("type", "prefwindow");
   }, true);

   // закрыть настройки прокси по клику на странице 
   gBrowser.addEventListener("click", function c() {
      this.removeEventListener("click", c);
      try { self.win.close() } catch(e) {}; 
   }, true);
};

В целом функция работает нормально - открывает настройки прокси в окне, но при этом в консоли появляется ошибка: TypeError: window.opener.gSubDialog is undefined
Поскольку всё работает, ошибка особо не напрягает, но всё же, возможно ли это поправить под [firefox] 68 ESR?


«The Truth Is Out There»

Отсутствует

 

№1396314-12-2019 16:57:42

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
Что с мульти в 71? Код config.js  ? В портабл в ini это MOZ_FORCE_DISABLE_E10S=1
Раньше обходилось  config.js и кнопкой   cbu.setPrefs(s, cbu.getPrefs(s) == true ? false : true);
Теперь наглухо только  MOZ_FORCE_DISABLE_E10S .... " Проблема небольшая с раскрытием подробностей в about:addons (кнопки СВ - редактирование) и аддоны тоже..
Шняга  вылазит ..Типа, не может протокол обработать ...
https://forum.mozilla-russia.org/viewto … 94#p775894

Отредактировано solombala (14-12-2019 17:29:03)

Отсутствует

 

№1396414-12-2019 20:17:29

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

Re: Custom Buttons

unter_officer пишет

поправить под [firefox] 68 ESR

Может так

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

Выделить код

Код:

function openConnections() {
    var url = "chrome://browser/content/preferences/connection.xul";
    var win = [...Services.wm.getEnumerator(null)].find(w => w.location == url);
    if (win) return win.focus();

    win = openDialog(url, "Proxy", "centerscreen");
    win.opener = {gSubDialog: {_dialogs: [{
        _frame: {get contentDocument() {
            var args = ["click", win.close.bind(win), true];
            var unload = () => gBrowser.removeEventListener(...args);
            gBrowser.addEventListener(...args);
            win.addEventListener("unload", unload, {once: true});

            delete this.contentDocument;
            return this.contentDocument = win.document;
        }},
        resizeVertically: () => win.sizeToContent()
    }]}};
}

solombala пишет

Раньше обходилось  config.js и кнопкой

Это похерили ещё в 70. Я пробовал тогда раскопать,
и мне показалось, что вот здесь расклад запрашивается
для решения, запускать ли socket-процесс, а будучи запрошенным,
всё, поздно уже что-то в config.js переставлять.
Но, разумеется, это гадание на кофейной гуще, факт лишь в том,
что работать перестало, и с этим ничего не поделаешь.

solombala пишет

Шняга  вылазит

Да, у меня воспроизводится, если отключить многопроцессность,
и кликать по названию аддона.
Как любителю правки omni.ja, могу предложить попробовать
добавить e.preventDefault(); в aboutaddons.js, вот сюда

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

Выделить код

Код:

default:
          // Handle a click on the card itself.
          if (
            !this.expanded &&
            (e.target === this.addonNameEl || !e.target.closest("a"))
          ) {
            e.preventDefault(); // <= ДОБАВИТЬ
            loadViewFn(`detail/${this.addon.id}`, e);
          } else if (

Отсутствует

 

№1396514-12-2019 20:31:57

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

Re: Custom Buttons

Dumby пишет

Может так

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

Выделить код

Код:

function openConnections() {
    var url = "chrome://browser/content/preferences/connection.xul";
    var win = [...Services.wm.getEnumerator(null)].find(w => w.location == url);
    if (win) return win.focus();

    win = openDialog(url, "Proxy", "centerscreen");
    win.opener = {gSubDialog: {_dialogs: [{
        _frame: {get contentDocument() {
            var args = ["click", win.close.bind(win), true];
            var unload = () => gBrowser.removeEventListener(...args);
            gBrowser.addEventListener(...args);
            win.addEventListener("unload", unload, {once: true});

            delete this.contentDocument;
            return this.contentDocument = win.document;
        }},
        resizeVertically: () => win.sizeToContent()
    }]}};
}

То, что надо! Спасибо!


«The Truth Is Out There»

Отсутствует

 

№1396614-12-2019 21:10:35

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
На аддонах вроде перестало вылазить...А на кнопках  СВ - также...Может там рихтануть?

Отсутствует

 

№1396714-12-2019 21:18:56

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

Re: Custom Buttons

solombala пишет

А на кнопках  СВ - также...Может там рихтануть?

Да, точно, в CB тоже e.preventDefault() надо дописать.
Я добавлю в addons-html.js, спасибо.

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

Выделить код

Код:

click(e) {
        if (this.cbAddonList && !e.button && !e.target.matches(this.ignoreClickSelectors)) {
            var addon = a(e);
            if (addon) e.stopPropagation(), e.preventDefault(), e.detail == 1 && this.editButton(addon);
        }
    },

Отсутствует

 

№1396814-12-2019 21:31:03

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Dumby
Курвье  маjка ! Три дня мудохался , пока мульти определил... Благо есть такие врачи ...Хвала, брате! Чува тебе Бог!

browser.tabs.remote.autostart - true // На СВ - влияет , только true !
Очередная поганка! https://forum.mozilla-russia.org/viewto … 53#p775953
Нужен код в config.js ///Эта штука в ini MOZ_FORCE_DISABLE_E10S=  все портит...

Отредактировано solombala (15-12-2019 20:45:42)

Отсутствует

 

№1396916-12-2019 12:15:21

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

Re: Custom Buttons

Dumby скажи пожалуйста в вновь создаваемых кнопках в секциях Код и Инициализация отсутствуют /*CODE*/ и /*Initialization Code*/ так должно быть или только у меня?
Версия 71 и custom_buttons-0.0.7.0.0.8-fx-paxmod.xpi, кеш сброшен

В кнопке

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

Выделить код

Код:

/*Initialization Code*/

((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);
            fp.appendFilter("", "*.png");
            fp.defaultString = getTabLabel() + ".png";   
            fp.open(res => {
                if (res == fp.returnCancel || !fp.file) return;
                var wbp = makeWebBrowserPersist(), args = [
                    Services.io.newURI(msg.data), document.nodePrincipal,
                    null, null, null, null, fp.file, null
                ];
                wbp.saveURI.length == 9 && args.splice(2, 0, null);
                wbp.saveURI(...args);
            });
        }
        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 = "#0f0 dashed 2px";
            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 2px solid; outline-offset: 2px; -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));
        }
    }`
});


перестали работать 2 и 3 пункты, ссылаясь на var scrX = (body.scrollLeft || html.scrollLeft) - html.clientLeft;
Глянте пожайлуста.

solombala пишет

Доп пункты в about;addons

обсуждалось на 556 стр. со значками и без. Мой вариант

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

Выделить код

Код:

//Дополнительные пункты контекстного меню на странице about:addons для аддонов, плагинов, тем, CB.....................................................................................
((id, g, iconizer) => addDestructor(r => r[5] == "e" && id in g && g[id].destroy()) + addEventListener("shown", {
    //------------------------------------------------------------------
    "Копировать имя_i": "",
    "Копировать имя"(addon, hideOn) {
        if (hideOn) return false;

        gClipboard.write(addon.name);
    },
    //------------------------------------------------------------------
    "Копировать ID_i": "",
    "Копировать ID"(addon, hideOn) {
        if (hideOn) return false;

        gClipboard.write(addon.id);
    },
    //------------------------------------------------------------------
    "Копировать версию_i": "",
    "Копировать версию"(addon, hideOn) {
        if (hideOn) return ["custombuttons"];

        gClipboard.write(addon.version);
    },
    //------------------------------------------------------------------
    "Копировать имя и версию_i": "",
    "Копировать имя и версию"(addon, hideOn) {
        if (hideOn) return ["custombuttons"];

        gClipboard.write(addon.name + " " + addon.version);
    },
    //------------------------------------------------------------------
    "Домашняя страница_i": "",
    "Домашняя страница"(addon, hideOn) {
        if (hideOn) return !addon.homepageURL;

        var url = addon.homepageURL;
        if (!url) {
        if (addon.reviewURL) {
        url = addon.reviewURL.replace(/\/reviews\/.*$/, "/");
        }
        }
       openURL(url);
    },
    //------------------------------------------------------------------
    "Поиск на АМО_i": "",
    "Поиск на АМО"(addon, hideOn) {
        if (hideOn) return ["custombuttons","theme","plugin"];

        var url = addon.homepageURL;
        if (!url) {
        url = "https://addons.mozilla.org/search/?q="
            + encodeURIComponent(addon.name);
        }
       openURL(url);
    },
    //------------------------------------------------------------------
    "Папка установки_i": "",
        "Папка установки"(addon, hideOn) {
        if (hideOn) return ["custombuttons","theme","plugin"];

                    switch (addon.type) {
                    case "plugin":
                    var pathes = addon.pluginFullpath;
                    for (var i = 0; i < pathes.length; i++) {
                        this.revealPath(pathes[i]);
                    }
                    return;
                case "userchromejs":
                    var file = addon._script.file;
                    if (file.exists())
                        file.reveal();
                    return;
            }

            var gecko = parseInt(Services.appinfo.platformVersion);
            var nsLocalFile = Components.Constructor("@mozilla.org/file/local;1", (gecko >= 14) ? "nsIFile" : "nsILocalFile",
                "initWithPath");

            var dir = Services.dirsvc.get("ProfD", Ci.nsIFile);
            dir.append("extensions");
            dir.append(addon.id);
            var fileOrDir = dir.path + (dir.exists() ? "" : ".xpi");

            try {
                (new nsLocalFile(fileOrDir)).reveal();
            } catch (ex) {
                var addonDir = /.xpi$/.test(fileOrDir) ? dir.parent : dir;
                try {
                    if (addonDir.exists()) {
                        addonDir.launch();
                    }
                } catch (ex) {
                    var uri = Services.io.newFileURI(addonDir);
                    var protSvc = Cc["@mozilla.org/uriloader/external-protocol-service;1"].
                    getService(Ci.nsIExternalProtocolService);
                    protSvc.loadUrl(uri);
                }
            }
    },
    //------------------------------------------------------------------
    "Файл установки_i": "",
        "Файл установки"(addon, hideOn) {
        if (hideOn) return ["custombuttons","theme","plugin"];

            var gecko = parseInt(Services.appinfo.platformVersion);
            var nsLocalFile = Components.Constructor("@mozilla.org/file/local;1", (gecko >= 14) ? "nsIFile" : "nsILocalFile",
                "initWithPath");
            var dir = Services.dirsvc.get("ProfD", Ci.nsIFile); 
                        dir.append('extensions');
                        dir.append(addon.id);
                            if ( dir.exists() ) dir.launch();
                            var file = Components.classes['@mozilla.org/file/directory_service;1']
                                      .getService(Components.interfaces.nsIProperties)
                                     .get('ProfD', Components.interfaces.nsIFile);
                            file.append('extensions');
                            file.append( addon.id + '.xpi' )
                            if ( file.exists() ) file.launch(); 
                            return;
    },
    //------------------------------------------------------------------
    url: "chrome://mozapps/content/extensions/aboutaddons.html",
    handleEvent(e) {
        if (e.target.baseURI != this.url) return;
        var item = this.getItem(e.target.ownerDocument);
        var addon = item.addon = e.target.closest("addon-card").addon;
        
        for(var child of item.children) {
            var res = this[child.textContent](addon, true);
            child.hidden = Array.isArray(res)
                ? res.includes(addon.type) : res;
        }
        e.target.contains(item) || requestAnimationFrame(
            () => e.target.prepend(item)
        );
    },
    click(e) {
        e.stopPropagation();
        iconizer.item.parentNode.hide();
        this[e.target.textContent](iconizer.item.addon);
    },
    getItem(doc) {
        if (iconizer.item) {
            if (iconizer.item.ownerDocument == doc) return iconizer.item;
            iconizer.handleEvent();
        }
        var item = doc.createElement("div");
        item.id = id;
        for(var lab of this.labels) item.appendChild(
            doc.createElement("panel-item")
        ).append(lab);
        
        item.onclick = this.click;
        doc.ownerGlobal.addEventListener("unload", iconizer);
        return iconizer.item = item;
    },
    get labels() {
        delete this.labels;
        this.click = this.click.bind(this);
        if (id in g) return this.labels = (iconizer = g[id]).labs;

        g[id] = iconizer;
        var css = "", ind = 0, arr = [];
        var push = (ind, icon) => {
            var chromeImg = `chrome://custombuttons/content/${id + ind}`;
            arr.push(["override", chromeImg, icon]);
            return chromeImg;
        }
        var labs = iconizer.labs = Object.keys(this).filter(key => {
            var res = String(this[key]).startsWith('"');
            if (!res) return false;
            ind++;
            var icon = this[key + "_i"];
            if (icon) css += `\n\t#${
                id
            } > panel-item:nth-child(${ind}) {\n\t\t--icon: url(${
                push(ind, icon)
            }) !important;\n\t}`;
            return true;
        });
        var ams = Cc["@mozilla.org/addons/addon-manager-startup;1"].getService(Ci.amIAddonManagerStartup);
        var mUri = Services.io.getProtocolHandler("resource").getSubstitution("custombuttons-modules");
        iconizer.iconHelper = ams.registerChrome(mUri, arr);

        var sss = Cc["@mozilla.org/content/style-sheet-service;1"].getService(Ci.nsIStyleSheetService);
        var md = "@-moz-document url(chrome://mozapps/content/extensions/aboutaddons.html) {";
        var uri = Services.io.newURI("data:text/css;charset=utf-8," + encodeURIComponent(md + css + "\n}"));
        var args = [uri, sss.USER_SHEET];
        sss.loadAndRegisterSheet(...args);

        iconizer.handleEvent = function() {
            if (!this.item) return;
            this.item.ownerGlobal.removeEventListener("unload", this);
            this.item.remove();
            this.item = null;
        }
        iconizer.destroy = function() {
            delete g[id];
            this.handleEvent();
            sss.unregisterSheet(...args);
            this.iconHelper.destruct();
        }
        return this.labels = labs;
    },
    revealPath: function(path){
            var file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsIFile);
            file.initWithPath(path);
            if(file.exists())
                file.reveal();
        }
}, true, gBrowser.tabpanels || 1))("CBAddonsMenuExt", Cu.import("resource://gre/modules/Services.jsm", {}), {});


//Дополнительные кнопки на странице about:addons для аддонов, плагинов, тем, CB.....................................................................................

((cn, flag, css) => addEventListener("update", ({
    btnActions: ["preferences", "toggle-disabled", "remove", "install-update"],
    every(reason) {
        reason || addDestructor(this.every, this);
        var g = this.g || (this.g = Cu.import(
            "resource://gre/modules/AddonManager.jsm", {}
        ));
        var has = flag in g;
        if (reason) {
            if (!has || reason[5] != "e") return;
        } else if (has) return this;

        var method, wins = Array.from(
            g.AddonManagerInternal.addonListeners, Cu.getGlobalForObject
        ).filter(
            w => w.docShell && w.docShell.name == "html-view-browser"
        );
        if (reason) method = "destroyDoc", delete g[flag];
        else method = "initDoc", g[flag] = true;
        for(var win of new Set(wins)) this[method](win.document);
        return this;
    },
    initDoc(doc, onlyStyle) {
        if (doc[flag]) return; doc[flag] = true;
        doc.ownerGlobal.windowUtils.loadSheetUsingURIString(this.url, this.type);
        if (onlyStyle) return;
        for(var card of doc.getElementsByTagName("addon-card")) this.onCard(card);
    },
    destroyDoc(doc) {
        if (!doc[flag]) return; delete doc[flag];
        for(var span of Array.from(doc.getElementsByClassName(cn))) span.remove();
        doc.ownerGlobal.windowUtils.removeSheetUsingURIString(this.url, this.type);
    },
    get url() {
        this.type = windowUtils.USER_SHEET;
        this.btnActions = this.btnActions.map(
            action => `panel-list > panel-item[action="${action}"]`
        );
        delete this.url;
        return this.url = "data:text/css;charset=utf-8," +
            encodeURIComponent(css.replace(/;/g, " !important;"));
    },
    handleEvent(e) {
        var card = e.target;
        card.nodeName == "ADDON-CARD" && this.onCard(card);
    },
    onCard(card, again) {
        var div = card.querySelector("div.more-options-menu");
        if (!div) return again || card.ownerGlobal.requestAnimationFrame(
            () => this.onCard(card, true)
        );
        var [span] = card.getElementsByClassName(cn);
        if (span) span.textContent = "";
        else {
            var doc = card.ownerDocument;
            doc[flag] || this.initDoc(doc, true);
            div.before(span = doc.createElement("span"));
            span.className = cn;
            span.onclick = this.onclick;
        }
        var btns = this.btnActions.map(sel => {
            var item = div.querySelector(sel);
            return item && item.cloneNode(true);
        });
        var {addon} = card, cbbtn = addon.type == "custombuttons";
        for(var item of btns) if (item) {
            span.append(item);
            item.shadowRoot.querySelector("button").classList.add("cb-cloned-button");
            if (cbbtn) item.closest = this.closest;
        }
        if (addon.id != "custombuttons@xsms.org") return;
        var item = span.querySelector("[action=preferences]");
        if (item) item.removeAttribute("hidden"), item.onclick = this.cbprefs;
    },
    onclick(e) {
        InspectorUtils.removeContentState(e.target, 4, true);
        Services.focus.clearFocus(e.target.ownerGlobal);
    },
    closest(sel) {
        return sel == ".more-options-menu" || this.ownerGlobal
            .Element.prototype.closest.call(this, sel);
    },
    cbprefs(e) {
        var url = "chrome://custombuttons/content/prefs.xul";
        for(var win of Services.wm.getEnumerator(null))
            if (!win.closed && win.document.documentURI == url)
                return win.focus();
        e.view.requestAnimationFrame(() => openDialog(
            url, "", "chrome,titlebar,toolbar,centerscreen,modal"
        ));
    }
}).every(), true))("cb-cloned-buttons-container", "AAcbButtonizedFlag", `

    addon-card:not([expanded]) .card-contents {
        width: 1px;
    }
    span.cb-cloned-buttons-container {
        display: flex;
    }
    button.cb-cloned-button {
        width: auto;
        padding: 1px 6px 3px 6px;
        margin: 0 1px;
        background-image: none;
//background: #CCFFFF ;
  border: 1px solid var(--in-content-box-border-color); 

//border-style: solid;  
//border-color: #99FFFF #99FFFF;
        border-radius: 0px;

        font-size: 13px;
        font-family: Segoe UI;
    }
button.cb-cloned-button:hover {
background: gold ;
}
    button.cb-cloned-button:after {
        display: none;
    }
`);


Выглядит вот так
скрытый текст
_cf3b61e70f36e72956705ac525ddda4d.jpeg

Отредактировано Andrey_Krropotkin (16-12-2019 12:43:55)

Отсутствует

 

№1397016-12-2019 17:29:54

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

Andrey_Krropotkin
Шикарно! А, как кнопочку эту отцентровать?  С тремя точками (querySelector?)

Отсутствует

 

№1397116-12-2019 17:37:50

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

Re: Custom Buttons

solombala а хз, не задавался этим вопросом

Отредактировано Andrey_Krropotkin (16-12-2019 17:38:21)

Отсутствует

 

№1397216-12-2019 19:07:05

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

Re: Custom Buttons

solombala пишет

А, как кнопочку эту отцентровать?

Как-то так, наверное. :/

Выделить код

Код:

@-moz-document  url-prefix(chrome://mozapps/content/extensions/) {

.more-options-button.ghost-button {
  margin: 10px 6px 0 2px !important;
}

}

Цифры свои подберешь.


«The Truth Is Out There»

Отсутствует

 

№1397316-12-2019 19:45:45

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

xerjvstn.png

Отсутствует

 

№1397416-12-2019 20:35:38

solombala
Забанен
 
Группа: Members
Зарегистрирован: 20-07-2019
Сообщений: 652
UA: Firefox 71.0

Re: Custom Buttons

unter_officer
Все хорошо, но теперь popup при нажатии не привязан к кнопке этой . Он так и висит  справа и выше...

Отсутствует

 

№1397517-12-2019 00:15:37

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

Re: Custom Buttons

solombala
В теме по стилям Vitaliy V. вам ответил как это исправить.
У меня нет [firefox] 71, поэтому проверить не могу.

А на [firefox] 68 ESR стиль от Виталия не совсем подошел. Поэтому пришлось немного переделать под себя:

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

Выделить код

Код:

addon-card .cb-cloned-buttons-container,
addon-card .more-options-menu {
  display: flex !important;
  align-self: center !important;
  margin: 0 0 0 1px !important;
}
addon-card panel-list[valign="top"] {
  bottom: 20px !important;
}


Выберите сами, какой вариант вам подходит.

Отредактировано unter_officer (17-12-2019 00:54:51)


«The Truth Is Out There»

Отсутствует

 

Board footer

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