Scripting der Oberfläche wird nur über die kommende Theming-API, Scripting von Webseiten weiter uneingeschränkt möglich sein.
userChrome.js Scripte für den Fuchs (Diskussion)
-
Endor -
12. Mai 2015 um 14:19 -
Erledigt
-
-
Das heißt dann ja, dass alles Scripte, die wir hier so zur Anpassung der Oberfläche haben,
zbs. einfügen von Schaltflächen mit und ohne Kontextmenü usw. dann für die Tonne sind.
Klar man könnte die dann, wenn überhaupt möglich, an diese Theming-API anpassen.
Aber wird sich auch jemand diese Mühe machen?
Mfg.
Endor -
Danke.
Edit: Ups, Folgeseite übersehen. Klingt ja eher demotiverend, schade.
-
Das Einfügen von Schaltflächen lässt sich ja problemlos per WebExtension umsetzen. Die meisten Add-ons fügen irgendwelche Schaltflächen zur Oberfläche dazu.
Die Theming-API selbst wird keine beliebigen Änderungen erlauben, sondern konkret die Dinge änderbar machen, welche mit den neuen Themes möglich sein werden. Das heißt: Farbe der Tabs dynmaisch ändern: klar; Icons dynmaisch austauschen: bestimmt; die Uhrzeit neben den Tabs einblenden: ganz sicher nicht.
-
Klar, nur aus jedem Script dann gleich eine Erweiterung machen, wird man auch nicht wollen.
Aber wie ich es verstanden habe, wird es anders dann nicht mehr möglich sein.
Mal abwarten was wird. Im Moment ist einfach noch zu wenig darüber bekannt.
Mfg.
Endor -
So traurige Antwort, aber es war abzusehen, Userchrome Scripte sind ab Firefox 57 erledigt.
Greasemonkey Scripte sind das einzige was noch bleiben wird.Hier was mir der Autor geschrieben hat:
Zitat von Alta88I haven't looked into it, but from what I understand it won't/can't be as privileged as it is currently. It can likely be made to execute snippets of code but not on chrome or using xpcom apis, ie I doubt it can add an xul statusbar to the dom, etc. so won't be hugely useful. And the xul overlay part is surely dead. For web page manipulation, whatever GreaseMonkey can do is an alternative. I'm also unlikely to take the time to figure it out/convert it, sorry.
The current state of thinking is well described by the door slam on the old bug to implement natively a userChrome.js type file:
https://bugzilla.mozilla.org/show_bug.cgi?id=332529Quelle: http://forums.mozillazine.org/viewtopic.php?p=14747813#p14747813
Mal sehen was er dann schlussendlich macht.
Mfg.
Endor -
Tja, lange haben wir uns hier gehalten, aber so wie es aussieht, werden wir jetzt von Mozilla selbst eingemottet
Die Frage ist, was macht man denn jetzt, weil den Fuchs von der Stange kann man so eigentlich nicht gebrauchen (ich zumindest nicht). Und wenn ich den Fuchs nicht mehr einfach nach meinen persönlichen Bedürfnissen mit Codes, Scripten, ect... verschrauben kann, mir stattdessen mühsam 100 Addons dafür suchen (ob es die alle gibt, weiß auch keiner) und alle probieren muss (dafür hab ich überhaupt keine Zeit), um am Ende feststellen zu müssen, dass es alles doch irgendwie nicht passt, ja dann...
Tja, was macht man dann?
-
Vielleicht das hier:
Quelle: http://forums.mozillazine.org/viewtopic.php?p=14747866#p14747866
Zitat von moratThe config file may still work in Firefox 57. You can use the mozilla.cfg file as an alternative to the userChrome.js file.
Customizing Firefox – Advanced Autoconfig Files
http://mike.kaply.com/2012/03/22/cus…toconfig-files/Demo mozilla.cfg file
http://mike.kaply.com/wp-content/blo…016/02/demo.cfgObserver Notifications
http://developer.mozilla.org/en-US/docs/Observer_NotificationsDeploying Firefox in an enterprise environment
http://developer.mozilla.org/en-US/Firefox/…rise_deploymentWenn man das, was userChromeJS macht, von der Konfigurationsdatei erledigen lassen kann, könnte das funktionieren.
-
Bitte testet mal:
In den Fx-Programmordner kommt eine Datei namens "config.js" mit nachfolgendem Inhalt.
Es ist das Skript userChrome_js.js aus dem Ordner components der Erweiterung mit einigen Änderungen.Code
Alles anzeigen// /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is the userChrome.js component. * * The Initial Developer of the Original Code is * Simon Bünzli <zeniko@gmail.com> * * Portions created by the Initial Developer are Copyright (C) 2007 * the Initial Developer. All Rights Reserved. * * Contributor(s): * alta88 <alta88@gmail.com> * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ try { /*Components.utils.import("resource://gre/modules/XPCOMUtils.jsm"); */ Components.utils.import("resource://gre/modules/osfile.jsm"); function UserChrome_js() { var os = Components.classes["@mozilla.org/observer-service;1"] .getService(Components.interfaces.nsIObserverService); os.addObserver(this, "final-ui-startup", false); }; UserChrome_js.prototype = { /*// Properties required for XPCOM registration: classDescription: "userChromeJS Loading Component", classID : Components.ID("{8DEB3B5E-7585-4029-B6D0-4733CE8DED50}"), contractID : "@userChromeJS;1", _xpcom_categories: [{ category: "app-startup", service: true }], */ /* ........ QueryInterface .................................................. */ /* QueryInterface: XPCOMUtils.generateQI([Components.interfaces.nsISupports, Components.interfaces.nsIObserver, Components.interfaces.nsIModule, Components.interfaces.nsIFactory, Components.interfaces.nsIDOMEventListener]), */ /* ........ nsIObserver ..................................................... */ observe: function(aSubject, aTopic, aData) { var os = Components.classes["@mozilla.org/observer-service;1"] .getService(Components.interfaces.nsIObserverService); switch (aTopic) { /* case "app-startup": case "profile-after-change": os.addObserver(this, "final-ui-startup", false); break; */ case "final-ui-startup": var file = Components.classes["@mozilla.org/file/directory_service;1"] .getService(Components.interfaces.nsIProperties) .get("UChrm", Components.interfaces.nsIFile); file.append("userChrome.js"); /* if (!file.exists()) { var componentFile = __LOCATION__; var componentsDir = componentFile.parent; var extensionDir = componentsDir.parent; extensionDir.append("README.txt"); if (extensionDir.exists()) extensionDir.copyTo(file.parent, "userChrome.js"); } */ if (file.exists() && file.isFile() && !Components.classes["@mozilla.org/xre/app-info;1"] .getService(Components.interfaces.nsIXULRuntime) .inSafeMode) { this.mFileURL = Components.classes["@mozilla.org/network/io-service;1"] .getService(Components.interfaces.nsIIOService) .getProtocolHandler("file") .QueryInterface(Components.interfaces.nsIFileProtocolHandler) .getURLSpecFromFile(file); var path = OS.Constants.Path.libDir; path = OS.Path.join(path, "userChromeJS.js"); this.uCFileURI = OS.Path.toFileURI(path); os.addObserver(this, "domwindowopened", false); } break; case "domwindowopened": aSubject.addEventListener("load", this, true); break; } }, /* ........ nsIDOMEventListener ............................................. */ handleEvent: function(aEvent) { var document = aEvent.originalTarget; if (document.location && document.location.protocol == "chrome:") { try { let loader = Components.classes["@mozilla.org/moz/jssubscript-loader;1"] .getService(Components.interfaces.mozIJSSubScriptLoader); /* loader.loadSubScript("chrome://userChromeJS/content/userChromeJS.js", document.defaultView, "UTF-8"); */ loader.loadSubScript(this.uCFileURI, document.defaultView, "UTF-8"); loader.loadSubScript(this.mFileURL, document.defaultView, "UTF-8"); } catch (ex) { // script execution can be stopped with |throw "stop";| if (ex !== "stop") { Components.utils.reportError(ex); } } } } }; /** * The following line is what XPCOM uses to create components. Each component * prototype must have a .classID which is used to create it. * * XPCOMUtils.generateNSGetFactory was introduced in Mozilla 2 (Firefox 4). */ /*var NSGetFactory = XPCOMUtils.generateNSGetFactory([UserChrome_js]); */ new UserChrome_js(); } catch(ex) { Components.utils.reportError(ex); };
In den Programmordner kommt noch eine zweite Datei namens "userChromeJS.js". Das ist das unveränderte Skript userChromeJS.js aus dem Ordner content der Erweiterung:
Code
Alles anzeigen/* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is the userChromeJS utilities. * * The Initial Developer of the Original Code is * alta88 <alta88@gmail.com> * * Portions created by the Initial Developer are Copyright (C) 2014 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ var EXPORTED_SYMBOLS = ["userChrome"]; /* ........ Utility functions ............................................... */ var userChrome = { path: null, dirToken: null, get loadOverlayDelay () { if (!this._loadOverlayDelay) this._loadOverlayDelay = 500; return this._loadOverlayDelay; }, set loadOverlayDelay(delay) { this._loadOverlayDelay = delay; }, get loadOverlayDelayIncr() { if (!this._loadOverlayDelayIncr) this._loadOverlayDelayIncr = 1600; return this._loadOverlayDelayIncr; }, set loadOverlayDelayIncr(delay) { this._loadOverlayDelayIncr = delay; }, import: function(aPath, aRelDirToken) { let file; this.path = aPath; this.dirToken = aRelDirToken; if (aRelDirToken) { // Relative file let absDir = this.getAbsoluteFile(aRelDirToken); if (!absDir) return; let pathSep = absDir.path.match(/[\/\\]/)[0]; file = absDir.path + (aPath == "*" ? "" : pathSep + aPath.replace(/[\/\\]/g, pathSep)); } else // Absolute file file = aPath; file = this.getFile(file); if (!file) return; if (file.isFile()) { if (/\.js$/i.test(file.leafName)) this.loadScript(file, aRelDirToken, null); else if (/\.xul$/i.test(file.leafName)) { let xul_files = []; xul_files.push(file); this.loadOverlay(xul_files, this.dirToken, null, this.loadOverlayDelay); // this.loadOverlayDelay = this.loadOverlayDelay + this.loadOverlayDelayIncr; } else this.log("File '" + this.path + "' does not have a valid .js or .xul extension.", "userChrome.import"); } else if (file.isDirectory()) this.importFolder(file); else this.log("File '" + this.path + "' is neither a file nor a directory.", "userChrome.import"); }, loadScript: function(aFile, aFolder, aRelDirToken) { setTimeout(function() { Components.classes["@mozilla.org/moz/jssubscript-loader;1"] .getService(Components.interfaces.mozIJSSubScriptLoader) .loadSubScript(userChrome.getURLSpecFromFile(aFile), null, // defaults to the global object of the caller. userChrome.charSet); // log it userChrome.log(aRelDirToken ? ("[" + aRelDirToken + "]/" + (aFolder && aFolder != "*" ? aFolder + "/" : "") + aFile.leafName) : aFile.path, "userChrome.loadScript"); }, 0); }, // XXX: Due to bug 330458, an overlay must finish before another can be // called, otherwise neither are successful. Implementing an observer to // serialize is better left as a fix in the core bug. Here, settimout values // are set to minimize but there is no quarantee; overlay cdata (if any) // needs to consider overlay completions and logging does not strictly mean // an overlay has completed, rather that the overlay file has been invoked. loadOverlay: function(aFiles, aRelDirToken, aFolder, aDelay) { //userChrome.log(aDelay+" multiple import delay", userChrome.loadOverlay); // Increment multiple import delay this.loadOverlayDelay = this.loadOverlayDelay + this.loadOverlayDelayIncr; setTimeout(function() { if (aFiles.length > 0) { //userChrome.log(userChrome.loadOverlayDelay+" inter folder delay", userChrome.loadOverlay); // log it userChrome.log(aRelDirToken ? ("[" + aRelDirToken + "]/" + (aFolder && aFolder != "*" ? aFolder + "/" : "") + aFiles[0].leafName) : aFiles[0].path, "userChrome.loadOverlay"); document.loadOverlay(userChrome.getURLSpecFromFile(aFiles.shift()), null); setTimeout(arguments.callee, userChrome.loadOverlayDelay); } }, aDelay); }, // Include all files ending in .js and .xul from passed folder importFolder: function(aFolder) { let files = aFolder.directoryEntries .QueryInterface(Components.interfaces.nsISimpleEnumerator); let xul_files = []; while (files.hasMoreElements()) { let file = files.getNext().QueryInterface(Components.interfaces.nsIFile); if (/\.js$/i.test(file.leafName) && file.leafName != "userChrome.js") this.loadScript(file, this.path, this.dirToken); else if (/\.xul$/i.test(file.leafName)) { xul_files.push(file); } } if (xul_files.length > 0) this.loadOverlay(xul_files, this.dirToken, this.path); }, getFile: function(aPath, aRelDirToken) { try { let file = Components.classes["@mozilla.org/file/local;1"] .createInstance(Components.interfaces.nsIFile); file.initWithPath(aPath); // Bad file doesn't throw on initWithPath, need to test if (file.exists()) return file; this.log("Invalid file '" + this.path + (this.dirToken ? ("' or file not found in directory with token '" + this.dirToken) : "") + "' or other access error.", "userChrome.getFile"); } catch (e) { // Bad folder throws on initWithPath this.log("Invalid folder '" + this.path + (this.dirToken ? ("' or folder not found in directory with token '" + this.dirToken) : "") + "' or other access error.", "userChrome.getFile"); } return null; }, getAbsoluteFile: function(aRelDirToken) { try { let absDir = Components.classes["@mozilla.org/file/directory_service;1"] .getService(Components.interfaces.nsIProperties) .get(aRelDirToken, Components.interfaces.nsIFile); return absDir; } catch (e) { this.log("Invalid directory name token '" + this.dirToken + "' or directory cannot be accessed.", "userChrome.getAbsoluteFile"); return null; } }, getURLSpecFromFile: Components.classes["@mozilla.org/network/io-service;1"] .getService(Components.interfaces.nsIIOService) .getProtocolHandler("file") .QueryInterface(Components.interfaces.nsIFileProtocolHandler) .getURLSpecFromFile, /* Console logger */ log: function(aMsg, aCaller) { Components.classes["@mozilla.org/consoleservice;1"] .getService(Components.interfaces.nsIConsoleService) .logStringMessage(this.date + " userChromeJS " + (aCaller ? aCaller +": " : "") + aMsg); }, get dateFormat() { if (!this._dateFormat) this._dateFormat = "%Y-%m-%d %H:%M:%S"; return this._dateFormat; }, set dateFormat(format) { this._dateFormat = format; }, get date() { let date = new Date(); return date.toLocaleFormat(this.dateFormat); }, set charSet(val) { this._charSet = val; }, get charSet() { if (!this._charSet) this._charSet = "UTF-8"; // use "UTF-8". defaults to ascii if null. return this._charSet; } };
Schließlich kommt noch eine Datei namens "config-prefs.js" in den Unterordner \defaults\pref des Programmordners. Darin steht:
userChromeJS ist deaktiviert oder nicht installiert. Die Datei userChrome.js und die zu importierenden Skripte befinden sich wie üblich im Ordner chrome.
Nun Neustart und die Skripte sollten ausgeführt werden.
-
aborix....
Du bist der Größte...Wahnsinn...alle Scripte funktionieren :klasse:
Herzlichen Dank dafür :klasse:
-
Ab welcher Fuchs Version soll dieser Hack erst greifen?
-
Hallo loshombre...
Ich habe das mit dem Firefox 55/Nightly getestet.
-
Gerade im aktuellen Fuchs getestet :klasse::klasse::klasse:
hombre aborix,
du solltest mal langsam darüber nachdenken, damit Kohle zu verdienen :wink:Läuft einwandfrei. Was ich hier bis jetzt an "Störungen" entdecken kann, ist die Tatsache, dass bei mir aktuell RestartFirefoxButtonM.uc.js in Version 0.3 damit nicht anspringen will. Button ist da, Funktion keine.
Allerdings mit vollem Profil mit zig anderen Scripten getestet. Kein neues Profil und auch nicht separat.Es scheint jedenfalls auf den ersten Blick genau die "einfache und simple" Alternative zu sein, die wir zumindest für die nahe Zukunft brauchen, um weiterhin Scripte laden zu können!
-
hombre aborix,
du solltest mal langsam darüber nachdenken, damit Kohle zu verdienen :wink:
Der Gedanke ist mir auch schon gekommen. :wink:In der config.js lässt sich noch einiges vereinfachen und individuell anpassen, aber eins nach dem anderen.
RestartFirefoxButtonM.uc.js funktioniert bei mir.
Wenn beim Klick eine Meldung in der Browserkonsole kommt, liefert die vielleicht einen Hinweis. -
Hallo aborix.
Erst mal vielen Dank für die ganze Mühe.
Werde ich heute dann auch mal testen und berichten.
:klasse:Klar, so kann man ab Firefox 57 auch Scripte laden, nur alles was an xul Scripten da ist
geht auch so nicht mehr. Und ob wir so die Firefox Oberfläche verändern können muss sich
dann auch erst noch raus stellen. Wahrscheinlich aber nicht. Mal sehen....
Mfg.
Endor -
alle Scripte funktionieren
.. nur alles was an xul Scripten da ist
geht auch so nicht mehr.
Da bin ich jetzt irritiert... entweder hat 2002Andreas keine XUL-Scripte am Start oder .... :-??
Ich benötige von meinen derzeit 10 XUL-Scripte mindestens diese 4..
https://github.com/ardiman/userCh…/supportmozilla
https://github.com/ardiman/userCh…eferencesbutton
https://github.com/ardiman/userCh…penchromefolder
https://github.com/ardiman/userCh…texttoclipboard
Ich muss das aber noch testen... nur heute ist anderes im Fokus..Edit : "ich" ergänzt
-
Endor hat von Firefox 57+ gesprochen.
-
Ja, stimmt.... ich war auf Nightly fokussiert..
Ggf. kann man (Jemand) ja schon mal im Vorgriff auf 57+ die XUL-Versionen kompatibel umstricken, soweit dies möglich ist... -
Ich finde es schon schräg, das Mozilla so ausgetrickst werden kann/werden muss.
Aber wenn es nicht anders möglich ist und (einige) Stylish-Code nicht mehr funktionieren, werde ich wohl auch notgedrungen auf diesen Zug aufspringen (müssen). -
Habe mein derzeitiges Nightly-Profil umgestellt und bin vollumfänglich zufrieden... eine Erweiterung eingespart..
-