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

В мире Mozilla происходит много интересных событий. Но вам не нужно постоянно посещать новостные сайты, чтобы быть в курсе всех изменений. Зайдите на ленту новостей Mozilla Россия.

№1620107-01-2022 21:32:21

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

Re: Custom Buttons

Dumby
Переделайте пожалуйста кнопочку для UCF.

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

Выделить код

Код:

/*Initialization Code*/

// Simple Session Manager (https://forum.mozilla-russia.org/viewtopic.php?pid=744023#p744023) ..........

// Подсказки для кнопки .....
this.tooltipText = "Simple Session Manager";


// Настройка функций кликов мыши .....
this.onmousedown =e=> {

	this.onmouseup =e=> {
		if ( e.button ) return;
		self._handleClick =()=> menupopup.openPopup(this, "after_start");
	}
	if ( e.button == 2 ) {
		gShowPopup(this);
	}

}
self.onclick =e=> e.preventDefault();


var menupopup = self.appendChild(document.createXULElement("menupopup"));
menupopup.id = "ssm_menupopup";

var scs = document.createXULElement("menuitem");
scs.setAttribute("label", "Сохранить сессию");
scs.setAttribute("class", "menuitem-iconic");
scs.setAttribute("image", "");
scs.addEventListener("command", saveCurrentSession, false);
menupopup.appendChild(scs);

var menusep = document.createXULElement("menuseparator"); // Сепаратор .....
menupopup.appendChild(menusep);


var savedSessions = loadFile(); // Сохраненный список .....
for (name in savedSessions) {
	makeitems(name);
}


// overwrite = 1 - Открыть сессию в текущем окне (все открытые вкладки будут закрыты) .....
// overwrite = 0 - Добавить вкладки в текущее окно (сессия будет добавлена к уже открытым вкладкам) .....
var overwrite = 1,
Cc = Components.classes,
Ci = Components.interfaces,
Cu = Components.utils,
SS = "nsISessionStore" in Components.interfaces ? ( Components.classes["@mozilla.org/browser/sessionstore;1"] || Components.classes["@mozilla.org/suite/sessionstore;1"] )
		.getService(Components.interfaces.nsISessionStore) : SessionStore;


if (!window.Services) { Cu.import("resource://gre/modules/Services.jsm"); }


// Функции работы с файлами .....
function saveFile(data) {
	var file = Services.dirsvc.get('UChrm', Ci.nsIFile);
	file.append("simple_session_manager.json");

	var suConverter = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter);
	suConverter.charset = 'UTF-8';
	data = suConverter.ConvertFromUnicode(data);

	var foStream = Cc['@mozilla.org/network/file-output-stream;1'].createInstance(Ci.nsIFileOutputStream);
	foStream.init(file, 0x02 | 0x08 | 0x20, 0664, 0);
	foStream.write(data, data.length);
	foStream.close();
}

function loadFile() {
	var file = Services.dirsvc.get('UChrm', Ci.nsIFile);
	file.append("simple_session_manager.json");
	if (file.exists() === false) return false;
	var fstream = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(Ci.nsIFileInputStream);
	var sstream = Cc["@mozilla.org/scriptableinputstream;1"].createInstance(Ci.nsIScriptableInputStream);
	fstream.init(file, -1, 0, 0);
	sstream.init(fstream);

	var data = sstream.read(sstream.available());
	try { data = decodeURIComponent(escape(data)); } catch(e) {}
	sstream.close();
	fstream.close();
	if (data === "undefined") return false;
	data = JSON.parse(data);
	return data;
}


// Получить текущее время .....
function getTime() {
var d = new Date();
	function addzero(t) {
		(t < 10) ? t = '0' + t : t;
		return t;
	}
	var t = addzero(d.getFullYear()) + '.' + addzero(d.getMonth()+1) + '.' + addzero(d.getDate()) + ' - ' + addzero(d.getHours()) + ':' + addzero(d.getMinutes()) + ':' + addzero(d.getSeconds());
	return t;
}


// Получить название вкладки .....
function getTabLabel() {
	var label = gBrowser.selectedTab.label;
	return label.substring(0, 70);
}


// Сохранение сессий .....
function saveSession(ssdata) {
	var countTabs_getTime = (gBrowser.tabs.length + "(B) " + "[" + getTime() + "]" );
	var saveName = prompt("Сохранить", getTabLabel() );
	var name = (saveName + ", " + countTabs_getTime );
	if (name != null) {
		if (loadFile() === false) { var data = {}; }
		else { var data = loadFile(); }
		if (data[name]) { alert('Сессия с тем же именем уже существует!'); return; }
		data[name] = JSON.parse(ssdata);
		saveFile(JSON.stringify(data));
		makeitems(name);
	}
}


// Сохранить текущую сессию .....
function saveCurrentSession() {
	var ssdata = SS.getBrowserState();
	saveSession(ssdata);
}


// Удалить сессию .....
function remove() {
	var node = this.parentNode.parentNode;
	var name = node.getAttribute("label");
	var cf = confirm('Вы уверены, что хотите удалить ' + name + ' ?');
	if (cf === true) {
		node.style.display = "none";
		var data = loadFile();
		delete data[name];
		saveFile(JSON.stringify(data));
	}
}


