JavaFX JavaFX Application mit Preloader sauber runterfahren

lam_tr

Top Contributor
Hallo zusammen,

ich habe eine Swing Anwendung die eine JavaFX Application mit einen Preloder startet, und zwar so

Code:
LauncherImpl#launchApplication(application, preloader, args)

Für das Beenden der Anwendung benutze ich Platform.exit(), die Swing Anwendung soll weiterhin laufen und gegebenfalls die JavaFX Anwendung erneut starten. Irgendwie fährt die JavaFX Anwendung beim beenden nicht wirklich runter. Was bedeutet nicht richtig runter fahren. Sobald ich über die Swing Anwendung die JavaFX Application mit den obigen Zeile starte, kommt bei mir diese Fehlermeldung

Code:
throw new IllegalStateException("Application launch must not be called more than once");

Kann es sein dass diese Anwendung im Swing UI Thread noch vorhanden ist, dass ich sie nicht mehr starten kann?

Ich habe in die launchApplication Methode reingeschaut

Frage: Wie kann ich eine JavaFX Anwendung richtig runterfahren?

Code:
    /**
     * This method is called by the standalone launcher.
     * It must not be called more than once or an exception will be thrown.
     *
     * Note that it is always called on a thread other than the FX application
     * thread, since that thread is only created at startup.
     *
     * @param appClass application class
     * @param preloaderClass preloader class, may be null
     * @param args command line arguments
     */
    public static void launchApplication(final Class<? extends Application> appClass,
            final Class<? extends Preloader> preloaderClass,
            final String[] args) {

        if (launchCalled.getAndSet(true)) {
            throw new IllegalStateException("Application launch must not be called more than once");
        }

        if (! Application.class.isAssignableFrom(appClass)) {
            throw new IllegalArgumentException("Error: " + appClass.getName()
                    + " is not a subclass of javafx.application.Application");
        }

        if (preloaderClass != null && ! Preloader.class.isAssignableFrom(preloaderClass)) {
            throw new IllegalArgumentException("Error: " + preloaderClass.getName()
                    + " is not a subclass of javafx.application.Preloader");
        }

//        System.err.println("launch standalone app: preloader class = "
//                + preloaderClass);

        // Create a new Launcher thread and then wait for that thread to finish
        final CountDownLatch launchLatch = new CountDownLatch(1);
        Thread launcherThread = new Thread(() -> {
            try {
                launchApplication1(appClass, preloaderClass, args);
            } catch (RuntimeException rte) {
                launchException = rte;
            } catch (Exception ex) {
                launchException =
                    new RuntimeException("Application launch exception", ex);
            } catch (Error err) {
                launchException =
                    new RuntimeException("Application launch error", err);
            } finally {
                launchLatch.countDown();
            }
        });
        launcherThread.setName("JavaFX-Launcher");
        launcherThread.start();

        // Wait for FX launcher thread to finish before returning to user
        try {
            launchLatch.await();
        } catch (InterruptedException ex) {
            throw new RuntimeException("Unexpected exception: ", ex);
        }

        if (launchException != null) {
            throw launchException;
        }
    }

Im JavaDoc steht auch schon dass man die Methode nicht mehrfach starten kann. Was kann ich stattdessen machen?

Ich habe spaßeshalber einfach mal launchCalled.getAndSet(true) vor der Exception auf true gesetzt und ausgeführt, daraus folgt eine neue Meldung mit "Toolkit has excited".

Ich benutze die LauncherImpl hauptsächlich nur wegen der Preloader.

Ich muss die Anwendung auf jeden Fall über die Swing Anwendung starten, da komme ich vorerst nicht drum rum.

ich hoffe das war einigermaßen verständlich beschrieben.

Grüße
lam
 

dzim

Top Contributor
Ich muss es leider so sagen: Nicht auf diese Art starten. Stattdessen kannst du das Toolkit manuell initialisieren (wäre ja auch denkbar, das du eine CLI-Anwendung hast, die unter Umständen ein JavaFX-Teil starten muss):
https://stackoverflow.com/questions...g-to-play-an-mp3-file-through-mediap/38883432

