Dumby
Как бы предупреждение сделать на удаление закладки на панели закладок ? Копал в controller.js , не то. Код бы IS...
Вставил в controller.js... Не фуфло ? Хотелось бы в скрипт и желательно без закладок из popup/
if (bmGuidsToRemove.length) {
if(Services.prompt.confirm(null, null, "Удалить ?"))
transactions.push(PlacesTransactions.Remove({ guids: bmGuidsToRemove }));
}
return totalItems;
},
Отредактировано ВВП (03-03-2022 19:01:46)
Отсутствует
Вставил в controller.js... Не фуфло ?
Зависит от того, что ты хочешь.
Так ведь и на папки будет confirm, и на квери,
и не только на панели закладок.
Хотелось бы в скрипт и желательно без закладок из popup/
Что значит «без закладок из popup/»?
Возможно, имеются в виду закладки, находящиеся
непосредственно на панели закладок, а не в её папках,
но это не значит, что таковые не могут появиться где-то в popup.
Плюс, через controller проходят операции множественного удаления, и чего угодно,
так что не понятно, что требуется.
Ладно, допустим так: если удаляется закладка с панели закладок (не из папок),
то выскакивает предупреждение.
Если «OK» — операция продолжается без изменений,
если «Отмена» — из операции удаления будут исключены все такие закладки.
Код для custom_script.js
(async repl => { var obj = `{\n ${ (await (await fetch("chrome://browser/content/places/controller.js")).text()) .match(/async _removeRange\(.+?\n\ +}(?=,\n)/s)[0] .replace("bmGuidsToRemove.push", repl) }\n}` var ps = await ChromeUtils.compileScript("data:,(" + encodeURIComponent(`${obj => { var patch = async ctor => { var proto = ctor.prototype, meth = proto?._removeRange; meth && Object.assign(proto, obj); } var key = "PlacesController"; var desc = Object.getOwnPropertyDescriptor(window, key); if (!desc) return; var {get} = desc; if (get) desc.get = () => { var val = get(); patch(val); return val; }, Object.defineProperty(window, key, desc); else patch(desc.value); }})(${obj});`)); var obs = doc => "PlacesController" in doc.ownerGlobal && ps.executeInGlobal(doc); var topic = "chrome-document-loaded"; Services.obs.addObserver(obs, topic); Services.obs.addObserver(function quit(s, t) { Services.obs.removeObserver(quit, t); Services.obs.removeObserver(obs, topic); }, "quit-application-granted"); })( `else if (!removedFolders.ignore && PlacesUtils.nodeIsBookmark(node)) { let info = await PlacesUtils.bookmarks.fetch( node.bookmarkGuid, null, {includePath: true} ); if ( info?.path.length == 1 && info.path[0].guid == "${PlacesUtils.bookmarks.toolbarGuid}" && !(removedFolders.ignore ??= Services.prompt.confirm( null, null, "Удалять закладки с панели закладок?" )) ) { totalItems--; continue; } } $&` );
Отсутствует
Dumby
Да где то мой косяк. И не в controller.js... в SessionFile.jsm. Кстати , папки на панели тоже надо предупреждать. Из-за них и геммор этот замутил.
Отредактировано ВВП (05-03-2022 01:45:47)
Отсутствует
ВВП, друже почему не отвечаешь в личку, может не видишь?
Win 10х64
Отсутствует
папки на панели тоже надо предупреждать. Из-за них и геммор этот замутил.
Весьма неожиданный поворот, чего сразу не сказал.
Хорошо, изменил код, чтобы подпадали папки и квери (также, только прямые потомки).
(async repl => { var obj = `{\n ${ (await (await fetch("chrome://browser/content/places/controller.js")).text()) .match(/async _removeRange\(.+?\n\ +}(?=,\n)/s)[0] .replace("// This is a common bookmark item.", repl) }\n}` var ps = await ChromeUtils.compileScript("data:,(" + encodeURIComponent(`${obj => { var patch = async ctor => { var proto = ctor.prototype, meth = proto?._removeRange; meth && Object.assign(proto, obj); } var key = "PlacesController"; var desc = Object.getOwnPropertyDescriptor(window, key); if (!desc) return; var {get} = desc; if (get) desc.get = () => { var val = get(); patch(val); return val; }, Object.defineProperty(window, key, desc); else patch(desc.value); }})(${obj});`)); var obs = doc => "PlacesController" in doc.ownerGlobal && ps.executeInGlobal(doc); var topic = "chrome-document-loaded"; Services.obs.addObserver(obs, topic); Services.obs.addObserver(function quit(s, t) { Services.obs.removeObserver(quit, t); Services.obs.removeObserver(obs, topic); }, "quit-application-granted"); })( `$& if (!removedFolders.ignore) { let info = await PlacesUtils.bookmarks.fetch(node.bookmarkGuid); if ( info?.parentGuid == "${PlacesUtils.bookmarks.toolbarGuid}" && !(removedFolders.ignore ??= Services.prompt.confirm( null, null, "Удалять с панели закладок?" )) ) { totalItems--; continue; } }` );
Отсутствует
Удалять с панели закладок?
А как вернуть ошибочно удалённую одну закладку или папку - есть для этого какой-либо скрипт, user-script или расширение ?
Подтверждение на удаление - это конечно полезная вешь, но не у всех этот код будет установлен…
Отсутствует
Dumby
№9
Как бы в этот скрипт closemenu="single" вставить? Ну, чтобы popup закладок не исчезал при возврате ?
Извиняюсь closemenu: "single", и все дела. Да, иконку не могу вставить "восстановить"
// ==/UserScript== if (typeof window.undobookmarksmenu == "undefined") { window.undobookmarksmenu = { popup: null, handleEvent: function(event) { switch (event.type) { case 'unload': this.uninit(); break; case 'popupshown': this.popupshown(event); break; } }, init: function() { window.addEventListener('unload', this, false); this.popup = document.getElementById("placesContext"); if (!this.popup) return; this.popup.addEventListener('popupshown', this, false); let template = (location.href == "chrome://browser/content/browser.xul") ? [ ] : [ ["menuitem", {id: "undobookmarksmenuUndo", disabled: "true", label: "Восстановить удаленное", key: "key_undo", oncommand: "PlacesTransactions.undo().catch(Cu.reportError);", closemenu: "single", selection: "any" }] ]; let ref = document.getElementById("placesContext_deleteSeparator"); ref.parentNode.insertBefore(this.jsonToDOM(template, document, {}), ref); }, uninit: function() { window.removeEventListener('unload', this, false); if (!this.popup) return; this.popup.removeEventListener('popupshown', this, false); }, popupshown: function(event){ var menuitem = document.getElementById("undobookmarksmenuUndo"); if (menuitem) menuitem.setAttribute('disabled', PlacesTransactions.topUndoEntry == null); menuitem = document.getElementById("undobookmarksmenuRedo"); if (menuitem) menuitem.setAttribute('disabled', PlacesTransactions.topRedoEntry == null); }, jsonToDOM: function(jsonTemplate, doc, nodes) { jsonToDOM.namespaces = { html: "http://www.w3.org/1999/xhtml", xul: "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul" }; jsonToDOM.defaultNamespace = jsonToDOM.namespaces.xul; function jsonToDOM(jsonTemplate, doc, nodes) { function namespace(name) { var reElemNameParts = /^(?:(.*):)?(.*)$/.exec(name); return { namespace: jsonToDOM.namespaces[reElemNameParts[1]], shortName: reElemNameParts[2] }; } // Note that 'elemNameOrArray' is: either the full element name (eg. [html:]div) or an array of elements in JSON notation function tag(elemNameOrArray, elemAttr) { // Array of elements? Parse each one... if (Array.isArray(elemNameOrArray)) { var frag = doc.createDocumentFragment(); Array.prototype.forEach.call(arguments, function(thisElem) { frag.appendChild(tag.apply(null, thisElem)); }); return frag; } // Single element? Parse element namespace prefix (if none exists, default to defaultNamespace), and create element var elemNs = namespace(elemNameOrArray); var elem = doc.createElementNS(elemNs.namespace || jsonToDOM.defaultNamespace, elemNs.shortName); // Set element's attributes and/or callback functions (eg. onclick) for (var key in elemAttr) { var val = elemAttr[key]; if (nodes && key == "key") { nodes[val] = elem; continue; } var attrNs = namespace(key); if (typeof val == "function") { // Special case for function attributes; don't just add them as 'on...' attributes, but as events, using addEventListener elem.addEventListener(key.replace(/^on/, ""), val, false); } else { // Note that the default namespace for XML attributes is, and should be, blank (ie. they're not in any namespace) elem.setAttributeNS(attrNs.namespace || "", attrNs.shortName, val); } } // Create and append this element's children var childElems = Array.prototype.slice.call(arguments, 2); childElems.forEach(function(childElem) { if (childElem != null) { elem.appendChild( childElem instanceof doc.defaultView.Node ? childElem : Array.isArray(childElem) ? tag.apply(null, childElem) : doc.createTextNode(childElem)); } }); return elem; } return tag.apply(null, jsonTemplate); } return jsonToDOM(jsonTemplate, doc, nodes); } } window.undobookmarksmenu.init(); }
Отредактировано ВВП (06-03-2022 20:26:33)
Отсутствует
closemenu: "single", и все дела. Да, иконку не могу вставить "восстановить"
Да так же, как closemenu: "single", только два:
class: "menuitem-iconic",
image: "chrome://browser/content/robot.ico"
И чего там за browser.xul недоудалён, тогда уж сразу
let template = ["menuitem", {…}];
Ну и label лучше просто "Отменить".
Транзакции в topUndoEntry не обязательно на удаление,
может быть добавление, редактирование, сортировка, перемещение.
То есть, добавил закладку, и пункт станет взведён отменить это добавление,
но название в меню всё равно будет торчать "Восстановить удаленное".
Отсутствует
ВВП
Да необязательно, можно и оставить.