// Переименовать сессию .....
function rename() {
	var node = this.parentNode.parentNode;
	var name = node.getAttribute("label");
	var newname = prompt('Переименовать ' + '"' + name + '"' + ' в:', 'введите новое имя');
	if (!newname) return;
	this.parentNode.parentNode.setAttribute("label", newname);
	var data = loadFile();
	var value = data[name];
	data[newname] = value;
	delete data[name];
	saveFile(JSON.stringify(data));
}


// Восстановить сессию .....
function restoreSession(stateString) {
	if (typeof stateString === "string") {
		var state = stateString;
	}
	else {
		var name = this.parentNode.parentNode.getAttribute("label");
		var data = loadFile();
		var state = JSON.stringify(data[name]);
	}
	switch (overwrite) {
		case 0:
		SS.setWindowState(window, state, false);
		break;

		case 1:
		SS.setBrowserState(state);
		break;
	}
}


// Создаем меню .....
function makeitems(name) {
	var ss = document.createXULElement("menu");
	ss.setAttribute("label", name);
	ss.setAttribute("class", "savedSessions");
	ss.className = "menu-iconic";
	ss.setAttribute("image", "");

	var ss_popup = document.createXULElement("menupopup");
	var rs = document.createXULElement("menuitem");
	rs.setAttribute("label", "Восстановить");
	rs.setAttribute("class", "menuitem-iconic");
	rs.setAttribute("image", "");
	rs.addEventListener("command", restoreSession, false);

	var rn = document.createXULElement("menuitem");
	rn.setAttribute("label", "Переименовать");
	rn.setAttribute("class", "menuitem-iconic");
	rn.setAttribute("image", "");
	rn.addEventListener("command", rename, false);

	var rm = document.createXULElement("menuitem");
	rm.setAttribute("label", "Удалить");
	rm.setAttribute("class", "menuitem-iconic");
	rm.setAttribute("image", "");
	rm.addEventListener("command", remove, false);

	ss_popup.appendChild(rs);
	ss_popup.appendChild(rn);
	ss_popup.appendChild(rm);
	ss.appendChild(ss_popup);
	menupopup.appendChild(ss);
}


// Восстановление выбранной сессии при открытии браузера .....
// Выбор сессии — двойной ЛКМ по соответствующему пункту .....
// Dumby: https://forum.mozilla-russia.org/viewtopic.php?pid=782655#p782655 .....
((g, id, pref, {obs, prefs, ww, dirsvc}, style) => {
	var popup = scs.parentNode;
	addEventListener("dblclick", g[id] || (g[id] = {
		get name() {
			return prefs.getStringPref(pref, null);
		},
		init() {
			obs.addObserver(this, "quit-application", false);
			prefs.addObserver(pref, this.upd = () => {
				var {name} = this;
				for(var win of ww.getWindowEnumerator("navigator:browser")) 
					win.toolbar.visible && this.updPopup(win, name);
				this.oldName = null;
			});
			var st = new Image().style;
			st.cssText = style;
			this.style = st.cssText;
			this.handleMuts = this.handleMuts.bind(this);
			return this;
		},
		destroy(reason) {
			delete g[id];
			obs.removeObserver(this, "quit-application");
			prefs.removeObserver(pref, this.upd);
			reason == "delete" && prefs.clearUserPref(pref);
		},
		observe(s, t, data) {
			this.destroy();
			if (data.includes("restart")) return;
			var {name} = this;
			if (name == null) return;
			var file = dirsvc.get("UChrm", Ci.nsIFile);
			file.append("simple_session_manager.json");
			var state;
			try {state = JSON.parse(Cu.readUTF8File(file))[name];} catch {}
			if (!state) return prefs.clearUserPref(pref);

			g.SessionStoreInternal.getCurrentState = () => state;
			prefs.setBoolPref("browser.sessionstore.resume_session_once", true);
		},
		handleEvent(e) {
			if (!e.button && e.target.nodeName == "menu")
				e.target.label == this.name
					? prefs.clearUserPref(pref)
					: prefs.setStringPref(pref, e.target.label);
		},
		oldName: null,
		updPopup(win, name = this.name) {
			var {style} = this, popup = win.document.getElementById("ssm_menupopup");
			if (popup) for(var menu of popup.getElementsByTagName("menu"))
				if (
					this.oldName != null && menu.label == this.oldName
					&& !void(menu.label = name) || menu.label == name
				)
					menu.style.cssText += style;
				else {
					var css = menu.style.cssText;
					if (css == style) menu.removeAttribute("style");
					else if (css.includes(style))
						menu.style.cssText = css.replace(style, "");
				}
		},
		opts: {
			attributes: true, attributeOldValue: true,
			attributeFilter: ["style", "label"], subtree: true
		},
		handleMuts(muts) {
			if(!muts[0].target.matches(":-moz-window-inactive")) for(var mut of muts)
				if (mut.attributeName == "label" && mut.oldValue == this.name)
					this.oldName = this.name,
					prefs.setStringPref(pref, mut.target.label);
				else if (mut.attributeName == "style" && mut.oldValue == this.style) {
					var css = mut.target.style.cssText;
					css && css != this.style && prefs.clearUserPref(pref);
				}
		}
	}).init(), false, popup || 1);
	g[id].updPopup(window);
	var mo = new MutationObserver(g[id].handleMuts);
	mo.observe(popup, g[id].opts);
	addDestructor(reason => mo.disconnect(reason[5] == "e" && g[id]?.destroy(reason)));
})(
	Cu.import("resource:///modules/sessionstore/SessionStore.jsm", {}),
	"CBSSMQuitApplicationObserver", "CB.SSM.sessionToRestore",  Services,
	"font-weight: bold !important; color: #AA0000 !important;"
);

