1. Nachrichten
  2. Forum
    1. Unerledigte Themen
    2. Forenregeln
  3. Spenden
  • Anmelden
  • Registrieren
  • Suche
Alles
  • Alles
  • Artikel
  • Seiten
  • Forum
  • Erweiterte Suche
  1. camp-firefox.de
  2. BrokenHeart

Beiträge von BrokenHeart

  • Probleme mit AnyDesk...

    • BrokenHeart
    • 18. September 2023 um 16:38

    AnyDesk (7.1.16, freie Vers.)

    Seit ein paar Stunden komme ich nicht mehr sofort ins AnyDesk-Netzwerk rein. Meldung "Keine Verbindung zum AnyDesk-Netzwerk". Dann immer wieder erfolglose Verbindungsversuche mit dem gleichen Ergebnis. Firewall und Ports sind offen (wurde auch nichts geändert).

    Irgendwann (nach ca. 10-15Minuten) funktioniert es dann und es erscheint oben folgender Balken dauerhaft(auch auf dem Remote-Desktop), der vorher auch nicht da war:

    Meine Frage: Hat sonst noch jemand Probleme mit AnyDesk oder eine Idee, was da bei mir oder AnyDesk schief läuft?

  • Firefox stürzt ab

    • BrokenHeart
    • 15. September 2023 um 18:14
    Zitat von Sören Hentzschel

    Erledigt. Ich hab das Graecum sowie das große Latinum.

    Dito :thumbup:.

    Zitat von Sören Hentzschel

    Altgriechisch und Latein sind „Minderheitensprachen“.

    Wahrscheinlich meinte milupo aber eher "lebende" Sprachen von noch "lebenden" Minderheiten. Ich kenne zumindest keinen "Altgriechen" oder "Altrömer" der sich mit anderen längere Zeit auf Altgriechisch oder Latein unterhält. ;)

    Sollte vermutlich bei den Sorben etwas anders sein...

  • Website öffnet nicht

    • BrokenHeart
    • 15. September 2023 um 14:01
    Zitat von 2002Andreas

    Ein Grund mehr, auf die aktuelle neuere ESR Version zu wechseln, wenn man sie denn nutzen will ;)

    Stimmt! Wenn ich das richtig in Erinnerung habe ist in ca. 10 Tagen ja eh das "Zwangsupdate" auf 115esr fällig. ;)

  • Website öffnet nicht

    • BrokenHeart
    • 15. September 2023 um 13:51

    Ich kann sein Problem reproduzieren, ebenfalls auf

    Zitat von andie-4458

    FF-Version 102.15.1-esr.

    Seite bleibt leer, auch im Fehlerbehebungsmodus.

  • Button für ein Popup zur Zeitanzeige verschiedener Städte in verschiedenen Zeitzonen

    • BrokenHeart
    • 14. September 2023 um 22:15
    Zitat von Mira_Belle

    Wie gestalte ich das Aussehen eines "Popups" per CSS, dass es so noch gar nicht gibt?

    Den Dialog-Code hat Sören doch hier gepostet (und in der Konsole ausprobiert):

    Beitrag

    RE: Button für ein Popup zur Zeitanzeige verschiedener Städte in verschiedenen Zeitzonen

    […]

    Funktioniert das im Script nicht? Da ich keine Scripts nutze, kann ich es nicht testen. Über die Konsole funktioniert das zumindest, das wird im DOM des Browsers eingefügt.

    (Quelltext, 18 Zeilen)

    Das sieht so nicht hübsch aus und macht nichts Sinnvolles, sondern soll nur die grundsätzliche Idee verständlich machen.



    […]

    Wir haben ja nicht wirklich fast 65.000 Mitglieder. Klar, so viele sind registriert, aber die meisten haben sich einmalig (oder auch ein paar Mal) wegen eines Problems…
    Sören Hentzschel
    14. September 2023 um 10:30

    und ich habe es dann in einem Skript getestet(siehe Edit in #25) und es läuft.

    Dürfte doch genau das sein, was du haben möchtest.. :/

  • Button für ein Popup zur Zeitanzeige verschiedener Städte in verschiedenen Zeitzonen

    • BrokenHeart
    • 14. September 2023 um 21:27
    Zitat von milupo

    weil es meistens nicht zu mehr reicht.

    Das ist doch schon mal die falsche Einstellung. Gerade bei dir würde ich das eben nicht unterschreiben. Es ist immer eine Frage der Motivation, welche nicht von "Tsjakkaa, du schaffst es"-Betrügern, sondern fast immer von positivem Feedback genährt wird. Ist wie bei ganz kleinen Kindern, denen man auch sagt, dass ihr total hässliches, selbstgemaltes Bild ganz doll toll aussieht und schon lacht der/die Kleine. Gut, bei fehlender Weiterentwicklung sollte man irgendwann dann so ehrlich sein und ihm sagen, dass es einfach nur sch...e aussieht und er/sie sich anderen Freizeitbeschäftigungen widmen sollte. :P

  • Button für ein Popup zur Zeitanzeige verschiedener Städte in verschiedenen Zeitzonen

    • BrokenHeart
    • 14. September 2023 um 20:43
    Zitat von Sören Hentzschel

    Funktioniert das im Script nicht? Da ich keine Scripts nutze, kann ich es nicht testen. Über die Konsole funktioniert das zumindest, das wird im DOM des Browsers eingefügt.

    Um ehrlich zu sein, habe ich das noch nie so ausprobiert (was ich aber nachholen werde). So ist es dann ein direktes Kindobjekt des Content-Bereichs, also Teil der anzuzeigenden Webseite, Was zwar nicht meinem Verständnis eines echten Dialogfensters entspricht, aber vielleicht funktioniert das ja auch so, dass man es global für einen Einstellungsdialog oder Meldungsbox verwenden kann. Das würde einem natürlich sehr viel Arbeit ersparen. Ich werde es einfach mal im Skript-Kontext testen. Danke für den Test in der Konsole!

    Zitat von Sören Hentzschel

    Für die allermeisten dürfte schlicht und ergreifend kein Bedarf an sowas bestehen. Die Einstiegshürde ist vergleichsweise hoch und andauernd werden dadurch Probleme verursacht oder die Scripts müssen aktualisiert werden. Eine Update-Funktion gibt es ja auch keine. Dazu die oft nicht gute Code-Qualität und nicht gewährleistete Sicherheit.

    Die Aussage stimmt sicher für den absolut überwiegenden Teil der User hier. Aber wenn ich mir die Zugriffszahlen und auch die Antworten hauptsächlich auf die CSS-Threads, aber auch auf die JavaScript-Threads anschaue, dann sind doch durchaus Begehrlichkeiten da, was "Sonderwünsche" in Bedienung und Aussehen des Firefox angehen. Klar sind das meist die üblichen Verdächtigen, aber immer wieder kommen dann auch von Usern mit wenigen Beiträgen solche Fragen wie "Kann man es nicht so einrichten, dass...". Und irgendwie war und ist meine Hoffnung, dass man vielleicht irgendwann vom "Konsumenten" zum "Produzenten" wird. Die meisten, die sich hier ja aktiv beteiligen, machen das ja nicht hauptberuflich oder haben eine entsprechende Ausbildung, sondern sind begeisterungsfähige "Hobby-Code-Frickler".

    Ja die Code-Qualität ist oft unter aller Sau. Da brauche ich nur mein MultiRowTabs-Skript anschauen, welches ich mit viel Enthusiasmus, aber mit noch wenigen JavaScript-Kenntnissen "zusammengefrickelt" habe. Aber man kann sich bzw. sollte sich ja weiterentwickeln...

    Wegen der Einstiegshöhe: Für das Erstellen: ja, das lässt sich nicht verhindern! Aber für das Nutzen von CSS/JS könnte sie viel niedriger sein. Ich bin mir sicher, dass man die Möglichkeiten zur Anpassung des Firefox im Unterforum "Individuelle Anpassungen" besser "vermarkten" müsste. Es gibt meines Erachtens zu viele angeheftete Threads auf der ersten Seite (damit meine ich auch ausdrücklich meine) die mittlerweile auf den ersten Blick mehr verwirren und eher abschrecken. Wenn man bedenkt, dass man um CSS zu nutzten, eigentlich nur eine zusätzliche Datei in einem bestimmten Verzeichnis und einen Eintrag in 'about:config' oder 'user.js' braucht und für das Funktionieren von Skripten nur eine Handvoll Dateien kopieren muss, dann schrecken so viele Threads und Hilfestellungen eher ab, als das es den Einstieg erleichtert. Da könnte man sicher vieles straffen oder zusammenfassen und irgendwie didaktischer aufbereiten, dass es nicht so aussieht, wie eine lang gewachsene "Loseblatt-Sammlung". ;)

    Mein Wunsch wäre ja, dass sich z.B. ein Windows/Linux-Skript Experte mal etwas Zeit nimmt und diese Voreinstellungen und Kopieraktionen für CSS und JavaScript automatisiert. Habe ich leider keine Ahnung davon. Ich wollte schon ein C++ Programm schreiben (Windows). Aber eine '.exe' Datei, auch wenn man Sourcen und sonstige BuildTools zum selber Bauen mitliefert, würde ich persönlich auch nicht gerne verwenden. Zumal ja alles noch mit Admin-Rechten ausgeführt werden muss. Außerdem: Kanonen->Spatzen...

    Zitat von Sören Hentzschel

    Besonders problematisch finde ich die Art von Script, welche interne Firefox-Funktionen überschreibt. Denn das sind tickende Zeitbomben.

    Das sehe ich auch so und ist in den allermeisten Fällen auch überhaupt nicht nötig bei den vielen Möglichkeiten durch die Services, die von Mozilla angeboten werden. Wenn doch jemand eine Änderung eines bestehenden Verhaltens haben möchte, die das Ersetzen von bestehendem Code erfordern würde, dann muss man halt sagen, dass genau hier die Grenze der Anpassbarkeit erreicht ist. Aber es gibt fast immer Alternativen...

    Zitat von Sören Hentzschel

    Meine Sicht auf das Thema ist am Ende einfach nicht die eines neugierigen Bastlers, der jeden Stein umdrehen möchte, nur weil es halt geht. Meine Zielsetzung für Firefox geht viel mehr in die Richtung von Mozilla, ein funktionierendes Produkt zu haben, welches nicht anfällig für selbst erzeugte Defekte ist. Zumal ich viele der in diesem Forum umgesetzten Dinge - und damit will ich niemandem zu nahe treten! - sinnlos und oft auch optisch wenig ansprechend finde, was in meinem Firefox einfach keinen Platz hat.

    Für mich war der Grund, mich mit dem dem Thema JavaScript auseinanderzusetzen, ein anderer. Ich würde mich durchaus als "neugierigen Bastler" bezeichnen, der aber wirklich nur zusätzliche Skripte in seinem FireFox haben möchte, die seine Arbeit auch wirklich spürbar erleichtern ;) . Deswegen nutze ich auch nur eine Handvoll Skripte. Ein anderer Grund war, dass ich seit Ewigkeiten programmiere, aber im Laufe meines Berufslebens nie mit 'Web-Technologien' zu tun hatte. Nachdem ich mich aktiv aus der professionellen Entwicklung verabschiedet habe, habe ich eben Lust bekommen das hobbymäßig nachzuholen, was ich nie brauchte. Und da bot sich die einmalige Möglichkeit des FireFox an, die eben das mit 'userChrome' erlaubt. :)

    Edit: das Dialog-Element funktioniert einwandfrei im Skript! War eigentlich auch klar, nachdem es auch schon in der Konsole läuft. Bis auf die "Nicht-Verschiebbarkeit" verhält es sich auch wirklich so, wie ich es von einem normalen modalen Dialogfenster erwarten würde. Danke vielmals... :thumbup:

  • Button für ein Popup zur Zeitanzeige verschiedener Städte in verschiedenen Zeitzonen

    • BrokenHeart
    • 14. September 2023 um 02:56

    Das Skript für die Anzeige der Uhrzeiten in der Alert-Box ist für mich uninteressant, gibt zig Lösungen dafür, die schneller zu bewerkstelligen sind und auch immer und überall funktionieren (z.B. WindowsTaste+D).

    Aber was ich interessant finde, ist die Frage, wie man eine universell einsetzbare Dialog-Box für userChrome-Skripte verwenden kann.

    Zitat von Sören Hentzschel

    Das ist auch kein Hexenwerk (dafür gibt es das dialog-Element, welches man frei via HTML und CSS gestalten kann), aber das ist dann natürlich gleich deutlich mehr Aufwand als ein „Fertigbauteil“ zu verwenden.

    Dafür braucht man aber im userChrome-Kontext erstmal einen "HTML-Playground", also ein Element vom Typ iFrame. Das müsste man dann wohl mittels XUL erzeugen (über ein browser-Element?) und irgendwo in ein anderes Element ('vBox', 'panel') einfügen. Also quasi als zusätzlicher Bereich irgendwo eingehängt. Oder sehe ich das falsch? :/

    Bin sicher, dass das funktioniert, aber auch ziemlich aufwendig. Die Lösung hätte aber den großen Vorteil, dass man das weitere Design dann recht bequem über HTML/CSS einstellen könnte.

    Eine andere Lösung wäre über XPCOM. Leider habe ich bis heute keine einfache Dokumentierung der einzelnen Schnittstellen- Funktionen gefunden, nur die idl-Dateien selber und was die Entwickler dazu geschrieben haben, was aber immerhin meistens recht ausführlich geschieht. Hier mal 2 Links, falls Interesse an der Thematik besteht:

    XPCOM — Firefox Source Docs documentation

    nsIPromptService.idl - mozsearch

    Mit dieser Methode lassen sich "echte Dialogfenster"(also Windows-Fenster) erzeugen, allerdings ist der Aufruf und die Anpassungen auch nicht sehr "handy" (asynchroner Aufruf usw.). Ich habe es ausprobiert und es funktioniert prinzipiell eigentlich ganz gut.

    Ich persönlich finde beide Ansätze interessant.

    Noch etwas OT: Ich verstehe nicht, dass sich in diesem Forum mit seinen fast 65Tsd Mitgliedern so wenige hier für die Thematik "userChrome JavaSkripte" interessieren und sich aktiv hier im Forum austauschen wollen. JavaScript gehört doch zu den populärsten Sprachen überhaupt, daher denke ich, dass es doch sicher einige User hier im Forum gibt, die dies auch beherrschen. Ist der Grund, dass man lieber Webextensions programmiert und die userChrome-Geschichte (betrifft auch CSS) eher als Spielerei und Hackerrei betrachtet? Schön wenn Leute wie Mira sich anscheinend noch dafür begeistern können, aborix ist leider untergetaucht(?) , der Mann (Name fällt mir jetzt nicht ein), der den Wetterfuchsbutton gemacht hat, auch nicht mehr aktiv, EffPeh auch schon lange nicht mehr da. Der einzige, der sich noch mit entsprechendem Hintergrundwissen dazu äußern kann, ist anscheinend Sören Hentzschel. Der hat aber schon öfters bekundet, kein großes Interesse an der userChrome-Geschichte zu haben. Alles maßlos traurig! so ich geh jetzt weinen und dann für ein paar Stunden ins Bett... ;( :sleeping:

    PS:Wenn Mira_Belle meint, dass das, was ich geschrieben habe, inhaltlich nicht mehr zu ihrem Thread passt, kann man den Inhalt auch gerne auslagern.

  • Notepad++ 8.5.7 behebt schwere sicherheitsrelevante Fehler

    • BrokenHeart
    • 8. September 2023 um 14:32

    Hier mal die Meldung von Heise zur Sicherheitslücke vor 10 Tagen:

    Entwickler von Notepad++ ignoriert offensichtlich Sicherheitslücken
    Mehrere Sicherheitslücken gefährden den Texteditor Notepad++. Trotz Informationen zu den Lücken und möglichen Fixes steht ein Sicherheitsupdate noch aus.
    www.heise.de

    Man streitet sich noch, ob der Entwickler bisher nur einfach ignorant war und die Sicherheitslücke wirklich so gefährlich ist, oder ob es sich bei dem Programmfehler doch eher um eine theoretische Lücke gehandelt hat (Öffnen einer präparierten Datei und manuelles Anstoßen der Konvertierung von UTF16 nach UTF8).

    Egal, ist ja jetzt wohl gefixt. :)

  • userChrome.js Scripte für den Fuchs (Diskussion)

    • BrokenHeart
    • 7. September 2023 um 19:52
    Zitat von milupo

    Ach ist das wieder zum K…. :( Ich wollte gerade schreiben, dass doppeltes parentNode auf einfaches parentNode reduziert werden muss, da blockierte meine Tastatur und ich musste den Computer neu starten. Dadurch war BrokenHeart nun schneller.

    Ja, so was ist ärgerlich.

    Nächstes Mal klappt's dann... ;)

  • userChrome.js Scripte für den Fuchs (Diskussion)

    • BrokenHeart
    • 7. September 2023 um 19:08
    Zitat von Boersenfeger

    Hoffentlich gibts eine Lösung... :|

    In Zeile 48 (im Skript von Boersenfeger ) ein .parentNode entfernen.

  • Firefox Benachrichtigungen verschwinden nicht mehr

    • BrokenHeart
    • 2. September 2023 um 20:04

    Auch wenn es deine Ausführungen und Vergleiche vermuten lassen, hier geht es überhaupt nicht um "Optionitis". Bei deinem Beispiel: Die Einstellungen in einem Forum sind nur lokal im Browserfenster von Bedeutung und beeinflussen nicht die Arbeit an meinem Rechner. Aber es gibt wohl keine andere Aktion bei einem Browser, die (gewollt) so stark in mein System eingreift wie die Push-Nachrichten. Ein Popup, welches kurz erscheint und dann zwischengespeichert wird, erregt zwar kurz meine Aufmerksamkeit, aber ich kann es auch vollständig ignorieren. Bei einem Popup, welches ich erst schließen muss, geht das nicht. Es blockiert den Platz auf meinem Desktop und befindet sich immer im Vordergrund. Auch wenn ich z.B. gerade eine E-Mail in einem anderen Programm schreiben möchte. Wenn das kein Grund ist, die "Zwangsschließbarkeit" auf Website-Basis zu überschreiben, dann weiß ich auch nicht mehr...

    Welchen Grund sollte es z.B. geben, dass eine Nachrichten-Website mir eine so wichtige Nachricht zukommen lässt, dass ich sie unbedingt lesen und quittieren muss? Ich würde dann genau dort diese Interaktions-Funktion verbieten. Bei meiner Bank würde es vielleicht wieder anders aussehen...

    Deswegen spricht sehr viel dafür, zumindest die Option 'requireInteraction: true/false' global über die normalen 'preferences:...' einstellen zu können. Meiner Meinung wäre es aber noch besser, man würde das für jede Site getrennt angeben können. Die Oberfläche und Logik dafür ist ja schon vorhanden...

  • Firefox Benachrichtigungen verschwinden nicht mehr

    • BrokenHeart
    • 2. September 2023 um 13:51
    Zitat von Sophie29

    Danke für die Info. Wobei ich den Mehrwert dahinter nicht verstehe

    Ich sehe das ähnlich. Ich kann verstehen, dass man als Nutzer einer bestimmten Website Popup-Benachrichtigungen wünscht, die ich erst selbst bestätigen muss. Einfach um auf jeden Fall so informiert zu werden, dass ich es nicht übersehen kann.

    Aber die Wichtigkeit einer Benachrichtigung hat doch nicht die Website zu bestimmen. So eine zusätzliche Bestätigung sollte ausschließlich der Besucher einer Website verändern dürfen und zwar bei den Website spezifischen Berechtigungen in den normalen Einstellungen.

  • Firefox Benachrichtigungen verschwinden nicht mehr

    • BrokenHeart
    • 1. September 2023 um 21:36

    dom.webnotifications.requireinteraction.enabled -> false

    Zum Testen des Verhaltens:

    NotificationOptions.requireInteraction Sample

    Zeigt zwei Benachrichtigungen: Einmal mit und einmal ohne User-Interaktion-Flag (s.o.) gesetzt: Wenn der Wert auf auf 'false' steht, verschwindet auch das Benachrichtigungspopup, welches von der Webseite mit User-Interaktion geflagt wurde.

  • Firefox 117.0 TAB Leiste unter Adressleiste bitte

    • BrokenHeart
    • 31. August 2023 um 09:10
    Zitat von GermanFreme82

    Ich habe das jetzt einfgefügt und es schent zu funktionieren nur das + ist riesig und ich weiß nicht wie man das kleiner machen kann

    Ersetze die Zeilen 189-194 in deiner 'userChrome.css' durch folgenden Code:

    CSS
    #TabsToolbar #tabs-newtab-button > image {
            padding: 7px !important;
    }

    Das ist eine sehr schlechte Lösung, aber sonst müsste ich das MultiRowTab-Skript wieder anfassen und entweder eine spezielle Lösung für dich einfügen oder prinzipiell die Größenänderung des NewTab-Buttons aus dem Skript entfernen. Zu beiden Sachen habe ich momentan keine Lust...

  • Firefox 117.0 TAB Leiste unter Adressleiste bitte

    • BrokenHeart
    • 30. August 2023 um 23:35
    Zitat von GermanFreme82

    ich habe heute eben das neue Update auf FF117.0 gemacht und nun ist plötzlich meine TAB Leiste wieder ganz nach oben gerutscht über die Adressleiste.

    Soweit ich weiß, nutzt du ja dieses Skript für die mehrzeilige Tableiste:

    Beitrag

    Mehrzeilige Tableiste für aktuelle Firefox-Versionen

    Neues Update (26.11.2024):

    (Versteckter Text)



    :!:Wichtig: Dieses Skript wird nicht mehr weiterentwickelt. Die letzte Firefox-Version, die noch unterstützt wird ist 137.:!:



    Für aktuelle Versionen ab FF133+:

    Das Script richtet sich in erster Linie an Nutzer, die sich nicht mit CSS beschäftigen wollen, aber trotzdem eine mehrzeilige und optisch angepasste Tableiste nutzen möchten.

    Diese neue Version besteht ausschließlich aus einem User-JavaScript. Es sind keine zusätzlichen CSS-Eintragungen mehr nötig!

    In…
    BrokenHeart
    8. Dezember 2019 um 01:19

    Hast du auch auf die letzte Version vom 03.08.2023 geupdatet? Hier der angepasste Code mit deinen 'User-Settings':

    JavaScript
    // 'MultiRowTabs.uc.js' V02 by BrokenHeart
    // based on 'MultiRowTab_LiteforFx48.uc.js' from 'http://space.geocities.yahoo.co.jp/gl/alice0775' (Alice0775)
    // Thanks to aborix...
    
    /* ///////////////////////////////////////////////////////////////////////////////////
    [13.07.2021 22:45Uhr]
    - 'Nur-JavaScript-Version' (CSS wird nicht mehr benötigt)
    - Sehr viele funktionale und optische Anpassungen hinzugefügt
    [20.02.2020 18:45Uhr]
    Fehler in allen CSS-Dateien beseitigt (:thumbup:Dank an diwa fürs melden )
    [18.02.2020 13:30Uhr]
    - Anpassen des User-JavaSkripts aus (3).
    - Skript 'Tabsrunter.uc.js' wurde entfernt, da jetzt bereits in (3) integriert.
    [11.12.2019 13:15Uhr]
    - Optische Anpassungen an Code (1) und (2).
    - Neuen Code hinzugefügt für 'Tabs mit abgerundeten Ecken' (2a).
    - Anpassen des User-JavaSkripts zum Verschieben der Tabs (3).
    [08.12.2019 17:45Uhr]
    - Anpassungen am Code (1) und (2) vorgenommen, um 'Tableiste unten' zu berücksichtigen.
    - JavaSkript hinzugefügt, welches die Tableiste nach unten befördert.
    [19.08.2022 21:28Uhr]
    - Fehler Menüleiste behoben
    [23.09.2022 01:58Uhr]
    - "Wheel-Event" Problem behoben
    [18.10.2022 15:20Uhr]
    -Fix: FF106+ -> 'Drag&Drop'
    [19.11.2022 09:28Uhr]
    -Fix: FF107+ -> 'contain' entfernt
    [15.12.2022 07:52Uhr]
    -Fix: FF108+ -> Toggle-menubar angepasst
    [15.12.2022 15:50Uhr]
    -Fix: FF108+ -> Position der vertikalen Tableiste korrigiert
    [19.12.2022 19:58Uhr]
    - Seite für neuen Tab wird aus Voreinstellungen gelesen
    - kleinere optische Anpassungen
    [21.12.2022 18:32Uhr]
    - Fix: Menüleiste per 'alt'/'F10'-Key einblenden
    [10.05.2023 00:08]
    - Fix: Anpassungen Flex-Container
    [10.05.2023 19:32]
    - Fix: Anpassungen "Vertikale Tableiste"
    [07.06.2023 18:35]
    - Fix: Siehe: https://www.camp-firefox.de/forum/thema/74296-entwicklung-firefox/?postID=1227494#post1227494
    [03.08.2023 17:12]
    - Fix: Import 'Services.jsm' entfernt
    /////////////////////////////////////////////////////////////////////////////////////*/
    
    
    "use strict";
    
    MultiRowTabs();
    function MultiRowTabs() {
        
        if (!window.gBrowser){
            return;
        }
    
    // ----------------------------
    // --- User-Settings: Start ---
    // ----------------------------
    
    // Position der Tab-Leiste:
    //
    var nTabbarPosition = 2; // [1] Tab-Leiste ist oberhalb aller Symbolleisten
    // [2] Tab-Leiste ist unterhalb aller Symbolleisten, aber über dem Inhaltsbereich
    // [3] Tab-Leiste ist vertikal auf der linken Seite
    // [4] Tab-Leiste ist vertikal auf der linken Seite - Tableiste wird erst angezeigt bei Mausbewegung an den linken Rand (Autohide/Autopopup)
    // [5] Tab-Leiste ist vertikal auf der rechten Seite
    // [6] Tab-Leiste ist unterhalb des Inhaltsbereichs
    
    
    // Tab-Größenangaben
    //
    var nTabWidth = 170; // Breite der einzelnen Tabs in Pixeln
    var nTabHeight = 35; // Höhe der einzelnen Tabs in Pixeln
    var nTabMargin = 1; // Abstand zwischen den Tab-Zeilen in Pixeln
    
    
    // sonstige Einstellungen
    //
    var nTabLines = 7; // Anzahl der sichtbaren Tab-Zeilen, darüber hinaus wird gescrollt <nur bei horizontaler Ausrichtung (Position:[1],[2],[6]) - sonst keine Funktion>
    var bTabScrollbar = true; // [true] Scrollbar für Tab-Leiste anzeigen, [false] Keine Scrollbar für Tab-Leiste anzeigen (Achtung: [false] = kein Scrollen mehr bei 'drag&drop' von Tabs!)
    var bTabTooltips = true; // [true] Tab-Tooltips werden angezeigt, [false] Tab-Tooltips werden nicht angezeigt
    
    
    // Tab-Schließen-Button
    //
    var bTabCloseButton = true; // [true] Tab-Schließen-Button anzeigen, [false] Tab-Schließen-Button verbergen
    var bTabCloseButRounded = false; // [false] quadratische Darstellung, [true] abgerundete Darstellung
    var nTabCloseButTransparency= 0.85; // Transparenzwert des Tab-Schließen-Button in Prozent. Wertebereich: [0]=vollständig durchscheinend bis [1]=vollständig deckend (z.B [0.75])
    var nTabCloseIconNr = 0; // [0] Standard-Icon wird angezeigt
    // [1] rotes Icon mit weißem Kreuz wird angezeigt
    // [2] schwarzes Icon mit weißem Kreuz wird angezeigt
    var nTabCloseButSize = 18; // Höhe und Breite des Tab-Schließen-Buttons in Pixeln
    
    
    // FavIcon
    //
    var nFavIconSize = 16; // Höhe und Breite des 'FavIcons' und der Ladeanimation,[16] = Standard
    
    
    // 'Throbber'-Animation
    //
    var bNewThroberAnimation = true; // [true] Alternative 'Throbber'-Animation auswählen, [false] Standard 'Throbber'-Animation beibehalten
    
    
    // Hintergrundfarbe der Tabs (für einfarbige Darstellung müssen die RGB-Farbwerte 1 und 2 jeweils identisch sein)
    //
    var strTabSelColor1 = "240,152,0"; // RGB-Farbwert1 selektierter Tab
    var strTabSelColor2 = "240,152,0"; // RGB-Farbwert2 selektierter Tab
    var strTabNotSelColor1 = "195,157,116";// RGB-Farbwert1 nicht selektierter Tab
    var strTabNotSelColor2 = "195,157,116";// RGB-Farbwert2 nicht selektierter Tab
    var nTabTransparency = 1; // Transparenzwert des unselektierten Tab Hintergrundes. Wertebereich: [0]=vollständig durchscheinend bis [1]=vollständig deckend (z.B [0.75])
    
    
    // Schriftart und Textdarstellung der Tabs
    //
    var strTabFontName = "Segoe UI"; // Name der Schriftart
    var strTabFontColorSel = "255,255,255";// RGB-Farbwert der Schrift des selektierten Tabs
    var strTabFontColorNotSel = "255,255,255";// RGB-Farbwert der Schrift des nicht selektierten Tabs
    var nTabFontWeight = 600; // Stärke der Schrift: Wertebereich: [100] = sehr dünn bis [900] = sehr dick(bold). [500] = normal
    var nTabFontSize = 12; // Größe/Höhe der Schrift in Pixeln
    var bTabFontTextShadow = true; // [true] Text wird mit Schatteneffekt ausgegeben, [false] Text wird ohne Schatteneffekt ausgegeben. (Effekt nur bei selektierten Tabs!)
    var bMarkUnreadTab = false; // [true] Kursive Schrift für ungelesene Tabs, [false] ungelesene Tabs werden nicht hervorgehoben
    
    
    // Rahmen um einzelne Tabs
    //
    var nTabBorderWidth = 0; // Breite des Tab-Rahmen ([0] = kein sichtbarer Rahmen)
    var nTabBorderRadius = 30; // Radius für abgerundete Ecken des Tabs ([0] = rechteckig, [80] = ideal abgerundet ).
    var strTabBorderColor = "128,128,128";// RGB-Farbwert des Rahmens
    
    
    // Neuer-Tab-Button
    //
    var strNewTabButtonColor = "255,255,255";// RGB-Farbwert des '+'-Zeichens
    
    
    // Hintergrund der Tab-Leiste (für einfarbige Darstellung müssen die RGB-Farbwerte 1 und 2 jeweils identisch sein)
    //
    var strTabBarBgColor1 = "21,21,12"; // RGB-Farbwert1 für Hintergrund der Tab-Leiste
    var strTabBarBgColor2 = "62,75,84"; // RGB-Farbwert2 für Hintergrund der Tab-Leiste
    
    
    
    
    var nTabBarTransparency = 0; // Transparenzwert des Tab-Leisten Hintergrundes. Wertebereich: [0]=vollständig durchscheinend bis [1]=vollständig deckend
    var strTabBarBgImagePath = ""; // Absoluter Dateipfad zu einem gepeicherten Bild (z.B.: "D://Bilder//Firefox//Hintergrund.jpg" )
    // "D://Programme%20(Portable)//Firefox%20Portable//FireFox%20ESR%2091//Profilordner//chrome//image//NavToolbarBackground02.png"
    var bTabBarBgImageRepeat = false; // [true] Bild wird für den gesamten Bereich mehrfach nebeneinander angezeigt, [false] Bild wird nur einmal angezeigt (Position: linke/obere Ecke)
    
    
    // Einstellungen für vertikale Tab-Leiste
    //
    var nVerticalTabbarWidth = 215; // Breite der Vertikalen Tab-Leiste in Pixeln <nur bei vertikaler Ausrichtung (Position:[3],[4],[5]) - sonst keine Funktion>
    var nVerticalAutoPopupHover = 2; // Abstand zum linken Fensterrand in Pixeln, ab der die vertikale Tab-Leiste sichtbar gemacht wird <Position[4] - sonst keine Funktion>
    var nVerticalAutoPopupAnim = 0.5; // Dauer der Animation beim 'Herausschieben' des vertikalen Tab-Leiste in Sekunden ([0] = keine Animation) <Position[4] - sonst keine Funktion>
    
    
    // Einstellungen für Maus-Bedienung
    //
    var bTabWheel = false; // [true]: Tab-Wheel-Selection(=Selektieren des nächsten/vorherigen Tabs mit dem Mausrad) einschalten, [false]: Tab-Wheel-Selection ausschalten
    var bPageScroll = true; // [true]: seitenweises Scrollen, [false]: zeilenweises Scrollen
    var bDblclickOnTabbarNewTab = true; // [true] Doppel-Klick über Tabbar öffnet neuen Tab, [false] Funktion wird nicht ausgeführt
    var bDblclickOnTabReloadTab = true; // [true] Doppel-Klick über Tab lädt diesen neu, [false] Funktion wird nicht ausgeführt
    
    // ----------------------------
    // --- User-Settings: Ende ---
    // ----------------------------
    
        
        let strHomepageURL;
        try {
          strHomepageURL = Services.prefs.getCharPref('browser.startup.homepage');
          console.log("homepage: " + strHomepageURL);
        } catch(e) {
            console.log("Error Homepage-String loading...");
        }
            
             
        if( nTabbarPosition < 1 || nTabbarPosition > 6 ) {
            nTabbarPosition = 1;
        }
        
        // Tab-Leiste ganz unten 
        if( nTabbarPosition == 6 )
        {
                let tabbarBoxBottom = document.createXULElement('vbox');
                tabbarBoxBottom.id = 'tabbarboxbottom';
                tabbarBoxBottom.style.background = '#000000';
            
                document.getElementById("navigator-toolbox").parentNode.parentNode.insertBefore( tabbarBoxBottom, document.getElementById("browser-bottombox"));
                let tabbar = document.getElementById("TabsToolbar");
                tabbarBoxBottom.appendChild(tabbar);
        }
        
        //Tableiste vertikal
        if(nTabbarPosition == 3 || nTabbarPosition == 5 )
        {
                let nTabbarWidth = nVerticalTabbarWidth;
                
                let tabbarBox = document.createXULElement('vbox');
                tabbarBox.id = 'tabbar-box';
                tabbarBox.style.width = nVerticalTabbarWidth + 'px';
                tabbarBox.style.background = '#000000';
            
                // Platz für Tab-Leiste auf der linken Seite 
                if(nTabbarPosition == 3) {
                    let sidebarBox = document.getElementById('sidebar-box');
                    sidebarBox.parentNode.insertBefore(tabbarBox, sidebarBox);
                }
                // Platz für Tab-Leiste auf der rechten Seite 
                else if( nTabbarPosition == 5) {
                    document.getElementById("browser").insertBefore(tabbarBox,document.getElementById("browser").lastChild.nextSibling);
                }
            
        }
    
        tabsetting: { 
            let css =` 
                tabs tab {
                    border-left: solid 1px hsla(0,0%,0%,0) !important;
                    border-right: solid 1px hsla(0,0%,0%,0) !important;
                    z-index: 2 !important;
                }
                tabs tab:after,tabs tab:before
                { 
                    display: none !important; 
                }
        
            `;
    
            let sss = Cc['@mozilla.org/content/style-sheet-service;1'].getService(Ci.nsIStyleSheetService);
            let uri = makeURI('data:text/css;charset=UTF=8,' + encodeURIComponent(css));
            sss.loadAndRegisterSheet(uri, sss.AUTHOR_SHEET);
        }
    
            
        //var nTabLineHeight = nTabHeight+2*nTabMargin;
        nTabMargin *= 2;
        var nTabLineHeight = nTabHeight+nTabMargin;
        
        //console.log("Margin: " + nTabMargin );
        
        multiLineTabSetup: {
            
            let service,uri,cssIn,cssOut;
            
            // ---
            // CSS-Icon-Settings 
            // ---
            cssIn =`
                
         
                /* Tab-Schließen-Button auf allen Tabs anzeigen oder verbergen */
                .tabbrowser-tab:not([pinned]) .tab-close-button, 
                .tabbrowser-tab:not([pinned]) .tab-close-button:not([selected="true"]) {
                    display: __strTabCloseVisible__ !important;
                }
            
                /* Icon für Tab-Schließen-Button anzeigen */
                .tabbrowser-tab:not([pinned]) .close-icon {
                    __CloseIcon1DisableStart__ list-style-image: url("") !important; __CloseIcon1DisableEnd__
                    __CloseIcon2DisableStart__ list-style-image: url("") !important; __CloseIcon2DisableEnd__
    
                    
                    border-radius: __nBorderRadius__px !important;
                    padding: 3px !important;
                    width: __nTabCloseButSize__px !important;
                    height: __nTabCloseButSize__px !important;
                    filter: opacity(__nTabCloseButTransparency__%) drop-shadow(-1px -1px 1px rgba(0,0,0,0.15)) drop-shadow(0px -1px 1px rgba(255,255,255,0.15)) !important; 
                }
                
                /* Neuer Tab Button */
                #TabsToolbar #tabs-newtab-button > image {
                    min-height: __nNewTabButtonSize__px !important;
                    max-height: __nNewTabButtonSize__px !important;
                    min-width: __nNewTabButtonSize__px !important;
                    max-width: __nNewTabButtonSize__px !important;
                    margin-top: -1px !important;
                    margin-left: -3px !important;
                    padding: 2px !important;
                    fill: rgba(__strNewTabButtonColor__,1) !important;
                }
                
                
                /* Throbber Icon ändern */
                __NewThrowberDisableStart__
                .tab-throbber[busy]::before,
                .tab-throbber[progress]::before {
                  width: 16px !important;
                  height: 16px !important;
                  animation: unset !important;
                  -moz-context-properties: unset !important;
                  fill: unset !important;
                  opacity: unset !important;
                }
                .tab-throbber[busy]::before {
                  background-image: url("") !important;
                }
                .tab-throbber-fallback[busy] {
                  list-style-image: url("") !important;
                }
                .tab-throbber[progress]::before{
                  background-image: url("") !important;
                }
                .tab-throbber-fallback[progress] {
                  list-style-image: url("") !important;
                }
                __NewThrowberDisableEnd__
            
            `;
            
            cssOut = cssIn;
            
            if(bTabCloseButton) {
                cssOut = cssOut.replace("__strTabCloseVisible__", "initial");
            }    
            else {
                cssOut = cssOut.replace("__strTabCloseVisible__", "none");
                nTabCloseIconNr = 0;
            }
            
            if( nTabCloseIconNr == 1 ) {
                cssOut = cssOut.replace(/__CloseIcon1DisableStart__/g, ' ');
                cssOut = cssOut.replace(/__CloseIcon1DisableEnd__/g, ' ');
                cssOut = cssOut.replace(/__CloseIcon2DisableStart__/g, '/*');
                cssOut = cssOut.replace(/__CloseIcon2DisableEnd__/g, '*/');
            }        
            else if( nTabCloseIconNr == 2 ) {
                cssOut = cssOut.replace(/__CloseIcon1DisableStart__/g, '/*');
                cssOut = cssOut.replace(/__CloseIcon1DisableEnd__/g, '*/');
                cssOut = cssOut.replace(/__CloseIcon2DisableStart__/g, ' ');
                cssOut = cssOut.replace(/__CloseIcon2DisableEnd__/g, ' ');
            }
            else
            {
                cssOut = cssOut.replace(/__CloseIcon1DisableStart__/g, '/*');
                cssOut = cssOut.replace(/__CloseIcon1DisableEnd__/g, '*/');
                cssOut = cssOut.replace(/__CloseIcon2DisableStart__/g, '/*');
                cssOut = cssOut.replace(/__CloseIcon2DisableEnd__/g, '*/');
                
            }
            
            if(bTabCloseButRounded)
                cssOut = cssOut.replace(/__nBorderRadius__/g, 20);
            else
                cssOut = cssOut.replace(/__nBorderRadius__/g, 0);
            
            cssOut = cssOut.replace(/__nTabCloseButTransparency__/g, nTabCloseButTransparency*100);
            cssOut = cssOut.replace(/__nTabCloseButSize__/g, nTabCloseButSize);
            
            
            cssOut = cssOut.replace(/__nNewTabButtonSize__/g, nTabHeight);
            cssOut = cssOut.replace(/__strNewTabButtonColor__/g, strNewTabButtonColor);
            
            if(bNewThroberAnimation)
            {
                cssOut = cssOut.replace(/__NewThrowberDisableStart__/g, ' ');
                cssOut = cssOut.replace(/__NewThrowberDisableEnd__/g, ' ');
            }
            else
            {
                cssOut = cssOut.replace(/__NewThrowberDisableStart__/g, '/*');
                cssOut = cssOut.replace(/__NewThrowberDisableEnd__/g, '*/');
            }
     
            //console.log("cssOut1: " + cssOut );
            
            service = Cc['@mozilla.org/content/style-sheet-service;1'].getService(Ci.nsIStyleSheetService);
            uri = makeURI('data:text/css;charset=UTF=8,' + encodeURIComponent(cssOut));
            service.loadAndRegisterSheet(uri, service.AUTHOR_SHEET);
           
    
            // ---
            // Scrollbars -> "no-drag"
            // ---
    
            cssIn =`
              scrollbar, scrollcorner, scrollbar thumb, scrollbar scrollbarbutton {
                -moz-window-dragging: no-drag !important;
            }
            `;
            
            service = Cc['@mozilla.org/content/style-sheet-service;1'].getService(Ci.nsIStyleSheetService);
            uri = makeURI('data:text/css;charset=UTF=8,' + encodeURIComponent(cssIn));
            service.loadAndRegisterSheet(uri, service.AGENT_SHEET);
    
            
    
    
    
            cssIn =`
    
            :root[tabsintitlebar][sizemode="maximized"]  #titlebar,
            :root[tabsintitlebar][sizemode="normal"]  #titlebar {
                appearance: none !important;
            }
    
            .tabbrowser-tab {
                --tab-label-mask-size: 1em;
            }
            
            #tabbrowser-tabs[haspinnedtabs]:not([positionpinnedtabs]) > #tabbrowser-arrowscrollbox > .tabbrowser-tab:nth-child(1 of :not([pinned], [hidden])) {
                margin-inline-start: -1px !important; 
            }
            
            #tabbrowser-tab-tooltip {
                visibility: strTabTooltips !important;
            }
            
            #TabsToolbar > .titlebar-buttonbox-container,
            #main-window[sizemode="fullscreen"]  #window-controls {
                display: none !important;
            }    
                
            #TabsToolbar { 
                /* Hintergrund der Tab-Leiste setzen */
                __strTabBarBackground__
            }
            
                    
                            
            /* Tab-Höhe */
            .tabbrowser-tab, .tab-stack, .tab-background {
                min-height: nTabHeightpx !important;
                max-height: nTabHeightpx !important;
                
                margin-top: 0px !important;
                margin-bottom: nTabMarginpx !important;
                
                margin-left: -1px !important;
                margin-right: 0px !important;
            }
    
    
            #tabbrowser-tabs .tab-label:not([fokus]){
                opacity: 1!important;
            }
        
        
                
            tabs > arrowscrollbox,
            tabs > arrowscrollbox::part(scrollbox-clip)        { 
                display: block; 
                contain: none !important; 
            }
            
                  
            tabs > arrowscrollbox::part(scrollbox)    {
                display: flex !important;
                position: relative !important;
                flex-wrap: wrap !important; 
                overflow: visible !important; 
                overflow-x: hidden !important;
                overflow-y: strScrollbar !important;
            }
                    
            .tabbrowser-tab:not([pinned]) {
                flex-grow: 0 !important;
                min-width: nTabWidthpx !important;             
            }
            
        
            /* Schriftart/Schriftgröße ändern */
            .tabbrowser-tab .tab-label {
                text-shadow: none !important; 
                color: rgb(strTabFontColorNotSel) !important; 
                font-weight: nTabFontWeight !important;
                font-size:   nTabFontSizepx !important;
                font-family: "strTabFontName" !important;
                font-style: normal; 
            }
    
            .tabbrowser-tab[selected="true"] .tab-label,
            .tabbrowser-tab[multiselected="true"] .tab-label {
                text-shadow: __strTextShadow__ !important; ;
                color: rgb(strTabFontColorSel) !important; 
                font-weight: nTabFontWeight !important;
                font-size:   nTabFontSizepx !important; 
                font-family: "strTabFontName" !important; 
                font-style: normal; 
            }
            
            /*Ungelesene Tabs werden markiert */
            .tabbrowser-tab[pending] .tab-label, .tabbrowser-tab[notselectedsinceload="true"] {
                font-style: __strMarkUnreadTab__ !important;
            }
            
    
            #TabsToolbar .tabbrowser-tab .tab-background
            {
                border-radius: nTabBorderRadiuspx !important; 
                border: nTabBorderWidthpx solid rgba(strTabBorderColor,0.66) !important;
            }
    
            
            /* Tab-Hintergrund  */
            /*--- selektiert ---*/
            .tab-background[selected=true]{
                background-image:  linear-gradient( rgba(strTabSelColor1,1), rgba(strTabSelColor2,1) ), none !important;
                filter: drop-shadow(2px 2px 2px rgba(0,0,0,0.5)) drop-shadow(-2px 2px 2px rgba(0,0,0,0.5)) !important; 
                border-radius: nTabBorderRadiuspx !important; 
            }
            /*--- hover: nicht selektiert ---*/
            .tabbrowser-tab:hover > .tab-stack > .tab-background:not([selected=true]) {
                filter: brightness(130%) contrast(110%) drop-shadow(1px 1px 1px rgba(0,0,0,0.5)) drop-shadow(-1px 1px 1px rgba(0,0,0,0.5)) !important; 
                border-radius: nTabBorderRadiuspx !important;    
            }
            .tabbrowser-tab:hover:not([selected=true]) :is(.tab-label,.tab-icon-stack) {
                filter: drop-shadow( 0px 0px 2px #AAAAAA) brightness(115%) !important; 
            }
            /*--- hover: Multi selektiert ---*/
            .tabbrowser-tab:hover > .tab-stack > .tab-background[multiselected=true]:not([selected=true]) {
                filter: brightness(120%) contrast(100%) drop-shadow(1px 1px 1px rgba(0,0,0,0.5)) drop-shadow(-1px 1px 1px rgba(0,0,0,0.5)) !important; 
                border-radius: nTabBorderRadiuspx !important;    
            }
            /*--- nicht selektiert ---*/
            .tab-background:not([selected=true]) {
                background-image:  linear-gradient( rgba(strTabNotSelColor1,nTabTransparency), rgba(strTabNotSelColor2,nTabTransparency) ),  none !important; 
                filter: brightness(115%) contrast(90%) drop-shadow(1px 1px 1px rgba(0,0,0,0.5)) drop-shadow(-1px 1px 1px rgba(0,0,0,0.5)) !important; 
                border-radius: nTabBorderRadiuspx !important; 
            }
            /*--- Multi selektiert ---*/
            .tab-background[multiselected=true]:not([selected=true]) {
                background-image:  linear-gradient( rgba(strTabSelColor1,1), rgba(strTabSelColor2,1) ), none !important;
                filter: brightness(130%) saturate(50%) drop-shadow(1px 1px 1px rgba(0,0,0,0.5)) drop-shadow(-1px 1px 1px rgba(0,0,0,0.5)) !important; 
                border-radius: nTabBorderRadiuspx !important; 
            }
        
            .tab-sharing-icon-overlay, .tab-icon-overlay:not([pinned]), .tab-icon-overlay[soundplaying] {
                transform:  scale(1.5) !important;  
            }
            
            .tabbrowser-tab .tab-label 
            {
                margin-top: -2px !important;
            }
                
            
            .tab-close-button
            {
                margin-top: -2px !important;
            }
            
            .tab-throbber, .tab-throbber-tabslist, .tab-icon-pending, .tab-icon-image, .tab-sharing-icon-overlay {
                height:     nFavIconSizepx !important;
                width:         nFavIconSizepx !important;
                fill: rgba(255,255,255,0.8) !important;
                margin-top: -2px !important;
            }    
            
            #tabbrowser-tabs, #tabbrowser-arrowscrollbox, #tabbrowser-tabs[positionpinnedtabs] > #tabbrowser-arrowscrollbox > .tabbrowser-tab[pinned] {
                min-height: 0 !important; 
            }        
    
            #tabbrowser-tabs {
                appearance: none !important;
            }
            
            .tabbrowser-tab:is([visuallyselected="true"], [multiselected]) > .tab-stack > .tab-background:-moz-lwtheme {
                box-shadow: none !important; 
            }
     
            
            `;
    
                    
            // --- CSS-Parameter ersetzen ---
            cssOut = cssIn.replace(/nTabHeight/g, nTabHeight);
            cssOut = cssOut.replace(/nTabMargin/g, nTabMargin);
            cssOut = cssOut.replace(/nTabWidth/g, nTabWidth);
            
                    
            if(bTabScrollbar) {
                cssOut = cssOut.replace("strScrollbar", "auto");
            }
            else {
                cssOut = cssOut.replace("strScrollbar", "hidden");
            }
            
            if(bTabTooltips) 
                cssOut = cssOut.replace("strTabTooltips", "visible");
            else
                cssOut = cssOut.replace("strTabTooltips", "hidden");
                    
            cssOut = cssOut.replace(/strTabSelColor1/g, strTabSelColor1);
            cssOut = cssOut.replace(/strTabSelColor2/g, strTabSelColor2);
            cssOut = cssOut.replace(/strTabNotSelColor1/g, strTabNotSelColor1);
            cssOut = cssOut.replace(/strTabNotSelColor2/g, strTabNotSelColor2);
            cssOut = cssOut.replace(/nTabBorderRadius/g, nTabBorderRadius);
            cssOut = cssOut.replace(/nTabBorderWidth/g, nTabBorderWidth);
            cssOut = cssOut.replace(/strTabBorderColor/g, strTabBorderColor);
            cssOut = cssOut.replace(/nTabTransparency/g, nTabTransparency);
            
            cssOut = cssOut.replace(/strTabFontName/g, strTabFontName);
            cssOut = cssOut.replace(/strTabFontColorSel/g, strTabFontColorSel);
            cssOut = cssOut.replace(/strTabFontColorNotSel/g, strTabFontColorNotSel);
            cssOut = cssOut.replace(/nTabFontSize/g, nTabFontSize);
            cssOut = cssOut.replace(/nTabFontWeight/g, nTabFontWeight);
            
            if (bTabFontTextShadow)
                cssOut = cssOut.replace(/__strTextShadow__/g, "1px 1px 0px #000000");
            else
                cssOut = cssOut.replace(/__strTextShadow__/g, "none");
            
            
            if( bMarkUnreadTab )
              cssOut = cssOut.replace(/__strMarkUnreadTab__/g, "italic");
            else
              cssOut = cssOut.replace(/__strMarkUnreadTab__/g, "normal");
            
            
                
            let strTabBarBackgroundOut = "";
            if( strTabBarBgImagePath ) {
                if( bTabBarBgImageRepeat ) {
                    strTabBarBackgroundOut = "background: #000000 url(\"file:" + strTabBarBgImagePath + "\") repeat !important;";
                }
                else {
                    strTabBarBackgroundOut = "background: #000000 url(\"file:" + strTabBarBgImagePath + "\") no-repeat !important;";
                }
            }
            else {
                 strTabBarBackgroundOut = "background-image: linear-gradient( rgba(" + strTabBarBgColor1 + "," + nTabBarTransparency +"), rgba(" + strTabBarBgColor2 + "," + nTabBarTransparency + ") ) !important;";
            }
                            
            cssOut = cssOut.replace(/__strTabBarBackground__/g, strTabBarBackgroundOut);
            
            cssOut = cssOut.replace(/nFavIconSize/g, nFavIconSize);
                            
            //console.log("CSSOut: " + cssOut );
    
         
            service = Cc['@mozilla.org/content/style-sheet-service;1'].getService(Ci.nsIStyleSheetService);
            uri = makeURI('data:text/css;charset=UTF=8,' + encodeURIComponent(cssOut));
            service.loadAndRegisterSheet(uri, service.AUTHOR_SHEET);
                    
            let tabsScrollbox     = document.getElementById('tabbrowser-arrowscrollbox').scrollbox;
            
            tabsScrollbox.style.minHeight      =  nTabLineHeight +'px';
            tabsScrollbox.style.maxHeight      =  nTabLineHeight*nTabLines +'px';
            
            //console.log("tabsScrollbox: " + tabsScrollbox );
            
        }
    
        
        if( nTabbarPosition >= 2 && nTabbarPosition <= 5 ) {
            let tabbar = document.getElementById("TabsToolbar");
            tabbar.parentNode.parentNode.appendChild(tabbar);
        }
    
        
        if( nTabbarPosition >= 3 && nTabbarPosition <= 5 )
        {
            let cssIn =`
            
            .StyleShowingTabsToolbar {
                transition:                margin-left nVerticalAutoPopupAnims !important; 
                display:                   flex !important;
                position:                  absolute !important;
                opacity:                   1;
                margin-left:               0px;
                z-index:                   100 !important;
                min-width:                 nVerticalTabbarWidthpx !important;
                filter:                    drop-shadow(4px 3px 2px rgba(0,0,0,0.33)) !important; 
            }
            
                    
            .StyleHidingTabsToolbar {
                display:                   flex !important;
                position:                  absolute !important;
                opacity:                   0;
                margin-left:               calc( (nVerticalTabbarWidthpx - nVerticalAutoPopupHoverpx) * -1 ); 
                z-index:                   100 !important;
                min-width:                 nVerticalTabbarWidthpx !important;
            }
      
            .toolbar-items[align="end"] {
                display:initial !important;
            }
      
            tabs > arrowscrollbox { 
                display:  flex !important;
                position: absolute !important;
                min-width: nVerticalTabbarWidthpx !important;
            }
        
            `;
        
            let cssOut;
            
            if ( nTabbarPosition != 4 ) {
                nVerticalAutoPopupAnim = 0;
            }        
            
                            
            cssOut = cssIn.replace(/nVerticalTabbarWidth/g, nVerticalTabbarWidth);
            cssOut = cssOut.replace(/nVerticalAutoPopupHover/g, nVerticalAutoPopupHover);
            cssOut = cssOut.replace(/nVerticalAutoPopupAnim/g, nVerticalAutoPopupAnim);
            
                            
            let sss = Cc['@mozilla.org/content/style-sheet-service;1'].getService(Ci.nsIStyleSheetService);
            let uri = makeURI('data:text/css;charset=UTF=8,' + encodeURIComponent(cssOut));
            sss.loadAndRegisterSheet(uri, sss.AUTHOR_SHEET);
        
    
            
            let tabsToolbar =  document.getElementById('TabsToolbar');
            
            if ( nTabbarPosition == 4 )
            {
                tabsToolbar.classList.add('StyleHidingTabsToolbar');
            }
            else
            {
                tabsToolbar.classList.add('StyleShowingTabsToolbar');
            }
            
        }
    
    //------------------------------------------------------------------------------------------
        
        var tabsToolbar     = document.getElementById('TabsToolbar');
        var tabbrowsertabs     = document.getElementById('tabbrowser-tabs');
        var tabsscrollbox    = document.getElementById('tabbrowser-arrowscrollbox');
        var ScrollBox = tabsscrollbox.scrollbox;    
    
        var bMouseEnter = false;
        var bToolbarLocked = false;
    
        // ----------------------------------
        // Load-Event:  
        // ----------------------------------
        function onReady() 
        {    
            console.log("OnReady");
            
            let cssElements =`
                #alltabs-button,
                hbox.titlebar-spacer/*,
                [class="scrollbutton-up"],
                [class="scrollbutton-up"] + spacer,
                scrollbox[part][orient="horizontal"] + spacer,
                [class="scrollbutton-down"]*/ { 
                    display: none !important; 
                }
                
                #tabbrowser-arrowscrollbox::part(scrollbutton-up), 
                #tabbrowser-arrowscrollbox::part(scrollbutton-down) {
                    display: none !important;
                }
                
                
            `;
            
            let service = Cc['@mozilla.org/content/style-sheet-service;1'].getService(Ci.nsIStyleSheetService);
            let ur         = makeURI('data:text/css;charset=UTF=8,' + encodeURIComponent(cssElements));
            service.loadAndRegisterSheet(ur, service.AUTHOR_SHEET);
        
            gBrowser.selectedTab.scrollIntoView(); 
        }    
        
        
        if (window.readyState !== "loading") {
            setTimeout(onReady,350); 
        }
        else 
        {
            window.addEventListener("DOMContentLoaded", onReady );
        }
        
        // ----------------------------------
        // ResizeObserver / Client
        // ----------------------------------
        var resizeClientObserver=null;
        
        let navigatorToolbox = document.getElementById('navigator-toolbox');
        let mainWindow = document.getElementById('main-window');
        
        if( nTabbarPosition >= 3 && nTabbarPosition <= 5 )
        {
            resizeClientObserver = new ResizeObserver(function(entries) {
                for (let entry of entries) {
                    
                    tabsToolbar.style.minHeight = tabsToolbar.style.maxHeight = entry.contentRect.height + 'px';
                    ScrollBox.style.maxHeight = entry.contentRect.height + 'px';
                    
                    tabsToolbar.style.top = navigatorToolbox.clientHeight + 'px';
                                    
                    if(mainWindow.getAttribute('inFullscreen') == 'true' ) {
                        if (navigatorToolbox.style.marginTop != "") {
                            tabsToolbar.style.top = '0px';
                        }
                    }
                    
                    if( nTabbarPosition == 5 )    {
                        tabsToolbar.style.marginLeft = (entry.contentRect.width - nVerticalTabbarWidth) + 'px';
                    }
                }
            });
                    
            resizeClientObserver.observe(document.getElementById("browser"));            
        }  
    
    
        // ----------------------------------
        // ToggleMenuObserver
        // ----------------------------------
        if( nTabbarPosition == 2 || nTabbarPosition == 6) {
            let observerToggleMenu=null;
            let configObserver=null;
            
            let bTabsintitlebar = document.querySelector('html#main-window').getAttribute('tabsintitlebar');
            let titlebar = document.querySelector('#navigator-toolbox>vbox#titlebar');
            let targetMenubar = document.getElementById('toolbar-menubar');
    
            const callback = (mutationList, observer) => {
                for (const mutation of mutationList) {
                    if (mutation.type === 'attributes') {
                        let bAutohide = targetMenubar.getAttribute('autohide');
                        let bInacive = targetMenubar.getAttribute('inactive');
                        if(bInactive == 'true')    {
                            titlebar.style.display = "none";
                            console.log("titlebar.style.display = none");
                        }
                        else {
                            titlebar.style.display = "initial";
                            console.log("titlebar.style.display = initial");
                            
                        }
                    }
                }
            }
        
            observerToggleMenu = new MutationObserver(callback);
            configObserver = { attributes: true, attributeFilter: ['autohide','inactive'] };
            
            observerToggleMenu.observe(targetMenubar, configObserver);    
        }
    
        // ----------------------------------
        // 'Middleclick' on Tab -> Close Tab
        // ----------------------------------
        gBrowser.tabContainer.addEventListener('click', function(event) 
        {
            if (event.button == 1)
            {
                let element = event.target.parentNode;
                while(element)
                {
                    if (element.localName == 'tab') 
                    {
                        gBrowser.removeTab(element, {animate: false});
                        event.preventDefault();
                        event.stopPropagation();    
                        return;
                    }
                    
                    element = element.parentNode;
                }
                
            };
        
        }, true);
    
        // ----------------------------------
        // 'Middleclick' on TabsToolbar -> Add Tab
        // ----------------------------------
        tabsToolbar.addEventListener('click', function(event)
        {
            if (event.button == 1)
            {
                if(event.target.parentNode.id == "TabsToolbar") 
                { 
                   event.target.ownerGlobal.openTrustedLinkIn(strHomepageURL,"tab");
                   return;
                }
            };
        }, true);
    
        
        // ----------------------------------
        // 'Doubleclick' on TabsToolbar -> Add Tab
        // ----------------------------------
        tabsToolbar.addEventListener('dblclick', function(event)
        {
            if(!bDblclickOnTabbarNewTab)
                return;
            
            if (event.button == 0)
            {  
                    
                if(event.target.parentNode.id == "TabsToolbar") 
                {
                   event.target.ownerGlobal.openTrustedLinkIn(strHomepageURL,"tab");
                   
                   event.preventDefault();
                   event.stopPropagation();    
                   return;
                }
            };
        }, true);
            
            
            
        // ----------------------------------
        // 'Doubleclick' on TabsContainer -> Add Tab
        // ----------------------------------
        gBrowser.tabContainer.addEventListener('dblclick', function(event)
        {
            if(!bDblclickOnTabbarNewTab)
                return;
            
            if (event.button == 0)
            {  
                   let element = event.target.parentNode;
                if (element == gBrowser.tabContainer ) 
                {
                   event.target.ownerGlobal.openTrustedLinkIn(strHomepageURL,"tab");
        
                   event.preventDefault();
                   event.stopPropagation();    
                   return;
                }
            };
        }, true);
        
        
        // ----------------------------------
        // 'Doubleclick' on Tab -> Reload Tab
        // ----------------------------------
        gBrowser.tabContainer.addEventListener('dblclick', function(event) 
        {
            if(!bDblclickOnTabReloadTab)
                return;
            
            if (event.button == 0)
            {
                let element = event.target.parentNode;
                while (element) 
                {
                    if (element.localName == 'tab') 
                    {
                        element.linkedBrowser.reload();
                        
                        return;
                    }
                     
                    element = element.parentNode;
                }
            }
        }, true);
        
        // ----------------------------------
        // 'mouseenter'
        // ----------------------------------
        tabsToolbar.addEventListener('mouseenter', event => {
            if( nTabbarPosition == 4  ) 
            {
                if(!bMouseEnter)
                {
                    bMouseEnter = true;
                    
                    tabsToolbar.classList.add('StyleShowingTabsToolbar');    
                    tabsToolbar.classList.remove('StyleHidingTabsToolbar');
                    
                }
            }
        }, true);
        
        // ----------------------------------
        // 'mouseleave'
        // ----------------------------------
        tabsToolbar.addEventListener('mouseleave', event => {
            if( nTabbarPosition == 4 ) 
            {    
                if(bMouseEnter)
                {      
                                
                    if( event.clientX >= 0 &&
                        event.offsetY > tabsToolbar.clientTop  &&
                        event.offsetX < tabsToolbar.clientWidth &&
                        event.offsetY < tabsToolbar.clientHeight )
                    {
                        return;
                    }
            
                    event.preventDefault();
                    event.stopPropagation();
                    
                    
                    tabsToolbar.classList.add('StyleHidingTabsToolbar');
                    tabsToolbar.classList.remove('StyleShowingTabsToolbar');
                    
    
                    bMouseEnter = false;
                }
            }                
        }, false);
    
    
        
        // ...  
        // TabSelect-Event:  
        // ...
        gBrowser.tabContainer.addEventListener("TabSelect", function(event) {
                    
            let bScroll    = false;
            let bScrollTop = true;
                    
            let scrollBoxY1     = event.target.parentElement.scrollbox.screenY;
            let scrollBoxHeight = event.target.parentElement.scrollbox.clientHeight;
            let scrollBoxY2     = scrollBoxY1+scrollBoxHeight;
            
            let TabSelY1     = event.target.screenY;
            let TabSelHeight = event.target.clientHeight;
            let TabSelY2     = TabSelY1+TabSelHeight;
            
            
            
            if( TabSelY2 > scrollBoxY2 )
            {
                bScroll = true;
                bScrollTop = false;
            }
            if( TabSelY1 < scrollBoxY1 ) 
            {
                bScroll = true;
                bScrollTop = true;
            }
            
            if( bScroll )
            {
                setTimeout(function() 
                {
                    event.target.scrollIntoView(bScrollTop);
                    //gBrowser.selectedTab.scrollIntoView(); 
                    //console.log("TabSelect_IntoView:" + bScrollTop );    
                }, 0);
            }
            
           //console.log("TabSelect:" + bScroll );    
            
        }, true);
          
        
          
          
        // ...  
        // Drag-Event: Start  
        // ...
        gBrowser.tabContainer.addEventListener("dragstart",  function(event) {
            //console.log("dragstart..." );    
            if(nTabLines==1 && ( nTabbarPosition == 1 || nTabbarPosition == 2 || nTabbarPosition == 6 )) {
                let tabsScrollbox  = document.getElementById('tabbrowser-arrowscrollbox').scrollbox;
                tabsScrollbox.style.maxHeight      =  nTabLineHeight*3 +'px';
                event.target.scrollIntoView(false);
            }
        }, false);
    
        // ...  
        // Drag-Event: Ende  
        // ...
        gBrowser.tabContainer.addEventListener("dragend",  function(event) {
            //console.log("dragend..." );    
            if(nTabLines==1 && ( nTabbarPosition == 1 || nTabbarPosition == 2 || nTabbarPosition == 6 )) {
                let tabsScrollbox  = document.getElementById('tabbrowser-arrowscrollbox').scrollbox;
                tabsScrollbox.style.maxHeight      =  nTabLineHeight +'px';
                event.target.scrollIntoView(false);
            }
        }, false);
        
        
        // ...  
        // Wheel-Event:  
        // ...
        let tabsScrollbox  = document.getElementById('tabbrowser-arrowscrollbox').scrollbox;
        tabsScrollbox.addEventListener("wheel", function(event) 
        {
            event.preventDefault();
            event.stopPropagation();
            
                
            let scrollUp        = true;
            let wrap            = false;
            let scrollBoxY1     = gBrowser.tabContainer._animateElement.scrollbox.scrollTop; 
            let scrollHeight;
            
            if(bPageScroll)
                {scrollHeight  = gBrowser.tabContainer._animateElement.scrollbox.clientHeight;}
            else            
                {scrollHeight  = nTabLineHeight;}
            
            let dir = (scrollUp ? 1 : -1) * Math.sign(event.deltaY);
            
            let bLastScrollLine = false; 
            
            if( gBrowser.tabContainer._animateElement.scrollbox.scrollTopMax == gBrowser.tabContainer._animateElement.scrollbox.scrollTop)
                {
                    bLastScrollLine = true;
                }        
            
    
            if( !bTabWheel || event.ctrlKey || event.originalTarget.localName == "thumb"  || event.originalTarget.localName == "slider" || event.originalTarget.localName == "scrollbarbutton" ) 
            {
                  setTimeout(function()
                {
                    let scrollBoxMod = scrollBoxY1%scrollHeight;
                    
                    if( scrollBoxMod > 0 )
                    {
                        if( dir == -1 )
                        {
                            scrollBoxY1 -= scrollBoxMod;
                            if(scrollBoxMod < nTabLineHeight && !bLastScrollLine) { scrollBoxY1 -= scrollHeight; }
                    
                        }
                        else
                        {
                            scrollBoxY1 += (scrollHeight-(scrollBoxMod));
                            if(scrollHeight-(scrollBoxMod) < nTabLineHeight) { scrollBoxY1 += scrollHeight; }
                            
                        }
                    }
                    else
                    {
                        if( dir == -1 )
                        {
                            scrollBoxY1 -= scrollHeight;
                        }
                        else
                        {
                            scrollBoxY1 += scrollHeight;
                        }
                    }
                    
                    gBrowser.tabContainer._animateElement.scrollbox.scrollTo({ top: scrollBoxY1, left: 0, behavior: 'auto'}); 
                }, 20);
            }
            
    
            if( bTabWheel && !event.ctrlKey )
            {    
                setTimeout(function()
                {
                    if( event.originalTarget.localName != "slider" && 
                        event.originalTarget.localName != "thumb"  &&
                        event.originalTarget.localName != "scrollbarbutton" )
                    {
                        gBrowser.tabContainer.advanceSelectedTab(dir, wrap);
                        if( (gBrowser.tabContainer._firstTab == gBrowser.selectedTab) || 
                            (gBrowser.tabContainer._lastTab  == gBrowser.selectedTab)     )
                            {
                                gBrowser.selectedTab.scrollIntoView(); 
                            }
            
                    }    
                }, 50);
            }
    
    
        }, false);
            
    
    //-------------------------------------------------------------------------------------------
     
        gBrowser.tabContainer.clearDropIndicator = function() {
            var tabs = this.allTabs;
            for (let i = 0, len = tabs.length; i < len; i++) {
                tabs[i].style.removeProperty("border-left-color");
                tabs[i].style.removeProperty("border-right-color");
            }
        }
        gBrowser.tabContainer.addEventListener("dragleave", function(event) {  this.clearDropIndicator(event); }, true);
    
    
    
        gBrowser.tabContainer.on_dragover = function(event) {
            this.clearDropIndicator();
            var effects = this.getDropEffectForTabDrag(event);
            var ind = this._tabDropIndicator;
            if (effects == "" || effects == "none") {
                ind.hidden = true;
                return;
            }
            event.preventDefault();
            event.stopPropagation();
            if (effects == "link") {
                let tab = this._getDragTargetTab(event, true);
                if (tab) {
                    if (!this._dragTime) {
                        this._dragTime = Date.now();
                    }
                    if (Date.now() >= this._dragTime + this._dragOverDelay) {
                        this.selectedItem = tab;
                    }
                    ind.hidden = true;
                    return;
                }
            }
            let newIndex = this._getDropIndex(event, effects == "link");
            let children = this.allTabs;
            if (newIndex == children.length) {
                children[newIndex - 1].style.setProperty("border-right-color","red","important");
            } else {
                children[newIndex].style.setProperty("border-left-color","red","important");
            }
        }
    
        gBrowser.tabContainer.on_drop = function(event) {
            this.clearDropIndicator();
            var dt = event.dataTransfer;
            var dropEffect = dt.dropEffect;
            var draggedTab;
            let movingTabs;
            if (dt.mozTypesAt(0)[0] == TAB_DROP_TYPE) {
                // tab copy or move
                draggedTab = dt.mozGetDataAt(TAB_DROP_TYPE, 0);
                // not our drop then
                if (!draggedTab) {
                    return;
                }
                movingTabs = draggedTab._dragData.movingTabs;
                draggedTab.container._finishGroupSelectedTabs(draggedTab);
            }
            this._tabDropIndicator.hidden = true;
            event.stopPropagation();
            if (draggedTab && dropEffect == "copy") {
                // copy the dropped tab (wherever it's from)
                let newIndex = this._getDropIndex(event, false);
                let draggedTabCopy;
                for (let tab of movingTabs) {
                    let newTab = gBrowser.duplicateTab(tab);
                    gBrowser.moveTabTo(newTab, newIndex++);
                    if (tab == draggedTab) {
                        draggedTabCopy = newTab;
                    }
                }
                if (draggedTab.container != this || event.shiftKey) {
                    this.selectedItem = draggedTabCopy;
                }
            } else if (draggedTab && draggedTab.container == this) {
                let oldTranslateX = Math.round(draggedTab._dragData.translateX);
                let tabWidth = Math.round(draggedTab._dragData.tabWidth);
                let translateOffset = oldTranslateX % tabWidth;
                let newTranslateX = oldTranslateX - translateOffset;
                if (oldTranslateX > 0 && translateOffset > tabWidth / 2) {
                    newTranslateX += tabWidth;
                } else if (oldTranslateX < 0 && -translateOffset > tabWidth / 2) {
                    newTranslateX -= tabWidth;
                }
                let dropIndex = this._getDropIndex(event, false);
                //  "animDropIndex" in draggedTab._dragData &&
                //  draggedTab._dragData.animDropIndex;
                let incrementDropIndex = true;
                if (dropIndex && dropIndex > movingTabs[0]._tPos) {
                    dropIndex--;
                    incrementDropIndex = false;
                }
                let animate = gBrowser.animationsEnabled;
                if (oldTranslateX && oldTranslateX != newTranslateX && animate) {
                    for (let tab of movingTabs) {
                        tab.setAttribute("tabdrop-samewindow", "true");
                        tab.style.transform = "translateX(" + newTranslateX + "px)";
                        let onTransitionEnd = transitionendEvent => {
                            if (
                                transitionendEvent.propertyName != "transform" ||
                                transitionendEvent.originalTarget != tab
                            ) {
                                return;
                            }
                            tab.removeEventListener("transitionend", onTransitionEnd);
                            tab.removeAttribute("tabdrop-samewindow");
                            this._finishAnimateTabMove();
                            if (dropIndex !== false) {
                                gBrowser.moveTabTo(tab, dropIndex);
                                if (incrementDropIndex) {
                                    dropIndex++;
                                }
                            }
                            gBrowser.syncThrobberAnimations(tab);
                        };
                        tab.addEventListener("transitionend", onTransitionEnd);
                    }
                } else {
                    this._finishAnimateTabMove();
                    if (dropIndex !== false) {
                        for (let tab of movingTabs) {
                            gBrowser.moveTabTo(tab, dropIndex);
                            if (incrementDropIndex) {
                                dropIndex++;
                            }
                        }
                    }
                }
            } else if (draggedTab) {
                let newIndex = this._getDropIndex(event, false);
                let newTabs = [];
                for (let tab of movingTabs) {
                    let newTab = gBrowser.adoptTab(tab, newIndex++, tab == draggedTab);
                    newTabs.push(newTab);
                }
                // Restore tab selection
                gBrowser.addRangeToMultiSelectedTabs(
                    newTabs[0],
                    newTabs[newTabs.length - 1]
                );
            } else {
                // Pass true to disallow dropping javascript: or data: urls
                let links;
                try {
                    links = browserDragAndDrop.dropLinks(event, true);
                } catch (ex) {}
                if (!links || links.length === 0) {
                    return;
                }
                let inBackground = Services.prefs.getBoolPref(
                    "browser.tabs.loadInBackground"
                );
                if (event.shiftKey) {
                    inBackground = !inBackground;
                }
                let targetTab = this._getDragTargetTab(event, true);
                let userContextId = this.selectedItem.getAttribute("usercontextid");
                let replace = !!targetTab;
                let newIndex = this._getDropIndex(event, true);
                let urls = links.map(link => link.url);
                let csp = browserDragAndDrop.getCSP(event);
                let triggeringPrincipal = browserDragAndDrop.getTriggeringPrincipal(
                    event
                );
                (async () => {
                    if (
                        urls.length >=
                        Services.prefs.getIntPref("browser.tabs.maxOpenBeforeWarn")
                    ) {
                        // Sync dialog cannot be used inside drop event handler.
                        let answer = await OpenInTabsUtils.promiseConfirmOpenInTabs(
                            urls.length,
                            window
                        );
                        if (!answer) {
                            return;
                        }
                    }
                    gBrowser.loadTabs(urls, {
                        inBackground,
                        replace,
                        allowThirdPartyFixup: true,
                        targetTab,
                        newIndex,
                        userContextId,
                        triggeringPrincipal,
                        csp,
                    });
                })();
            }
            if (draggedTab) {
                delete draggedTab._dragData;
            }
        }
    
        gBrowser.tabContainer._getDropIndex = function(event, isLink) {
            var tabs = this.allTabs;
            var tab = this._getDragTargetTab(event, isLink);
            if (!RTL_UI) {
                for (let i = tab ? tab._tPos : 0; i < tabs.length; i++) {
                    if (
                        event.screenY <
                        tabs[i].screenY + tabs[i].getBoundingClientRect().height
                    ) {
                        if (
                            event.screenX <
                            tabs[i].screenX + tabs[i].getBoundingClientRect().width / 2
                        ) {
                            return i;
                        }
                        if (
                            event.screenX >
                            tabs[i].screenX + tabs[i].getBoundingClientRect().width / 2 &&
                            event.screenX <
                            tabs[i].screenX + tabs[i].getBoundingClientRect().width
                        ) {
                            return i + 1;
                        }
                    }
                }
            } else {
                for (let i = tab ? tab._tPos : 0; i < tabs.length; i++) {
                    if (
                        event.screenY <
                        tabs[i].screenY + tabs[i].getBoundingClientRect().height
                    ) {
                        if (
                            event.screenX <
                            tabs[i].screenX + tabs[i].getBoundingClientRect().width &&
                            event.screenX >
                            tabs[i].screenX + tabs[i].getBoundingClientRect().width / 2
                        ) {
                            return i;
                        }
                        if (
                            event.screenX <
                            tabs[i].screenX + tabs[i].getBoundingClientRect().width / 2
                        ) {
                            return i + 1;
                        }
                    }
                }
            }
            return tabs.length;
        }
    }
    Alles anzeigen

    ---

    Zitat von GermanFreme82

    Der RESTART Button ist ebenfalls verschwunden aus dem Menü, den ich per Script eingefügt habe.

    In deiner Version ist wahrscheinlich noch die 'Services.jsm' drin, nimm dieses Skript hier:

    JavaScript
    // Restart button script for Firefox 60+ by Aris
    //
    // left-click on restart button: normal restart
    // middle-click on restart button: restart + clear caches
    // right-click on restart button: restart + clear caches
    // Edit by 2002Andreas
    // based on 'Quit' button code by 2002Andreas
    // restart code from Classic Theme Restorer add-on
    // invalidate caches from Session Saver add-on
    
    (function() {
    
    try {
      Components.utils.import("resource:///modules/CustomizableUI.jsm");
      var sss = Components.classes["@mozilla.org/content/style-sheet-service;1"].getService(Components.interfaces.nsIStyleSheetService);
      
      var button_label = "Restart";
        
        try {
          switch (document.getElementById("nav-bar").getAttribute("aria-label")) {
            case "Navigations-Symbolleiste": button_label = "Neustarten"; break;
            case "Панель навигации": button_label = "Перезапустить"; break;
          }
        } catch(e) {}
      
      CustomizableUI.createWidget({
        id: "restart-button-ucjs", // button id
        defaultArea: CustomizableUI.AREA_NAVBAR,
        removable: true,
        label: button_label, // button title
        tooltiptext: button_label, // tooltip title
        onClick: function(event) {
          
          var cancelQuit   = Components.classes["@mozilla.org/supports-PRBool;1"].createInstance(Components.interfaces.nsISupportsPRBool);
          var observerSvc  = Components.classes["@mozilla.org/observer-service;1"].getService(Components.interfaces.nsIObserverService);
                
          if(event.button=='1') { // middle-click - clear caches
            Components.classes["@mozilla.org/xre/app-info;1"].getService(Components.interfaces.nsIXULRuntime).invalidateCachesOnRestart();
          }
          
          if(event.button=='0' || event.button=='2') { // left-click - restart/right-click - clear caches
            observerSvc.notifyObservers(cancelQuit, "quit-application-requested", "restart");
                
            if(cancelQuit.data) return false;
                    
            Services.startup.quit(Services.startup.eRestart | Services.startup.eAttemptQuit);
          }
        },
        onCreated: function(button) {
          return button;
        }
            
      });
      
      // style button icon
      var uri = Services.io.newURI("data:text/css;charset=utf-8," + encodeURIComponent('\
        \
          #restart-button-ucjs .toolbarbutton-icon {\
            list-style-image: url("");  \
              }\
        \
      '), null, null);
      
      sss.loadAndRegisterSheet(uri, sss.AGENT_SHEET);
      
    } catch (e) {
        Components.utils.reportError(e);
    };
    
    })();
    Alles anzeigen
  • userChrome.js Scripte für den Fuchs (Diskussion)

    • BrokenHeart
    • 29. August 2023 um 23:37
    Zitat von mkpcxxl

    Hallo, seit dem update auf Version 117 funktioniert meine Tooltips.us.js nicht mehr.

    Dein Skript funktioniert hier, vielleicht einfach mal den Skript-Cache löschen (entweder über Starten/Schließen des Fehlerbehebungsmodus oder über 'about:support' -> Start-Cache löschen ).

  • Vorschau auf Erkennung unglaubwürdiger Produktbewertungen in Firefox

    • BrokenHeart
    • 26. August 2023 um 23:48
    Zitat von .DeJaVu

    Wird das schon in der ersten 119 Nightly (ab Dienstag) verfügbar sein?

    Anschauen kann man sich das jetzt schon, gibt es ja seit einiger Zeit als Erweiterung: Fakespot. Bewertungen sind eher mau. Hatte es mir auch kurz installiert, aber dann ziemlich bald wieder gelöscht. Kann sein, dass die FireFox-Integration da mehr an Informationen und Features liefert...


    Zum Artikel:

    In dem Beispiel bzw. den Screenshots finde ich den Link des "Trusted Deals" etwas unpassend: eine (richtige) Universal-Fernbedienung als Alternative für ein Baby-"Kau"-Spielzeug? Schlecht für das Baby und die Universal-Fernbedienung... :/

  • Button ändern bei einem Skript

    • BrokenHeart
    • 24. August 2023 um 20:08
    Zitat von Mitleser

    Freiheit für den Dateipfad 🤡️ .

    Was ist an einer Konvention "unfrei"? :/


    Zitat von milupo

    Aber nun gut. Ich nehme alles zurück und behaupte das Gegenteil.

    Nein, du hast -vorausgesetzt ich habe dich richtig verstanden- vollkommen recht mit deinem Vorschlag!

  • Button ändern bei einem Skript

    • BrokenHeart
    • 24. August 2023 um 19:55
    Zitat von milupo

    Ich denke, es sollte eine Konvention geben.

    Genau das hatte ich mir auch schon überlegt :thumbup:. Das Base64-Format hat ja nur wirklich den einzigen Vorteil, dass das Skript dann sofort "Ready-To-Use" ist, aber ist eben auch mit etlichen Nachteilen behaftet: Performance, Unübersichtlichkeit, etc. .

    Das Ermitteln des aktuellen Profilpfades und die Angabe eines Standard-Verzeichnisses für Icons/Images im 'chrome'-Pfad wären sicher die beste Lösung (z.B. 'icons' oder 'images'), so wie Mira_Belle es in #11 beschrieben hat. Man könnte dann dieses Standard-Verzeichnis auch schon in die Zip-Datei für das Einrichten der User-Skripte mitaufnehmen. Also ein Bilder-Verzeichnis, auf dessen Vorhandensein man sich verlassen kann. Ergänzend könnte man in dem Verzeichnis für die Bilder auch schon ein Universal-Standard-Icon mit dazu packen (keine Weltkugel!!!! ;)), damit wenigstens etwas angezeigt wird....

    So als Idee... :/

    .

Unterstütze uns!

Jährlich (2025)

101,9 %

101,9% (662,48 von 650 EUR)

Jetzt spenden
  1. Kontakt
  2. Datenschutz
  3. Impressum
Community-Software: WoltLab Suite™
Mastodon