Wenn du noch immer auf Java 8 bist, kannst du einfach in Swing JFXPanel instantiieren (du musst es nicht einmal weiter für in die Swing-Anwendung integrierte JavaFX-Teile verwenden). Damit ist das Toolkit initialisiert und du kannst z.B. eine neue Stage öffnen.

Für Java 9+, bzw. dann ab Java 11 für JavaFX 11+, gibt es https://docs.oracle.com/javase/10/d...ion/Platform.html#startup(java.lang.Runnable)

Preloader-Funktionalität ist ja eigentlich recht easy implementiert - statt den Preloader-Events nutzt du halt eigene Properties (Object- oder BooleanProperty), um dem PseudoPreloader zu sagen, dass er wegzugehen hat, bzw. dem eigentlichen JavaFX ApplicationWindow, dass es zu erscheinen hat - je nachdem, wie du es implementierst.
 

lam_tr

Top Contributor
Wie zu erwarten mal wieder gute Tipps bekommen, werde ich nachher sofort testen.

Ich danke dir.

Die Variante vom letzten Post habe ich auch schon getestet, hatte leider nichts bewirkt.
 

lam_tr

Top Contributor
Hi dzim,

hast du noch weitere Ideen für mich?
all dies hat leider nicht funktioniert.

Wenn ich System.exist(0) mache, schließt die JavaFX meine komplette Anwendung. Das soll nicht sein.

Grüße
lam
 

dzim

Top Contributor
System.exit(0) beendet die JVM und gibt den Return Code 0 an den Caller zurück... Das willst du wirklich nicht... ;)
 

dzim

Top Contributor
Bin momentan auf meinem Branch unterwegs und hab deswegen dort mal ein Beispiel extra nur für dich erstellt.
https://github.com/bgmf/poc/blob/up...tests/swing/SwingFxInteroptWithPreloader.java

Java:
import javafx.animation.KeyFrame;
import javafx.animation.Timeline;
import javafx.application.Platform;
import javafx.beans.property.BooleanProperty;
import javafx.beans.property.SimpleBooleanProperty;
import javafx.beans.value.ChangeListener;
import javafx.embed.swing.JFXPanel;
import javafx.geometry.Rectangle2D;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.layout.BorderPane;
import javafx.scene.layout.Pane;
import javafx.scene.paint.Color;
import javafx.scene.paint.Paint;
import javafx.stage.*;
import javafx.util.Duration;

import javax.swing.*;
import java.util.function.Supplier;