Отсутствует
Хочу чтобы пункт "восстановить" не появлялся раньше времени... Только после удаления.
// ==/UserScript== if (typeof window.undobookmarksmenu == "undefined") { window.undobookmarksmenu = { popup: null, handleEvent: function(event) { switch (event.type) { case 'unload': this.uninit(); break; case 'popupshown': this.popupshown(event); break; } }, init: function() { window.addEventListener('unload', this, false); this.popup = document.getElementById("placesContext"); if (!this.popup) return; this.popup.addEventListener('popupshown', this, false); let template = ["menuitem", {id: "undobookmarksmenuUndo", disabled: "false", label: "Восстановить удаленное", class: "menuitem-iconic", image: "", oncommand: "PlacesTransactions.undo().catch(Cu.reportError);", closemenu: "single" }] ; let ref = document.getElementById("placesContext_deleteSeparator"); ref.parentNode.insertBefore(this.jsonToDOM(template, document, {}), ref); }, uninit: function() { window.removeEventListener('unload', this, false); if (!this.popup) return; this.popup.removeEventListener('popupshown', this, false); }, popupshown: function(event){ var menuitem = document.getElementById("undobookmarksmenuUndo"); if (menuitem) menuitem.setAttribute('disabled', PlacesTransactions.topUndoEntry == null); menuitem = document.getElementById("undobookmarksmenuRedo"); if (menuitem) menuitem.setAttribute('disabled', PlacesTransactions.topRedoEntry == null); }, jsonToDOM: function(jsonTemplate, doc, nodes) { jsonToDOM.namespaces = { html: "http://www.w3.org/1999/xhtml", xul: "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul" }; jsonToDOM.defaultNamespace = jsonToDOM.namespaces.xul; function jsonToDOM(jsonTemplate, doc, nodes) { function namespace(name) { var reElemNameParts = /^(?:(.*):)?(.*)$/.exec(name); return { namespace: jsonToDOM.namespaces[reElemNameParts[1]], shortName: reElemNameParts[2] }; } // Note that 'elemNameOrArray' is: either the full element name (eg. [html:]div) or an array of elements in JSON notation function tag(elemNameOrArray, elemAttr) { // Array of elements? Parse each one... if (Array.isArray(elemNameOrArray)) { var frag = doc.createDocumentFragment(); Array.prototype.forEach.call(arguments, function(thisElem) { frag.appendChild(tag.apply(null, thisElem)); }); return frag; } // Single element? Parse element namespace prefix (if none exists, default to defaultNamespace), and create element var elemNs = namespace(elemNameOrArray); var elem = doc.createElementNS(elemNs.namespace || jsonToDOM.defaultNamespace, elemNs.shortName); // Set element's attributes and/or callback functions (eg. onclick) for (var key in elemAttr) { var val = elemAttr[key]; if (nodes && key == "key") { nodes[val] = elem; continue; } var attrNs = namespace(key); if (typeof val == "function") { // Special case for function attributes; don't just add them as 'on...' attributes, but as events, using addEventListener elem.addEventListener(key.replace(/^on/, ""), val, false); } else { // Note that the default namespace for XML attributes is, and should be, blank (ie. they're not in any namespace) elem.setAttributeNS(attrNs.namespace || "", attrNs.shortName, val); } } // Create and append this element's children var childElems = Array.prototype.slice.call(arguments, 2); childElems.forEach(function(childElem) { if (childElem != null) { elem.appendChild( childElem instanceof doc.defaultView.Node ? childElem : Array.isArray(childElem) ? tag.apply(null, childElem) : doc.createTextNode(childElem)); } }); return elem; } return tag.apply(null, jsonTemplate); } return jsonToDOM(jsonTemplate, doc, nodes); } } window.undobookmarksmenu.init(); }
Отредактировано ВВП (07-03-2022 10:52:53)
Отсутствует
Dumby
Приморила псевдосессия. На пустую вкладку , как newtab прет сессия. Старый код не того. Или в модулях копать ?
Нашел код,снят вопрос. Хотя, хотелось бы рихтануть в модулях SessionFile.jsm и SessionStore.jsm... Или этого мало ?
Отредактировано ВВП (07-03-2022 22:30:28)
Отсутствует
Хочу чтобы пункт "восстановить" не появлялся раньше времени... Только после удаления.
Вот, появление только при наличии в topUndoEntry транзакции удаления.
(async sep => { if (!sep) return; var key = "hasRemoveTransaction"; var g = Cu.import("resource://gre/modules/PlacesTransactions.jsm", {}); if (!g[key]) { Services.scriptloader.loadSubScript( `data:,this.${key}=TransactionsHistory.proxifiedToRaw;`, g ); var raws = g[key]; g[key] = entry => { for(var tr of entry) if (raws.get(tr) instanceof PlacesTransactions.Remove) return true; } } var menuitem = document.createXULElement("menuitem"); for(var args of Object.entries({ closemenu: "single", class: "menuitem-iconic", id: "placesCmd_undoRemove", label: "Восстановить удалённое", oncommand: "PlacesTransactions.undo().catch(Cu.reportError);", image: "", })) menuitem.setAttribute(...args); var desc = Object.getOwnPropertyDescriptor(XULElement.prototype, "hidden"); var {set} = desc; desc.set = () => { var entry = PlacesTransactions.topUndoEntry; var vis = entry && g[key](entry); vis && menuitem.removeAttribute("disabled"); set.call(menuitem, !vis); } Object.defineProperty(menuitem, "hidden", desc); sep.before(menuitem); })(document.getElementById("placesContext_deleteSeparator"));
Отредактировано Dumby (08-03-2022 22:24:01)
Отсутствует
Dumby
Вот, появление только при наличии в topUndoEntry транзакции удаления.
Высший класс ! Нужная фича, однозначно. А, как бы дорихтовать фишку ? Удалил и не хочу восстанавливать, а пункт "восстановить" висит дальше. Сброс бы сделать ?
Отредактировано ВВП (08-03-2022 11:43:05)
Отсутствует
Dumby
В корне панели закладок пункт неактивный.
Еще скрипт создает в конт.меню панели закладок лишний разделитель, в нижней части меню, хотя сам пункт появляется в секции "изменить/удалить".
Отсутствует
Удалил и не хочу восстанавливать, а пункт "восстановить" висит дальше. Сброс бы сделать ?
В смысле удалить topUndoEntry?
Если да, то по какому действию?
Иными словами, что укажет на произволение «не хочу восстанавливать»,
чтобы сделать сброс.
И, чисто теоретически,
если topUndoEntry займёт предыдущая операция (if any),
и если она тоже содержит транзакцию удаления,
то «пункт "восстановить"» продолжит висеть. Это окей?
В корне панели закладок пункт неактивный.
Да, вижу. И на шевроне тоже.
Добавил принудительное удаление атрибута "disabled".
скрипт создает в конт.меню панели закладок лишний разделитель, в нижней части меню
Нет, этого не вижу. Ну, не «создает» — это стопроцентно.
Может как-то приводит к его появлению (что интересно).
Есть чистый профиль с UCF, чтобы проверить?
Отредактировано Dumby (08-03-2022 22:22:45)
Отсутствует
Dumby
Да, не создает. Похоже что возвращает безID-ишный разделитель отделяющий пункты расширений от основного меню.
..
Меню переупорядочено и сепаратор был скрыт как #placesContext > menuseparator:nth-child(x2) (с этим скриптом - который создает два пункта), а тут теперь #placesContext > menuseparator:nth-child(x1) нужен, так как пункт всего один. (похоже что так)
Отредактировано _zt (09-03-2022 01:47:32)
Отсутствует