--- parent: Code Howtos --- # JavaFX > [JavaFX](https://github.com/openjdk/jfx?tab=readme-ov-file#openjfx) is an open source, next generation client application platform for desktop, mobile and embedded systems based on JavaSE. > It is a collaborative effort by many individuals and companies with the goal of producing a modern, efficient, and fully featured toolkit for developing rich client applications. JavaFX is used on JabRef for the user interface. ## Resources * [JavaFX Documentation project](https://fxdocs.github.io/docs/html5/index.html): Collected information on JavaFX in a central place * [curated list of awesome JavaFX frameworks, libraries, books and etc...](https://github.com/mhrimaz/AwesomeJavaFX?tab=readme-ov-file#awesome-javafx-) * [FXTutorials](https://github.com/AlmasB/FXTutorials?tab=readme-ov-file#fxtutorials) A wide range of practical tutorials focusing on Java, JavaFX and FXGL * [ControlsFX](http://fxexperience.com/controlsfx/features/) amazing collection of controls * [CSS Reference](http://docs.oracle.com/javafx/2/api/javafx/scene/doc-files/cssref.html) * [mvvm framework](https://github.com/sialcasa/mvvmFX/wiki) * [Validation framework](https://github.com/sialcasa/mvvmFX/wiki/Validation) * [additional bindings](https://github.com/lestard/advanced-bindings) or [EasyBind](https://github.com/TomasMikula/EasyBind) * [Undo manager](https://github.com/FXMisc/UndoFX) * [Docking manager](https://github.com/alexbodogit/AnchorFX) or [DockFX](https://github.com/RobertBColton/DockFX) * [Kubed](https://github.com/hudsonb/kubed): data visualization (inspired by d3) * [Foojay](https://foojay.io) Java and JavaFX tutorials ### Resources of historical interest * [FXExperience](http://fxexperience.com) JavaFX Links of the week ## Architecture: Model - View - (Controller) - ViewModel (MV(C)VM) The goal of the MVVM architecture is to separate the state/behavior from the appearance of the ui. This is archived by dividing JabRef into different layers, each having a clear responsibility. * The _Model_ contains the business logic and data structures. These aspects are again encapsulated in the _logic_ and _model_ package, respectively. * The _View_ controls the appearance and structure of the UI. It is usually defined in a _FXML_ file. * _View model_ converts the data from logic and model in a form that is easily usable in the gui. Thus it controls the state of the View. Moreover, the ViewModel contains all the logic needed to change the current state of the UI or perform an action. These actions are usually passed down to the _logic_ package, after some data validation. The important aspect is that the ViewModel contains all the ui-related logic but does _not_ have direct access to the controls defined in the View. Hence, the ViewModel can easily be tested by unit tests. * The _Controller_ initializes the view model and binds it to the view. In an ideal world all the binding would already be done directly in the FXML. But JavaFX's binding expressions are not yet powerful enough to accomplish this. It is important to keep in mind that the Controller should be as minimalistic as possible. Especially one should resist the temptation to validate inputs in the controller. The ViewModel should handle data validation! It is often convenient to load the FXML file directly from the controller. The only class which access model and logic classes is the ViewModel. Controller and View have only access the ViewModel and never the backend. The ViewModel does not know the Controller or View. More details about the MVVM pattern can be found in [an article by Microsoft](https://msdn.microsoft.com/en-us/magazine/dd419663.aspx) and in [an article focusing on the implementation with JavaFX](https://web.archive.org/web/20140825151304/http://blog.buildpath.de/javafx-decouple-the-view-and-its-behavior-to-create-a-testable-ui/). ### Example #### ViewModel * The ViewModel should derive from `AbstractViewModel` ```java public class MyDialogViewModel extends AbstractViewModel { } ``` * Add a (readonly) property as a private field and generate the getters according to the [JavaFX bean conventions](https://docs.oracle.com/javafx/2/binding/jfxpub-binding.htm): ```java private final ReadOnlyStringWrapper heading = new ReadOnlyStringWrapper(); public ReadOnlyStringProperty headingProperty() { return heading.getReadOnlyProperty(); } public String getHeading() { return heading.get(); } ``` * Create constructor which initializes the fields to their default values. Write tests to ensure that everything works as expected! ```java public MyDialogViewModel(Dependency dependency) { this.dependency = Objects.requireNonNull(dependency); heading.set("Hello " + dependency.getUserName()); } ``` * Add methods which allow interaction. Again, don't forget to write tests! ```java public void shutdown() { heading.set("Goodbye!"); } ``` #### View - Controller * The "code-behind" part of the view, which binds the `View` to the `ViewModel`. * The usual convention is that the controller ends on the suffix `*View`. Dialogs should derive from `BaseDialog`. ```java public class AboutDialogView extends BaseDialog ``` * You get access to nodes in the FXML file by declaring them with the `@FXML` annotation. ```java @FXML protected Button helloButton; @FXML protected ImageView iconImage; ``` * Dependencies can easily be injected into the controller using the `@Inject` annotation. ```java @Inject private DialogService dialogService; ``` * It is convenient to load the FXML-view directly from the controller class. The FXML file is loaded using `ViewLoader` based on the name of the class passed to `view`. To make this convention-over-configuration approach work, both the FXML file and the View class should have the same name and should be located in the same package. Note that fields annotated with `@FXML` or `@Inject` only become accessible after `ViewLoader.load()` is called. a `View` class that loads the FXML file. ```java private Dependency dependency; public AboutDialogView(Dependency dependency) { this.dependency = dependency; this.setTitle(Localization.lang("About JabRef")); ViewLoader.view(this) .load() .setAsDialogPane(this); } ``` * Dialogs should use [setResultConverter](https://docs.oracle.com/javase/8/javafx/api/javafx/scene/control/Dialog.html#setResultConverter-javafx.util.Callback-) to convert the data entered in the dialog to the desired result. This conversion should be done by the view model and not the controller. ```java setResultConverter(button -> { if (button == ButtonType.OK) { return viewModel.getData(); } return null; }); ``` * The initialize method may use data-binding to connect the ui-controls and the `ViewModel`. However, it is recommended to do as much binding as possible directly in the FXML-file. ```java @FXML private void initialize() { viewModel = new AboutDialogViewModel(dialogService, dependency, ...); helloLabel.textProperty().bind(viewModel.helloMessageProperty()); } ``` * calling the view model: ```java @FXML private void openJabrefWebsite() { viewModel.openJabrefWebsite(); } ``` #### View - FXML The view consists a FXML file `MyDialog.fxml` which defines the structure and the layout of the UI. Moreover, the FXML file may be accompanied by a style file that should have the same name as the FXML file but with a `css` ending, e.g., `MyDialog.css`. It is recommended to use a graphical design tools like [SceneBuilder](http://gluonhq.com/labs/scene-builder/) to edit the FXML file. The tool [Scenic View](https://github.com/JonathanGiles/scenic-view) is very helpful in debugging styling issues. ## FXML The following expressions can be used in FXML attributes, according to the [official documentation](https://docs.oracle.com/javase/8/javafx/api/javafx/fxml/doc-files/introduction_to_fxml.html#attributes) | Type | Expression | Value point to | Remark | | -------------------------------- | ------------------------------------------------ | ---------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------- | | Location | `@image.png` | path relative to the current FXML file | | | Resource | `%textToBeTranslated` | key in ResourceBundle | | | Attribute variable | `$idOfControl` or `$variable` | named control or variable in controller (may be path in the namespace) | resolved only once at load time | | Expression binding | `${expression}` | expression, for example `textField.text` | changes to source are propagated | | Bidirectional expression binding | `#{expression}` | expression | changes are propagated in both directions (not yet implemented in JavaFX, see [feature request](https://bugs.openjdk.java.net/browse/JDK-8090665)) | | Event handler | `#nameOfEventHandler` | name of the event handler method in the controller | | | Constant | `` | constant (here `MYSTRING` in the `Strings` class) | | ## JavaFX Radio Buttons Example All radio buttons that should be grouped together need to have a ToggleGroup defined in the FXML code Example: ```markup ``` ## JavaFX Dialogs All dialogs should be displayed to the user via `DialogService` interface methods. `DialogService` provides methods to display various dialogs (including custom ones) to the user. It also ensures the displayed dialog opens on the correct window via `initOwner()` (for cases where the user has multiple screens). The following code snippet demonstrates how a custom dialog is displayed to the user: ```java dialogService.showCustomDialog(new DocumentViewerView()); ``` If an instance of `DialogService` is unavailable within current class/scope in which the dialog needs to be displayed, `DialogService` can be instantiated via the code snippet shown as follows: ```java DialogService dialogService = Injector.instantiateModelOrService(DialogService.class); ``` ## Properties and Bindings JabRef makes heavy use of Properties and Bindings. These are wrappers around Observables. A good explanation on the concept can be found here: [JavaFX Bindings and Properties](https://edencoding.com/javafx-properties-and-binding-a-complete-guide/) ## Features missing in JavaFX * bidirectional binding in FXML, see [official feature request](https://bugs.openjdk.java.net/browse/JDK-8090665)