public class SwingFxInteroptWithPreloader {
    public static void main(String[] args) {

        // init the ToolKit - can only be done once!
        new JFXPanel();
        // prevent implicit exit of the TK - we can't restart it in JFX 11
        Platform.setImplicitExit(false);

        // property to prevent, we are showing more than one "Application"
        final BooleanProperty appVisible = new SimpleBooleanProperty(false);
        // switch from Preloader to main App window
        final BooleanProperty showApp = new SimpleBooleanProperty(false);
        // keep in mind, we need to reset these properties!

        // init Swing frame
        JFrame f = new JFrame();
        // and add a Swing button to showcase this app
        JButton b = new JButton("Show 'Preloader'");
        b.setBounds(100, 100, 180, 40);

        f.add(b);

        // important -> kills the app
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.setSize(400, 280);
        f.setLayout(null);

        // show preloader on action event
        b.addActionListener(event -> {

            // check, whether the app is already visible -> could be done better, if we just remebemer the old app reference, I guess
            if (appVisible.get())
                return;

            // scene.setFill(Color.TRANSPARENT); root.setBackground(Background.EMPTY); primaryStage.initStyle(StageStyle.TRANSPARENT);

            // we can't directly interact with JavaFX from Swing, so we need a Platform.runLater()
            Platform.runLater(() -> {

                // preloader window
                final StageWrapper pre = new StageWrapper(StageStyle.TRANSPARENT, Modality.APPLICATION_MODAL, null, -1, -1, 320, 240, "Preloader",
                        false);
                pre.setSceneFill(Color.TRANSPARENT);

                // main window
                final StageWrapper app = new StageWrapper(StageStyle.DECORATED, Modality.APPLICATION_MODAL, null, -1, -1, 640, 480, "Application",
                        false);

                // the main window wrapper content creator
                app.setSceneContentSupplier(() -> {
                    BorderPane rootPane = new BorderPane();
                    rootPane.setCenter(new Label("Application"));
                    return rootPane;
                });

                // react on the showApp property, indicating, that the preloader is ready and the main window may appear
                // we keep the reference here, to remove it later on in the main window close event
                // otherwise we have many references on dead windows, we don't want that
                final ChangeListener<Boolean> showAppListener = (obs, o, n) -> {
                    if (n) {
                        pre.hide();
                        app.show();
                    }
                };

                // add the listener to the property
                showApp.addListener(showAppListener);
                // handle main window close request - clean everything up
                app.setOnCloseRequest(e -> {
                    showApp.removeListener(showAppListener); // remove the showApp listener
                    showApp.set(false); // reset showApp
                    appVisible.set(false); // reset appVisible
                });

                // preloader window creator combined with showing it... just for the simplicity
                pre.showContent(false, () -> {

                    BorderPane rootPane = new BorderPane();
                    rootPane.setCenter(new Label("Preloader"));

                    // handle layout changes:
                    // - set the preloader window position
                    // - init the showcase timeline, that will indicate, when the preloader is done
                    rootPane.layoutBoundsProperty().addListener((obs, o, n) -> {

                        if (n.getHeight() <= 10.0 && n.getWidth() <= 10.0)
                            return;

                        Screen primaryScreen = Screen.getPrimary();
                        Rectangle2D vb = primaryScreen.getVisualBounds();
                        double x = vb.getMinX() + (vb.getMaxX() - vb.getMinX()) / 2.0;
                        double y = (vb.getMaxY() - vb.getMinY()) / 2.0;

                        pre.setX(x - n.getWidth() / 2.0);
                        pre.setY(y - n.getHeight() / 2.0);

                        Timeline timeline = new Timeline(new KeyFrame(Duration.millis(2500), ae -> showApp.set(true)));
                        timeline.play();
                    });

                    // don't forget to set the property, that the app is now visible
                    // (regardless of whether it's just the preloader or the main window)
                    appVisible.set(true);

                    return rootPane;
                });
            });
        });

        // show the Swing window
        f.setVisible(true);
    }
}
Stage Wrapper, nur als Convenience-Ding
Java:
public class StageWrapper extends Stage {

    private final double width;
    private final double height;

    private Paint sceneFill = Color.WHITE;
    private Supplier<Pane> sceneContentSupplier = BorderPane::new;

    public StageWrapper(StageStyle style, Modality modality, Window owner, double minWidth, double minHeight, double width, double height,
            String title, boolean alwaysOnTop) {

        super(style);

        initModality(modality);
        initOwner(owner);

        if (minWidth > 0)
            setMinWidth(minWidth);
        if (minHeight > 0)
            setMinHeight(minHeight);

        this.width = width;
        this.height = height;

        setTitle(title);
        setAlwaysOnTop(alwaysOnTop);
    }

    public void setSceneFill(Paint sceneFill) {
        this.sceneFill = sceneFill;
    }

    public void setSceneContentSupplier(Supplier<Pane> sceneContentSupplier) {
        this.sceneContentSupplier = sceneContentSupplier;
        createContent();
    }

    private void createContent() {
        Pane content = sceneContentSupplier.get();
        Scene scene = new Scene(content, width, height);
        scene.setFill(sceneFill);
        setScene(scene);
    }

    public void showContent(boolean showAndWait, Supplier<Pane> sceneContentSupplier) {

        this.sceneContentSupplier = sceneContentSupplier;

        createContent();

        if (showAndWait)
            showAndWait();
        else
            show();
    }
}
 

lam_tr

Top Contributor
Du steckst immer so viel Arbeit in die Beispiele, dafür bin ich dir 1000 mal dank. Ich check das nachher up und gebe dir dann eine Rückmeldung.
 

dzim

