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

Общайтесь со знакомыми и друзьями в нашей группе в Контакте.

№1612606-01-2022 01:09:32

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

Re: Custom Buttons

Dumby

Dumby пишет

SessionStore.canRestoreLastSession = false;

Класс!

Отсутствует

 

№1612706-01-2022 10:09:02

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

Re: Custom Buttons

unter_officer пишет

думаю на это можно не обращать внимания

Ну почему же, весьма интересное наблюдение. Можно заменить originalURI на URI
он без "view-source:" идёт. Это если и для таких запросов нужно UA устанавливать.
А если не нужно, то можно отсечь, вписать куда-нибудь проверку на это дело, например,

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

Выделить код

Код:

(async topic => {
	var observer = channel => {
		if (channel instanceof Ci.nsIHttpChannel) {
			var uri = channel.originalURI;
			if (uri.schemeIs("view-source")) return;
			var ua, {host} = uri;

Отсутствует

 

№1612806-01-2022 14:51:09

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

Re: Custom Buttons

Dumby пишет

Ну почему же, весьма интересное наблюдение. Можно заменить originalURI на URI
он без "view-source:" идёт. Это если и для таких запросов нужно UA устанавливать.
А если не нужно, то можно отсечь, вписать куда-нибудь проверку на это дело, например,

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

Выделить код

Код:

(async topic => {
	var observer = channel => {
		if (channel instanceof Ci.nsIHttpChannel) {
			var uri = channel.originalURI;
			if (uri.schemeIs("view-source")) return;
			var ua, {host} = uri;

Dumby, спасибо. Теперь всё супер.


«The Truth Is Out There»

Отсутствует

 

№1612907-01-2022 07:58:40

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

Re: Custom Buttons

Подскажите способ проверить наличие файла с системным путём, то есть не привязанным к chrome://…
например файл "/usr/bin/konsole" — ищем в Линукс, "C:\windows\explorer.exe" — в Windows.


а затем запустить найденный файл в Линукс или Windows, в зависимости, в какой системе открыт Firefox.

Отсутствует

 

№1613007-01-2022 14:51:11

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

Re: Custom Buttons

Dobrov пишет

Подскажите способ проверить наличие файла

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

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

nsIFile.idl

Выделить код

Код:

alert(
	FileUtils.File(String.raw`C:\windows\explorer.exe`).exists()
);

Правда щас модно (асинхрон)
IOUtils.webidl

Выделить код

Код:

(async () => alert(
	await IOUtils.exists(String.raw`C:\windows\explorer.exe`)
))();

затем запустить найденный файл

Выделить код

Код:

FileUtils.File(String.raw`C:\windows\explorer.exe`).launch();

Отсутствует

 

№1613107-01-2022 21:32:21

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 596
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»

Отсутствует

 

№1613208-01-2022 01:32:36

Dobrov
Участник
 
Группа: Members
Зарегистрирован: 04-10-2011
Сообщений: 471
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)

Отсутствует

 

№1613311-01-2022 22:04:58

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

Re: Custom Buttons

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

Отсутствует

 

№1613413-01-2022 21:03:37

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2247
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
);

Отсутствует

 

№1613514-01-2022 03:02:57

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 596
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»

Отсутствует

 

№1613614-01-2022 22:56:52

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

Re: Custom Buttons

Dumby

Dumby пишет

Simple Session Manager

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

Отсутствует

 

№1613715-01-2022 00:43:48

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

Re: Custom Buttons

ВВП пишет

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

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

Отсутствует

 

№1613815-01-2022 01:06:15

unter_officer
Участник
 
Группа: Members
Откуда: Санкт-Петербург
Зарегистрирован: 27-03-2011
Сообщений: 596
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»

Отсутствует

 

№1613915-01-2022 09:33:13

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2247
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");

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

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

Отсутствует

 

№1614015-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

Отсутствует

 

№1614115-01-2022 12:46:27

Dumby
Участник
 
Группа: Members
Зарегистрирован: 12-08-2012
Сообщений: 2247
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;
        }
*/

Отсутствует

 

№1614215-01-2022 13:19:33

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

Re: Custom Buttons

Dumby

ВВП пишет

Simple Session Manager

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

Отсутствует

 

№1614315-01-2022 13:37:06

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

Re: Custom Buttons

Dumby пишет

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

Dumby пишет

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

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

Отсутствует

 

№1614415-01-2022 17:09:47

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

Re: Custom Buttons

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

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

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

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


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


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


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


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

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


«The Truth Is Out There»

Отсутствует

 

№1614515-01-2022 20:57:41

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

Re: Custom Buttons

unter_officer пишет

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

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


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


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


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


ВВП пишет

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

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

Отсутствует

 

№1614615-01-2022 22:38:53

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

Re: Custom Buttons

DEL

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


«The Truth Is Out There»

Отсутствует

 

№1614716-01-2022 01:33:44

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

Re: Custom Buttons

Dumby
unter_officer
browser.sessionstore.interval

Отсутствует

 

№1614816-01-2022 03:22:38

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

Re: Custom Buttons

_zt пишет

Dumby
unter_officer
browser.sessionstore.interval

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


«The Truth Is Out There»

Отсутствует

 

№1614916-01-2022 04:10:16

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

Re: Custom Buttons

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

Отсутствует

 

№1615016-01-2022 09:04:04

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

Re: Custom Buttons

ВВП пишет

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

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

скриншот

Выделить код

Код:




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

Отсутствует

 

Board footer

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