2012-07-08 5 views
20

Quelle est la bonne façon d'initialiser le runtime JavaFX afin de pouvoir tester des unités (avec JUnit) utilisant les fonctions de concurrence et Platform.runLater(Runnable)? L'appel Application.launch(...) de la méthode @BeforeClass entraîne un blocage mort. Si Application.launch(...) n'est pas appelé alors l'erreur suivante est générée:Comment tester un contrôleur JavaFX avec JUnit


java.lang.IllegalStateException: Toolkit not initialized 
    at com.sun.javafx.application.PlatformImpl.runLater(PlatformImpl.java:121) 
    at com.sun.javafx.application.PlatformImpl.runLater(PlatformImpl.java:116) 
    at javafx.application.Platform.runLater(Platform.java:52) 
    at javafx.concurrent.Task.runLater(Task.java:1042) 
    at javafx.concurrent.Task.updateMessage(Task.java:987) 
    at com.xyz.AudioSegmentExtractor.call(AudioSegmentExtractor.java:64) 
    at com.xyz.CompletionControllerTest.setUp(CompletionControllerTest.java:69) 
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) 
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
    at java.lang.reflect.Method.invoke(Method.java:601) 
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44) 
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15) 
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41) 
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:27) 
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:76) 
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50) 
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193) 
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52) 
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191) 
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42) 
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184) 
    at org.junit.runners.ParentRunner.run(ParentRunner.java:236) 
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50) 
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390) 
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197) 

Followup: c'est le motif que je l'ai utilisé en fonction de recommandation @SergeyGrinev.

... // Inside test class 

public static class AsNonApp extends Application { 
    @Override 
    public void start(Stage primaryStage) throws Exception { 
     // noop 
    } 
} 

@BeforeClass 
public static void initJFX() { 
    Thread t = new Thread("JavaFX Init Thread") { 
     public void run() { 
      Application.launch(AsNonApp.class, new String[0]); 
     } 
    }; 
    t.setDaemon(true); 
    t.start(); 
} 
... // controller tests follow... 

Répondre

14

Appel launch() de @BeforeClass est une approche correcte. Notez simplement que launch() ne renvoie pas le contrôle au code appelant. Donc, vous devez l'envelopper dans new Thread(...).start().

7

J'ai trouvé ceci pour fonctionner, ... mais seulement après avoir ajouté un Thread.sleep (500) après avoir démarré le thread d'application JavaFX. Vraisemblablement, il faut un certain temps pour obtenir l'environnement de change et prêt (environ 200ms sur ma rétine MacBook Pro)

@BeforeClass 
public static void setUpClass() throws InterruptedException { 
    // Initialise Java FX 

    System.out.printf("About to launch FX App\n"); 
    Thread t = new Thread("JavaFX Init Thread") { 
     public void run() { 
      Application.launch(AsNonApp.class, new String[0]); 
     } 
    }; 
    t.setDaemon(true); 
    t.start(); 
    System.out.printf("FX App thread started\n"); 
    Thread.sleep(500); 
} 
+3

[Application.launch] (Il ne doit pas être appelé plus d'une fois ou une exception sera levée.) Les états de documentation: "Il ne doit pas être appelé plus d'une fois ou une exception sera levée.", juste quelque chose à savoir si vous essayez d'exécuter la méthode setUpClass() dans cette réponse plus d'une fois. – jewelsea

Questions connexes