Для FF91.


«The Truth Is Out There»

Отсутствует

 

№1620208-01-2022 01:32:36

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

Re: Custom Buttons

Dumby пишет

Шутить изволишь?

Нет, не нашёл простых примеров такого кода. Ещё просьба добавить универсальности:
чтобы проверка наличия файла работала с разными путями — файловой системы `C:\windows\explorer.exe` и `chrome://user_chrome_files/content/user_chrome.js`
но в Линукс имена могут содержать практически любые символы, поэтому String.raw может привести к проблемам


Прошу улучшить скрипт ucf_BookmarkDir (FF 90+), чтобы он работал хотя бы с версии Firefox 84.

Выделить код

Код:

(async (id, sel) => { // расположение закладки в Избранном, Недавняя папка (F90+)
	var g = Cu.getGlobalForObject(Cu), stt = g[id];
	if (!stt) {
		var {obs, prefs} = Services, pu = PlacesUtils, {bookmarks: bm, observers: pobs} = pu, stt = g[id] = {
			bm, // клики заданы в ucf_hookClicks.js
			help_star: "Правый клик	➜ Быстрая закладка\nЛев.клик+Alt	Библиотека закладок\n\nКлик дважды	Перевод сайта/выд.текста\n",
			pref: `ucf.${id}Guid`,

			async init() {
				var args = [
					["bookmark-added", "bookmark-moved"],
					events => {for(var e of events) e.isTagging || this[e.constructor.name](e);}
				];
				pobs.addListener(...args);
				pu.registerShutdownFunction(() => {
					pobs.removeListener(...args);
					prefs.setStringPref(this.pref, this.lastGuid);
				});
				this.args = [
					res => this.fetchRes.push(res),
					{concurrent: true, includePath: true}
				];
				var guid = prefs.getStringPref(this.pref, "");
				if (!guid) try {var [guid] = await PlacesUtils.metadata.get(
					PlacesUIUtils.LAST_USED_FOLDERS_META_KEY, [])} catch {}
				this.lastGuid = guid || await PlacesUIUtils.defaultParentGuid || bm.unfiledGuid;
			},
			PlacesBookmarkAddition(e) {
				if (e.itemType == bm.TYPE_BOOKMARK && e.source == bm.SOURCES.DEFAULT)
					this.lastGuid = e.parentGuid;
			},
			PlacesBookmarkMoved(e) {
				e.parentGuid != e.oldParentGuid && this.PlacesBookmarkAddition(e);
			},
			find: obj => obj.name == "tooltiptext",
			tt(win) {
				var list = win.InspectorUtils.getChildrenForNode(win.document.documentElement, true);
				return list.item(list.length - 1);
			},
			mapInfs(inf) {
				return inf.path.map(this.mapPaths).join("\\");
			},
			mapPaths: path => bm.getLocalizedTitle(path) || "[Безымянная папка]",
		};
		stt.init();

		var func = id => this[id].handleEvent = async function(e) {
			var win = e.view, star = e.target;
			var starred = win.BookmarkingUI._itemGuids.size;
			var arg = starred ? {url: win.gBrowser.currentURI.spec} : this.lastGuid;

			var arr = this.fetchRes = [];
			await this.bm.fetch(arg, ...this.args);
			if (!star.matches(":hover")) return;

			!starred && arr.length && arr[0].path.push(arr[0]);
			var paths = arr.length ? arr.map(this.mapInfs, this).join("\n") : "<Folder Not Found>";

			if (!star.matches(":hover")) return;
			var footer = `Колёсико	масштаб ${Math.round(win.ZoomManager.zoom*100)}% Текст | Всё\n\n★ ` + (
				starred
					? (arr.length > 1 ? "Данные закладки добавлены" : "Данная закладка добавлена") + " в"
					: "Недавно добавлялось в папку"
			) + ":\n" + paths;

			var header = (await win.document.l10n.formatMessages([{ // стандартная подсказка
				id: star.getAttribute("data-l10n-id"),
				args: JSON.parse(star.getAttribute("data-l10n-args"))
			}]))[0].attributes.find(this.find).value;

			var text = header + '\n' + this.help_star + footer;
			var tt = star.linkedTooltip;
			star.contains(tt.triggerNode) ? tt.label = text : star.tooltipText = text;
		}
		var url = "data:;charset=utf-8," + encodeURIComponent(`(${func})("${id}")`);
		g.ChromeUtils.compileScript(url).then(ps => ps.executeInGlobal(g));
	}
	await delayedStartupPromise;

	var stars = Array.from(document.querySelectorAll(sel));
	for(var star of stars) {
		star.linkedTooltip = stt.tt(window);
		star.addEventListener("mouseenter", stt);
	}
	var destructor = () => {
		for(var star of stars)
			star.removeEventListener("mouseenter", stt);
	}
	var ucf = window.ucf_custom_script_win || window.ucf_custom_script_all_win;
	if (ucf)
		ucf[id] = {destructor}, ucf.unloadlisteners.push(id);
	else
		window.addEventListener("unload", destructor, {once: true});
})("ucfBookmarksStarFTooltipHelper", "#star-button, #star-button-box, #context-bookmarkpage");