Top Contributor
Wenn ich das Thema interessant finde, mache ich das auch gern... ;)
Hat mich tatsächlich auch ein paar Anläufe und etwas Recherche gekostet, bis alles so lief, wie es sollte. Speziell das Problem mit dem "Implicit Exit" hat mich etwas gekostet. Hab mich durch-debuggt, bis ich den Grund dafür gesehen habe, warum der "Preloader" immer nur ein einziges Mal kommt. Danach war es easy: bloß noch "setImplicitExit" auf "false" und etwas den Code aufräumen, et voilà.
War ne interessant ca. 3/4 Stunde.
 
Ähnliche Java Themen
  Titel Forum Antworten Datum
H javafx application does not exist AWT, Swing, JavaFX & SWT 16
N JavaFX Projekt nachträglich zu JavaFX Application machen? AWT, Swing, JavaFX & SWT 1
R javafx erste application AWT, Swing, JavaFX & SWT 12
N JavaFX Logging des JavaFX Application Threads mit Log4J AWT, Swing, JavaFX & SWT 3
S JavaFX Exception in thread "JavaFX Application Thread" AWT, Swing, JavaFX & SWT 3
E JavaFX JavaFX Application in Thread ausführen AWT, Swing, JavaFX & SWT 1
Juelin starten maven javafx programm ohne netbeans AWT, Swing, JavaFX & SWT 38
Juelin javax.swing in javafx AWT, Swing, JavaFX & SWT 1
MiMa JUnit5 im JavaFX Projekt AWT, Swing, JavaFX & SWT 2
Juelin in javafx Event auslösen AWT, Swing, JavaFX & SWT 4
MiMa Fonts, Icons, Bilder in JavaFX Anwendung AWT, Swing, JavaFX & SWT 5
MiMa SwingFXUtils in JavaFX 20 AWT, Swing, JavaFX & SWT 6
MiMa JavaFX Fenstertitel zu klein. AWT, Swing, JavaFX & SWT 1
MiMa JavaFX JAR unter Windows ausführen schlägt fehl? AWT, Swing, JavaFX & SWT 5
I JavaFX JavaFx-Anwendung für die Erstellung einer Windows-Anwendung? AWT, Swing, JavaFX & SWT 6
Hatsi09 Javafx MediaPlayer spielt nicht immer AWT, Swing, JavaFX & SWT 3
Hatsi09 Javafx Neuladen von ImageView und MediaView AWT, Swing, JavaFX & SWT 3
Maxim6394 JavaFX Umlaute in JavaFX GUI AWT, Swing, JavaFX & SWT 12
Maxim6394 JavaFX Scene Builder - Crash bei eigener Komponente AWT, Swing, JavaFX & SWT 2
Ernesto95 JavaFX JavaFX GUI mit sehr vielen Update requests AWT, Swing, JavaFX & SWT 4
Telisti Javafx Image wird nicht richtig integiert AWT, Swing, JavaFX & SWT 8
J Netbeans die JavaFX-Anwendung wird nicht ausgeführt AWT, Swing, JavaFX & SWT 16
MartinNeuerlich Kann mir jemand, der einen Mac mit einem m1 oder m2-Chip hat, eine POM geben mit der Javafx-Fullscreen beim Mac mit m-Chip funktioniert? AWT, Swing, JavaFX & SWT 1
tommybalbor JavaFx Anwendung klappt nicht für macOs Nutzern, wenn ich zwei dependecies bei maven hinzufüge AWT, Swing, JavaFX & SWT 6
JavaSchmecktLecker JavaFX JavaFX Ordner automatisch verlinken AWT, Swing, JavaFX & SWT 2
melaniemueller Taschenrechner JavaFX AWT, Swing, JavaFX & SWT 4
R auto. Importanweisungen für javafx funktioniert in Eclipse nicht mehr AWT, Swing, JavaFX & SWT 4
thor_norsk JavaFX Anwendung stürzt ab AWT, Swing, JavaFX & SWT 4
berserkerdq2 Skalieren sich javafx objekte automatisch auf die Bildschirmgröße AWT, Swing, JavaFX & SWT 6
berserkerdq2 Wie füge ich ein Bild in javafx mit dem Scenebuilder ein, das automatisch mitgezogen wird, wenn das Fenster vergrößert wird oder Vollbildmodus AWT, Swing, JavaFX & SWT 6
B Java Projekt mit JavaFX und jfoenix ausführbar machen AWT, Swing, JavaFX & SWT 46
H JavaFX wie JavaFX Projekt aufsetzen? AWT, Swing, JavaFX & SWT 10
thor_norsk JavaFX - Grafikkarte AWT, Swing, JavaFX & SWT 7
MiHimbert Rückmeldung an den aufrufenden JAVAFX-Dialog AWT, Swing, JavaFX & SWT 1
MiMa JavaFX Runtime components are Missing??? AWT, Swing, JavaFX & SWT 3
J JavaFx PDF in einem Element in einem Fenster anzeigen. AWT, Swing, JavaFX & SWT 11
B JavaFX Sprachumschaltung mit Button auf der HMI AWT, Swing, JavaFX & SWT 6
H JavaFX Fehlende JavaFX Package AWT, Swing, JavaFX & SWT 10
K JavaFX unterschiedliche (mehrere Fenster) in seperater Main Methode AWT, Swing, JavaFX & SWT 26
_user_q Kann man ein 2. JavaFX-Fenster auch beenden (exit) statt schließen (close) lassen? AWT, Swing, JavaFX & SWT 8
G JavaFX Line Chart mit Farbverlauf/Gradient in Linie AWT, Swing, JavaFX & SWT 1
thor_norsk JavaFX, FXML und SceneBuilder AWT, Swing, JavaFX & SWT 6
_user_q Über installDist exportiertes Programm wirft "Unsupported JavaFX configuration" AWT, Swing, JavaFX & SWT 0
CodingBerlin JavaFX Programm läuft nur unter Eclipse AWT, Swing, JavaFX & SWT 1
H Fehler: Zum Ausführen dieser Anwendung benötigte JavaFX-Runtime-Komponenten fehlen AWT, Swing, JavaFX & SWT 44
temi JavaFX "Frames" in JavaFx - passende Komponente? AWT, Swing, JavaFX & SWT 13
G JavaFX Steuerung bzw. Test von externer JavaFX Anwendung (liegt nur als jar vor) AWT, Swing, JavaFX & SWT 9
_user_q [JavaFX] Spinner so einstellen, dass er nicht leer bleiben darf? AWT, Swing, JavaFX & SWT 6
S Javafx getResource-Pfad wird nicht erkannt AWT, Swing, JavaFX & SWT 7
A JavaFX exportierte Jar ohne beim starten die Libs hinzufügen? AWT, Swing, JavaFX & SWT 2
J JavaFX Schiffe versenken mit JavaFX und Scene builder AWT, Swing, JavaFX & SWT 3
Encera ArrayList mit eigenen Objekten in JavaFX sortieren und ausgeben AWT, Swing, JavaFX & SWT 50
L JavaFx Textformatierung mittels Datenbank und Funktion anpassen AWT, Swing, JavaFX & SWT 5
sserio Wie funktioniert ein Controller bei JavaFx? AWT, Swing, JavaFX & SWT 1
sserio Kann man bei JavaFx ein Fenster aufkommen lassen? AWT, Swing, JavaFX & SWT 1
Jose05 JavaFx Fxml: GUI aus einer anderen Klasse starten AWT, Swing, JavaFX & SWT 1
Tassos JavaFX/Problem mit der Maussteuerung in Stackpane AWT, Swing, JavaFX & SWT 7
S Ich bringe Code mit JavaFX unter Apache NetBeans IDE 12.6 nicht zum laufen. AWT, Swing, JavaFX & SWT 14
K Bekomme (u.a) javafx.fxml.LoadException trotz "korrektem" Code AWT, Swing, JavaFX & SWT 8
S JavaFX: voneinander abhängige TextFields AWT, Swing, JavaFX & SWT 33
M Gluon will JavaFX in den Browser stecken AWT, Swing, JavaFX & SWT 0
A JavaFX Controller Problem AWT, Swing, JavaFX & SWT 1
izoards JavaFX TableView mit Array Inhalt füllen AWT, Swing, JavaFX & SWT 1
M Javafx versuch Bibliothek zu erstellen AWT, Swing, JavaFX & SWT 0
N JavaFX Javafx intelij Projekt zu ausführbaren jar Datei Machen AWT, Swing, JavaFX & SWT 1
K JavaFx, Sound Aufnahme und Thread AWT, Swing, JavaFX & SWT 0
izoards JavaFX TextFlow - Sonderzeichen AWT, Swing, JavaFX & SWT 1
maximstein JavaFX WebView - java.lang.NoSuchMethodError: 'boolean com.sun.prism.ResourceFactory.isDisposed()' AWT, Swing, JavaFX & SWT 4
N JavaFX Unicode zeichnen in javafx Label verwenden AWT, Swing, JavaFX & SWT 2
MiHimbert javaFX openfx (17) datepicker AWT, Swing, JavaFX & SWT 3
A Mit JavaFX einzelne Zeilen in TableView farbig markieren AWT, Swing, JavaFX & SWT 5
melaniemueller JavaFX Taschenrechner mit SceneBuilder AWT, Swing, JavaFX & SWT 12
Jose05 Javafx Label Höhe=Breite AWT, Swing, JavaFX & SWT 1
Jose05 JavaFX: eigene FXML-Datei für einen Button AWT, Swing, JavaFX & SWT 3
izoards JavaFX editierbare Tabelle AWT, Swing, JavaFX & SWT 4
N javafx Position der Bustaben finden label AWT, Swing, JavaFX & SWT 1
D Verschieden Scenen ansprechen mit dem Scene Builder und JavaFX (Eclipse) AWT, Swing, JavaFX & SWT 16
izoards JavaFX Background Task warten auf Knopfdruck AWT, Swing, JavaFX & SWT 4
M Zufallsgenerator bei JavaFx AWT, Swing, JavaFX & SWT 1
N Label Schriftart Ändern javafx AWT, Swing, JavaFX & SWT 2
L JavaFX JavaFX, MVVM und SceneBuilder AWT, Swing, JavaFX & SWT 4
S JavaFx Album AWT, Swing, JavaFX & SWT 137
I JavaFX - Pane wechseln über 2. Controller AWT, Swing, JavaFX & SWT 5
melaniemueller JavaFX Beispiel kann nicht ausgeführt werden AWT, Swing, JavaFX & SWT 4
T FXML Datei in Java Code einbinden: javafx.fxml.LoadException AWT, Swing, JavaFX & SWT 2
J JavaFX - Included FXML - Entfernen feststellen AWT, Swing, JavaFX & SWT 2
J JavaFX JavaFX/ Taskmenu / UML Klassendiagramm AWT, Swing, JavaFX & SWT 2
Davee JavaFX JavaFX Jar ausführbar jedoch nicht alle Stages AWT, Swing, JavaFX & SWT 3
2 JavaFX die ChoiceBox leitet den String nicht weiter oder es komm zu einem NullPointer AWT, Swing, JavaFX & SWT 8
C MouseEvent JavaFX AWT, Swing, JavaFX & SWT 4
L JavaFX javafx.fxml.LoadException bei einem Taschenrechner AWT, Swing, JavaFX & SWT 5
M4cM4rco0707 JavaFX Custom-Komponente mit Custom-Controller AWT, Swing, JavaFX & SWT 3
M Error occurred during initialization of boot layer java.lang.module.FindException: Module javafx.controls not found AWT, Swing, JavaFX & SWT 14
G javafx build.fxbuild in eclipse und ANT AWT, Swing, JavaFX & SWT 9
M Kollisionensbehandlung mit JavaFX AWT, Swing, JavaFX & SWT 1
N JavaFX - Toolkit not initialized AWT, Swing, JavaFX & SWT 6
G JavaFX , Duplicate erzeugt aber ich weis nicht wo AWT, Swing, JavaFX & SWT 4
K Javafx Plugin (javafx:jlink) mit moditect Plugin nutzen ... AWT, Swing, JavaFX & SWT 5
J Saubere Trennung Model, View, Controller Javafx AWT, Swing, JavaFX & SWT 10
G JavaFX BarChart während der Runtime aktualisieren AWT, Swing, JavaFX & SWT 4

Ähnliche Java Themen

Neue Themen


Oben