Ещё вопрос - как создать ассоциативный массив и получить элемент массива по его имени: array.get("имя ключа")

Отредактировано Dobrov (09-01-2022 07:34:26)

Отсутствует

 

№1620311-01-2022 22:04:58

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

Re: Custom Buttons

Нет вопросов к 96...Норм.

Отсутствует

 

№1620413-01-2022 21:03:37

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

Re: Custom Buttons

unter_officer пишет

Переделайте пожалуйста кнопочку для UCF.

Вроде как-то давно пробовал возиться с ней, ещё под CB.
Всё что-то не так. Ладно, вот что есть.

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

Выделить код

Код:

(async pid => CustomizableUI.createWidget(({
	id: "797321",
	label: "Simple Session Manager",
	tooltiptext: "Simple Session Manager",
	localized: false,
	init() {
		this.handleEvent = e => this[e.type](e);
		this.onTimeout = () => 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;
	},
	onCreated(btn) {
		btn.type = "menu";
		btn.phTimestamp = 0;
		btn.render = this.render;
		btn._handleClick = this.click;
		btn.setAttribute("image", "resource://usercontext-content/cart.svg");

		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);
			return e.preventDefault();
		}
		e.detail == 2 && trg.nodeName == "menu" && this.boot(trg);
	},
	boot(trg) {
		var popup = trg.parentNode;
		var old = popup.querySelector("[boot]");
		if (old != trg) old?.removeAttribute("boot");
		trg.toggleAttribute("boot");
		popup.dblMD = true;
	},
	click() {
		var win = this.ownerGlobal;
		if (win.event.target != this) return;
		win.clearTimeout(this.tid);
		if (this.mdTimestamp - this.phTimestamp > 50) this.open = true;
	},
	command(e) {
		var arg, trg = e.target, cmd = trg.value;
		if (cmd.startsWith("r")) {
			arg = trg.parentNode.parentNode.label;
			if (cmd.startsWith("res"))
				return this[cmd](arg, (e.button == 1 || e.ctrlKey) && e.view);
		}
		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;
		}
		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();
		if (!this.dragData && !popup.dblMD) return;

		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.dblMD) {
			delete popup.dblMD;
			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) 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);
	},
	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);
		},
		ArrowDown(menu) {
			var ns = menu.nextSibling;
			if (ns) ns.after(menu), this.arrow(menu);
		},
		ArrowUp(menu) {
			var ps = menu.previousSibling;
			if (ps.nodeName == "menu") ps.before(menu), this.arrow(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="restoreSession"/>
			<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("");
				}
				#${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("data:image/png;charset=utf-8;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABmJLR0QA/wD/AP+gvaeTAAABt0lEQVQ4ja2RT2sTURTFz8tk0nntFAQrVLBQxIUg/tkkMoLyIEVECHZhNrrxI+hOP5Mb6eBCcEipYIjZ2LpvhFasUtBSkwzOve+6mE6YSScuxLt67753fudyLvA/a7T+qCHmiTfrfWCMd3j9diPfq2SHX+uPnzJx71gP33w3bX9avBcE2j+SkK30Di7ffJ71VebMxD1hC2GGEEfH41prpftynIm92A0t2aYQQYgBThrndz/2KwCgf9Z2LNtImFOAtc356ijcCwJdJrZEUWzPfJpMAAAStPWR/h2Ktc10CoIQRUKEabFNvNbKfndcAKSjtrVfHW5YorWTzxDKYAxLtOWM+P7yt51hIYPpsDTNTyB/EwNAtWxdMuV8GCfbgxjxIviUYSV/SQOrbVjitbx4N8alBeXcdZR+3Tl3xS8FlIkt0dbnWH3xlbPgKgUo3HEq+tX7C4EuAAbGeOmqCuLIJt69s3PeQ1epKGfapCQJO6vGmwAWf/C1Wau6td8dV123BaAAUfHw6gSwtP3ugxA/y6X9INszAOQgbwFAIC/MQb9/Kv2vF2/UByejlVVn1Xiby/X6rPd/qj/1ak71UYKuwQAAAABJRU5ErkJggg==");
				}
				#${pid} > menuseparator:last-child,
				#${pid} > menu[maxwidth] > .menu-right {
					display: none;
				}
				#${pid} > menu[boot] {
					color: red;
					font-weight: bold;
				}
				#${pid} > menu[maxwidth] {
					color: blue;
					font-weight: bold;
					outline-offset: -2px;
					outline: 2px solid orangered;
				}
			}
		`.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", {});
	},
	getState() {
		return JSON.parse(this.gs.SessionStore.getBrowserState());
	},
	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() {
		(this.dataFile = Services.dirsvc.get("UChrm", Ci.nsIFile))
			.append("simple_session_manager.json");
		
		try {this.data = JSON.parse(Cu.readUTF8File(this.dataFile));}
		catch {this.data = {};}

		var mp = "{07cae4f5-18b0-487b-80eb-973304af9528}";
		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;
	},
	async save(excWin) {
		var {IOUtils} = Cu.getGlobalForObject(Cu);
		await IOUtils.makeDirectory(this.dataFile.parent.path);
		var {path} = this.dataFile;
		delete this.dataFile;
		(this.save = excWin => {
			IOUtils.writeJSON(path, this.data);
			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]];
		this.gs.SessionStoreInternal.getCurrentState = () => state;
		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;
	},
	getName(state) {
		var wl = state.windows.length, tl = 0;
		for(var w of state.windows) tl += w.tabs.length;
		return `${
			this.bwt.getTopWindow().gBrowser.selectedTab.label.slice(0, 70)
		} ${wl}/${tl} [${
			new Date().toLocaleString("mn").replace(" ", "-")
		}]`;
	},
	exists(name) {
		this.meta;
		return (this.exists = name => name in this.data &&
			!this.prompter.alert("Сессия с тем же именем уже существует!"))(name);
	},
	saveSession(state = this.getState(), name = this.getName(state)) {
		var name = this.prompter.prompt("Сохранить:", name);
		if (name == null) return true;

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

		this.data[name] = state;

		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");

Dobrov пишет

чтобы проверка наличия файла работала с разными путями — файловой системы `C:\windows\explorer.exe` и `chrome://user_chrome_files/content/user_chrome.js`

Ну, например, так.
Можно, наверно, запросом, как fetch или xhr, но это асинхрон.
Не, xhr можно и синхронный, но он что-то неуместное в консоль выдаёт,
к тому же можно и асинхронный, и не отправлять (уже будет xhr.channel.URI).

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

Выделить код

Код:

var path = "chrome://user_chrome_files/content/user_chrome.js";

if (path.startsWith("chrome://")) alert(

	Cc["@mozilla.org/chrome/chrome-registry;1"]
		.getService(Ci.nsIXULChromeRegistry)
		.convertChromeURL(Services.io.newURI(path))
		.QueryInterface(Ci.nsIFileURL).file.exists()
	/*
	Services.io.newChannelFromURIWithLoadInfo(
		Services.io.newURI(path),
		docShell.currentDocumentChannel.loadInfo
	).URI.QueryInterface(Ci.nsIFileURL).file.exists()
	*/
);

String.raw может привести к проблемам

Ну так не используй, раз «может».

Прошу улучшить скрипт ucf_BookmarkDir (FF 90+), чтобы он работал хотя бы с версии Firefox 84.

Вот уж нет. Рыться в этом, когда уже сделано как если бы это,
и плюшка вписана, уволь.

как создать ассоциативный массив и получить элемент массива по его имени: array.get("имя ключа")

Объект иногда называют «ассоциативным массивом»,
а если, зачем-то, нужен метод get(), то можно добавить.
Ну и Map подходит под описание, разумеется.

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

Выделить код

Код:

var something = {
	get(key) {
		return this[key];
	},
	"имя ключа": 2 * 22,
};
alert(
	something.get("имя ключа") // 44
);

//-------------------------------------

//var map = new Map([["имя ключа", 7 * 111]]);

var map = new Map();
map.set("имя ключа", 7 * 111);

alert(
	map.get("имя ключа") // 777
);

Отсутствует

 

№1620514-01-2022 03:02:57

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

Re: Custom Buttons

Dumby пишет

Вроде как-то давно пробовал возиться с ней, ещё под CB.
Всё что-то не так. Ладно, вот что есть.

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

Выделить код

Код:

(async pid => CustomizableUI.createWidget(({
	id: "797321",
	label: "Simple Session Manager",
	tooltiptext: "Simple Session Manager",
	localized: false,
	init() {
		this.handleEvent = e => this[e.type](e);
		this.onTimeout = () => 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;
	},
	onCreated(btn) {
		btn.type = "menu";
		btn.phTimestamp = 0;
		btn.render = this.render;
		btn._handleClick = this.click;
		btn.setAttribute("image", "resource://usercontext-content/cart.svg");

		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);
			return e.preventDefault();
		}
		e.detail == 2 && trg.nodeName == "menu" && this.boot(trg);
	},
	boot(trg) {
		var popup = trg.parentNode;
		var old = popup.querySelector("[boot]");
		if (old != trg) old?.removeAttribute("boot");
		trg.toggleAttribute("boot");
		popup.dblMD = true;
	},
	click() {
		var win = this.ownerGlobal;
		if (win.event.target != this) return;
		win.clearTimeout(this.tid);
		if (this.mdTimestamp - this.phTimestamp > 50) this.open = true;
	},
	command(e) {
		var arg, trg = e.target, cmd = trg.value;
		if (cmd.startsWith("r")) {
			arg = trg.parentNode.parentNode.label;
			if (cmd.startsWith("res"))
				return this[cmd](arg, (e.button == 1 || e.ctrlKey) && e.view);
		}
		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;
		}
		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();
		if (!this.dragData && !popup.dblMD) return;

		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.dblMD) {
			delete popup.dblMD;
			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) 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);
	},
	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);
		},
		ArrowDown(menu) {
			var ns = menu.nextSibling;
			if (ns) ns.after(menu), this.arrow(menu);
		},
		ArrowUp(menu) {
			var ps = menu.previousSibling;
			if (ps.nodeName == "menu") ps.before(menu), this.arrow(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="restoreSession"/>
			<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("");
				}
				#${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("data:image/png;charset=utf-8;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABmJLR0QA/wD/AP+gvaeTAAABt0lEQVQ4ja2RT2sTURTFz8tk0nntFAQrVLBQxIUg/tkkMoLyIEVECHZhNrrxI+hOP5Mb6eBCcEipYIjZ2LpvhFasUtBSkwzOve+6mE6YSScuxLt67753fudyLvA/a7T+qCHmiTfrfWCMd3j9diPfq2SHX+uPnzJx71gP33w3bX9avBcE2j+SkK30Di7ffJ71VebMxD1hC2GGEEfH41prpftynIm92A0t2aYQQYgBThrndz/2KwCgf9Z2LNtImFOAtc356ijcCwJdJrZEUWzPfJpMAAAStPWR/h2Ktc10CoIQRUKEabFNvNbKfndcAKSjtrVfHW5YorWTzxDKYAxLtOWM+P7yt51hIYPpsDTNTyB/EwNAtWxdMuV8GCfbgxjxIviUYSV/SQOrbVjitbx4N8alBeXcdZR+3Tl3xS8FlIkt0dbnWH3xlbPgKgUo3HEq+tX7C4EuAAbGeOmqCuLIJt69s3PeQ1epKGfapCQJO6vGmwAWf/C1Wau6td8dV123BaAAUfHw6gSwtP3ugxA/y6X9INszAOQgbwFAIC/MQb9/Kv2vF2/UByejlVVn1Xiby/X6rPd/qj/1ak71UYKuwQAAAABJRU5ErkJggg==");
				}
				#${pid} > menuseparator:last-child,
				#${pid} > menu[maxwidth] > .menu-right {
					display: none;
				}
				#${pid} > menu[boot] {
					color: red;
					font-weight: bold;
				}
				#${pid} > menu[maxwidth] {
					color: blue;
					font-weight: bold;
					outline-offset: -2px;
					outline: 2px solid orangered;
				}
			}
		`.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", {});
	},
	getState() {
		return JSON.parse(this.gs.SessionStore.getBrowserState());
	},
	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() {
		(this.dataFile = Services.dirsvc.get("UChrm", Ci.nsIFile))
			.append("simple_session_manager.json");
		
		try {this.data = JSON.parse(Cu.readUTF8File(this.dataFile));}
		catch {this.data = {};}

		var mp = "{07cae4f5-18b0-487b-80eb-973304af9528}";
		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;
	},
	async save(excWin) {
		var {IOUtils} = Cu.getGlobalForObject(Cu);
		await IOUtils.makeDirectory(this.dataFile.parent.path);
		var {path} = this.dataFile;
		delete this.dataFile;
		(this.save = excWin => {
			IOUtils.writeJSON(path, this.data);
			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]];
		this.gs.SessionStoreInternal.getCurrentState = () => state;
		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;
	},
	getName(state) {
		var wl = state.windows.length, tl = 0;
		for(var w of state.windows) tl += w.tabs.length;
		return `${
			this.bwt.getTopWindow().gBrowser.selectedTab.label.slice(0, 70)
		} ${wl}/${tl} [${
			new Date().toLocaleString("mn").replace(" ", "-")
		}]`;
	},
	exists(name) {
		this.meta;
		return (this.exists = name => name in this.data &&
			!this.prompter.alert("Сессия с тем же именем уже существует!"))(name);
	},
	saveSession(state = this.getState(), name = this.getName(state)) {
		var name = this.prompter.prompt("Сохранить:", name);
		if (name == null) return true;

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

		this.data[name] = state;

		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");

Dumby, огромное спасибо.


«The Truth Is Out There»

Отсутствует

 

№1620614-01-2022 22:56:52

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

Re: Custom Buttons

Dumby

Dumby пишет

Simple Session Manager

Для последних FF подходит?  А, то чуть лочистил куки и историю и все...Ни черта не восстанавливает....(93-96)

Отсутствует

 

№1620715-01-2022 00:43:48

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

Re: Custom Buttons

ВВП пишет

Для последних FF подходит?

Ну, я поставил на 98.0a1, вроде работает.
Но всё, конечно, не тестировал, так, потыкал чуть-чуть.

Отсутствует

 

№1620815-01-2022 01:06:15

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

Re: Custom Buttons

ВВП пишет

Dumby

Dumby пишет

Simple Session Manager

Для последних FF подходит?  А, то чуть лочистил куки и историю и все...Ни черта не восстанавливает....(93-96)

Эта кнопка всегда была с придурью.
Например, если в настройках [firefox] поставить "При закрытии Firefox должен автоматически удалять: Журнал посещений и загрузок", то сессию можно буде восстановить только вручную. Автоматическое восстановление не сработает.
Есть и другие косяки.
Я держу эту кнопку только из-за того, что она при восстановлении сессии также восстанавливает положение прокрутки страниц. А для меня это важно.
Все современные дополнения этого делать не умеют. Иначе я бы уже давно поставил что-нибудь типа "Tab Session Manager" и не мучил Dumby правкой этой кнопки.


«The Truth Is Out There»

Отсутствует

 

№1620915-01-2022 09:33:13

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

Re: Custom Buttons

unter_officer пишет

Например, если в настройках [firefox] поставить "При закрытии Firefox должен автоматически удалять: Журнал посещений и загрузок", то сессию можно буде восстановить только вручную. Автоматическое восстановление не сработает.

О! Я вижу это. Вот такая правка вроде помогает

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

Выделить код

Код:

…
		//this.gs.SessionStoreInternal.getCurrentState = () => state;
		var ssi = this.gs.SessionStoreInternal;
		ssi.getCurrentState = () => state;
		Services.obs.removeObserver(ssi, "browser:purge-session-history");

Есть и другие косяки.

Заинтригован, вдруг там что-то простое.

Отсутствует

 

№1621015-01-2022 09:56:55

shadow_user
Участник
 
Группа: Members
Зарегистрирован: 14-02-2007
Сообщений: 244
UA: Firefox 91.0

Re: Custom Buttons

Dumby
Пожалуйста, поправьте код кнопки для вставки символов https://forum.mozilla-russia.org/viewto … 86#p776486 , чтобы в темной теме браузера символы были читаемыми. Сейчас выглядит так, слева в светлой теме, справа в темной. Спасибо!
char.png

Отсутствует

 

№1621115-01-2022 12:46:27

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

Re: Custom Buttons

shadow_user пишет

чтобы в темной теме браузера символы были читаемыми

Можно чёрный color в css вписать и будут читаемыми, типа так

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

Выделить код

Код:

/**/
        #${_id} menuitem:not(:hover) > .menu-text {
            color: black;
        }


А можно закомментировать/удалить этот кусок, он был добавлен,
чтобы закрасить вертикальный отделитель места для иконок в меню,
который здесь неуместен, но в Win10, как я вижу, такового просто нет.
Тогда в темной теме менюшка будет тёмной.
скрытый текст

Выделить код

Код:

/*
        #${_id} menugroup,
        #${_id} > menupopup > arrowscrollbox {
            background-color: menu;
        }
*/

Отсутствует

 

№1621215-01-2022 13:19:33

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

Re: Custom Buttons

Dumby

ВВП пишет

Simple Session Manager

Можно кнопкой это и если есть косячки, то и исправить ? Буксует на ютубе из-за авторизации.

Отсутствует

 

№1621315-01-2022 13:37:06

shadow_user
Участник
 
Группа: Members
Зарегистрирован: 14-02-2007
Сообщений: 244
UA: Firefox 91.0

Re: Custom Buttons

Dumby пишет

Можно чёрный color в css вписать и будут читаемыми, типа так

Dumby пишет

А можно закомментировать/удалить этот кусок, он был добавлен,

Просто супер, спасибо! :beer:

Отсутствует

 

№1621415-01-2022 17:09:47

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

Re: Custom Buttons

Dumby пишет
unter_officer пишет

Есть и другие косяки.

Заинтригован, вдруг там что-то простое.

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


Например, есть очень большая статья, в которой меня заинтересовали несколько абзацев, но в данный момент времени нет времени с этим разбираться.
В FF52 я пользовался менеджером сессий, который встроен в дополнение Tab Mix Plus. Я делал так.


Просматриваю статью. Останавливаюсь на нужном мне абзаце. Клонирую (дублирую) вкладку.
Перехожу в клонированную вкладку и прокручиваю дальше, до следующего нужного абзаца. Снова клонирую (дублирую) вкладку.
Опять перехожу в следующую клонированную вкладку и опять прокручиваю дальше, до следующего нужного абзаца. И так далее.
Получается, например, 5 вкладок одной и той же страницы, но позиция прокрутки во всех вкладках разная. Далее сохраняю сессию.


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


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

Надеюсь, не очень путанно описал проблему.


«The Truth Is Out There»

Отсутствует

 

№1621515-01-2022 20:57:41

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

Re: Custom Buttons

unter_officer пишет

И боюсь, что описание получится длинновато.
Надеюсь, не очень путанно описал проблему.

Замечательно. Чётко, подробно, доходчиво.


Попробовал воспроизвести, но не получилось.
Вернее, получилось частично. В первых четырёх вкладках позиция прокрутки
сохранилась актуальная, а в пятой (последней) осталась как в четвёртой (предпоследней).


Подумал, может нужно время, чтобы устоялось, и точно, так и есть.
Проделал то же самое, но не сразу полез сохранять, а подождал сколько-то [десятков секунд(?)].
Затем сохранил, восстановил, и у каждой из пяти восстановилась своя позиция прокрутки.


В любом случае, со стороны кода здесь вряд ли что-то можно сделать,
он же не сам собирает сессионные данные, а просто запрашивает у лисы SessionStore.getBrowserState().
Если бы точно знать где затык, то можно было бы попробовать подумать как пнуть, но разве найдёшь.


ВВП пишет

Можно кнопкой это

Из кнопки это жутко неудобно.
В чём проблема, закинул в UCF custom_script.js и все дела.

Отсутствует

 

№1621615-01-2022 22:38:53

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

Re: Custom Buttons

DEL

Отредактировано unter_officer (16-01-2022 03:30:17)


«The Truth Is Out There»

Отсутствует

 

№1621716-01-2022 01:33:44

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

Re: Custom Buttons

Dumby
unter_officer
browser.sessionstore.interval

Отсутствует

 

№1621816-01-2022 03:22:38

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

Re: Custom Buttons

_zt пишет

Dumby
unter_officer
browser.sessionstore.interval

_zt, спасибо за наводку.
У меня этот параметр был изменён с 15000 (по умолчанию) на 300000.
Вернул значение по умолчанию и вроде всё стало нормально сохраняться.
Для полной уверенности потестирую ещё и понаблюдаю, но думаю, что проблема была именно в этом.


«The Truth Is Out There»

Отсутствует

 

№1621916-01-2022 04:10:16

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

Re: Custom Buttons

Dumby
Почему сразу ::-moz-selection ?
2ss7y6pq.jpg

Отсутствует

 

№1622016-01-2022 09:04:04

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

Re: Custom Buttons

ВВП пишет

Почему сразу ::-moz-selection ?

Потому, что у них так сделано.
resource://gre/modules/CommonDialog.jsm

скриншот

Выделить код

Код:




А почему у них так сделано я не знаю.
Наверно считается, что это удобно.

Отсутствует

 

№1622116-01-2022 11:52:18

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

Re: Custom Buttons

Dumby

Dumby пишет

Наверно считается, что это удобно.

Так, а шо делать ? Рихтую,рихтую и никак...Вру, сделал...Похожая шняга с editBMPanel_namePicker , от тут не знаю.
Да нет, нашел в editBookmark.js ,короче, меняю на focused

Отредактировано ВВП (16-01-2022 12:41:45)

Отсутствует

 

№1622216-01-2022 21:29:06

Азат55555
Участник
 
Группа: Members
Зарегистрирован: 01-11-2018
Сообщений: 28
UA: Yandex 22

Re: Custom Buttons

Простите что отвлекаю. Как заархивировать папку? Есть папка по пути C:\\Папка нужно через CB превратить в Папка.rar

Отсутствует

 

№1622317-01-2022 21:11:54

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

Re: Custom Buttons

Азат55555 пишет

Как заархивировать папку? Есть папка по пути C:\\Папка нужно через CB превратить в Папка.rar

Запустить программу-архиватор с аргументами, очевидно же.

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

Выделить код

Код:

var process = Cc["@mozilla.org/process/util;1"].createInstance(Ci.nsIProcess);
process.startHidden = true;

process.init(FileUtils.File("C:\\Program Files (x86)\\WinRAR\\rar.exe"));

var args = ["a", "-ep1", "-r", "Y:\\Папка.rar", "C:\\Папка\\*"];

process.runw(false, args, args.length);

Отсутствует

 

№1622419-01-2022 15:31:28

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

Re: Custom Buttons

Dumby
Переделайте пожалуйста кнопочку для UCF.

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

Выделить код

Код:

/*Initialization Code*/

// Сохранить как PNG ..........
((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");

            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()) + '\u00F7' + 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 = "#0f0 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));
        }
    }`
});


«The Truth Is Out There»

Отсутствует

 

№1622520-01-2022 17:05:29

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

Re: Custom Buttons

unter_officer пишет

Переделайте пожалуйста кнопочку для UCF.

Нехочу. Можешь просто сам обернуть, типа

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

Выделить код

Код:

(async func => CustomizableUI.createWidget({
	id: "797743",
	label: "Сохранить как PNG",
	tooltiptext: "Сохранить как PNG",
	localized: false,
	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", "resource://usercontext-content/pet.svg");
	}
}))(() => {

// Здесь код

});


Bug 1743099 - Remove unused AddonType bits (Firefox 98+)
Кнопки с about:addons тю-тю. Соберу сейчас, на всякий случай.


Custom Buttons 0.0.7.0.0.23, paxmod и bootstrap в zip-папке.

Отсутствует

 

Board footer

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