На JavaOne 2011 было объявлено о выпуске финальной версии JavaFX 2.0. Рассмотрим из чего оно состояит и какие средства предлагает.

Создадим и запустим первое приложение на JavaFX. Для этого нам понадобится JavaFX 2.0 SDK и последний Netbeans. О настройке JavaFX в Netbeans можно прочитать на странице http://netbeans.org/kb/docs/java/javafx-setup.html.
Запускаем Netbeans, выбираем из меню File/New Project, в диалоге выбираем категорию JavaFX и тип проекта JavaFX Application. Будет создан проект с одной тестовой формой как на картинке ниже (можно назвать проект HelloJavaFX или как-то иначе). Код класса формы:
результат:

Посмотрим что у нас появится в папке dist после компиляции. Там будут следующие файлы:
— папка web-files — содержит картинки для запуска в виде апплета
— HelloJavaFX.html — страница с встроенным апплетом приложения
— HelloJavaFX.jar — приложение для запуска обычным двойным щелчком мыши
— HelloJavaFX.jnlp — описатель запуска через WebStart
Нас интересует само приложение HelloJavaFX.jar. Откроем его любым архиватором (.jar это обычный .zip-архив) и посмотрим на манифест. Главная строка запуска:
говорит о том что в каждое JavaFX-приложение Netbeans'ом добавляется класс-стартер com.javafx.main.Main, который инициализирует все библиотеки JavaFX и запускает само приложение. Имя главного класс берётся из поля JavaFX-Application-Class, в нашем случает это hellojavafx.HelloJavaFX
Ничего необычного. На данный момент JavaFX это просто набор библиотек, функции которых можно использовать из Java-кода.
Рассмотрим пример Swing-формы со встроенным JavaFX-компонентом. Создадим в Netbeans новый проект и добавим в него ссылку на Runtime-библиотеку JavaFX (по умолчанию она находится в файле C:\Program Files\Oracle\JavaFX Runtime 2.0\lib\jfxrt.jar).
После подключения Runtime, можно встраивать любые JavaFX-компоненты с помощью класса JFXPanel. Дабавим текстовый редактор с помощью следующего кода:
В результате получится такая форма:

Как видно по коду кнопки Read, мы в любой момент можем получить доступ к данным в текстовом редакторе. Для запуска такого приложения мы должны добавить в class-path путь к Runtime-библиотеки JavaFX, например так:
Всё достаточно тривиально и не отличается от использования других библиотек в Java-проектах.
С возможностями JavaFX можно ознакомиться на странице платформы. В API есть набор классов для рисования примитивов, графиков, набор компонентов GUI с собственным оформлением. Наконец-то появился компонент веб-браузера и текстовый редактор с возможностью форматирования.
На странице http://www.oracle.com/technetwork/java/javafx/overview/index.html можно запустить демо-приложение Ensemble (это аналог Flex Component Explorer).

Binding (связывание) предназначен для связывания свойств объектов. Например вы можете привязать координаты кнопки к ширине окна и в результате при изменении размеров формы, переместится и привязанная кнопка.
Рассмотрим на примере:
Результат:

Строка с кодом
создаёт переменную db и привязывает её значение к ширине формы минус 150 пикселов. К сожалению мы не можем просто привязать координаты кнопки к этой переменной (типа btn.layoutX.bind(db); что было бы очевидно и удобно). Поэтому к созданной переменной надо добавить Listener, в котором и обновлять координаты кнопки:
Кроме того, binding накладывает дополнительные ограничения — если вы привяжете свойство к переменной то не сможете изменить простым присваиванием, будет выброшено исключение во время работы приложения.
Сам код API чрезмерно раздут, например класс SimpleDoubleProperty имеет иерархию наследования из шести классов и имплементирует девять интерфейсов. Это новое понимание характеристики simple.
В результате хорошая идея выглядит не очень привлекательно. Много кода, неявные ошибки. Хотя простая и понятная реализация связывания есть для многих языков и библиотек, в том числе и для Swing.
Оформление интерфейса можно задавать с помощью CSS. Для этого нужно создать описание стилей, например такое:
— стиль cssbutton задаёт шрифт, отступы, заливку и эффект тени. Далее нужно подключить файл стилей к сцене:
и добавить стиль к компоненту по имени:
Кнопка будет выглядеть примерно так:

Описание доступных в CSS свойств можно прочитать на странице документации.
Не очень понятно зачем вообще нужно задавать оформление стилями. Всё что можно задать стилями можно сделать и в коде, в CSS нельзя использовать переменные или вычисления, нельзя создавать новые свойства, стили сложно отлаживать.
Компоновку форм можно задать в отдельном файле FXML и при старте приложения загрузить из него. Такой подход используется во многих средах разработки.
В Netbeans для этих целей предлагается создать проект с типом JavaFX FXML Application. Будет создана начальная форма примерно такого содержания:
Аннотация @FXML служит для обозначения переменных и функций доступных при загрузке из FXML-файла. Сам файл выглядит примерно так:
Как видно в примере, и метод handleButtonAction и свойство label заданные в коде, используются внутри FXML-описания формы.
Загрузка FXML-формы при старте приложения вызывается примерно так:
Каких-то преимуществ подобное создание GUI не даёт. Визуального редактора нет, а править руками одинаково затратно как Java-код, так и FXML-описание.
Форморисовалка есть в Visual Studio для VB и C#, в Adobe Flex, в Oracle JBuilder есть прекрасный редактор для Swing. Для JavaFX визульный редактор был объявлен ещё 2 года назад. Можно даже видео с ним посмотреть на странице http://www.flickr.com/photos/douglasbullard/3609213630/in/photostream/.
Но пока публичного релиза не было и неизвестно когда будет.
Общая оценка скорей всего неудовлетворительная. Демо-приложения тормозят, внешний вид компонентов уступает Macintosh Aqua, Adone Flex или Microsoft Metro. Первая версия JavaFX появилась ещё в далёком 2008 году но Sun так и не смог довести технологию до ума.
Тем не менее, на последней JavaOne о JavaFX говорилось очень много. Будем надеяться что Oracle железной рукой направит разработчиков в нужное русло и заставит выдать наконец-то рабочий инструмент весто обещаний.

Hello JavaFX
Создадим и запустим первое приложение на JavaFX. Для этого нам понадобится JavaFX 2.0 SDK и последний Netbeans. О настройке JavaFX в Netbeans можно прочитать на странице http://netbeans.org/kb/docs/java/javafx-setup.html.
Запускаем Netbeans, выбираем из меню File/New Project, в диалоге выбираем категорию JavaFX и тип проекта JavaFX Application. Будет создан проект с одной тестовой формой как на картинке ниже (можно назвать проект HelloJavaFX или как-то иначе). Код класса формы:
package hellojavafx; import javafx.application.*; import javafx.event.*; import javafx.scene.*; import javafx.scene.control.*; import javafx.stage.*; public class HelloJavaFX extends Application { public static void main(String[] args) { Application.launch(args); } @Override public void start(Stage primaryStage) { primaryStage.setTitle("Hello World"); Group root = new Group(); Scene scene = new Scene(root, 300, 250); Button btn = new Button(); btn.setLayoutX(100); btn.setLayoutY(80); btn.setText("Hello World"); btn.setOnAction(new EventHandler<ActionEvent>() { public void handle(ActionEvent event) { System.out.println("Hello World"); } }); root.getChildren().add(btn); primaryStage.setScene(scene); primaryStage.show(); } }
результат:

Посмотрим что у нас появится в папке dist после компиляции. Там будут следующие файлы:
— папка web-files — содержит картинки для запуска в виде апплета
— HelloJavaFX.html — страница с встроенным апплетом приложения
— HelloJavaFX.jar — приложение для запуска обычным двойным щелчком мыши
— HelloJavaFX.jnlp — описатель запуска через WebStart
Нас интересует само приложение HelloJavaFX.jar. Откроем его любым архиватором (.jar это обычный .zip-архив) и посмотрим на манифест. Главная строка запуска:
Main-Class: com/javafx/main/Mainговорит о том что в каждое JavaFX-приложение Netbeans'ом добавляется класс-стартер com.javafx.main.Main, который инициализирует все библиотеки JavaFX и запускает само приложение. Имя главного класс берётся из поля JavaFX-Application-Class, в нашем случает это hellojavafx.HelloJavaFX
Ничего необычного. На данный момент JavaFX это просто набор библиотек, функции которых можно использовать из Java-кода.
JavaFX в Swing
Рассмотрим пример Swing-формы со встроенным JavaFX-компонентом. Создадим в Netbeans новый проект и добавим в него ссылку на Runtime-библиотеку JavaFX (по умолчанию она находится в файле C:\Program Files\Oracle\JavaFX Runtime 2.0\lib\jfxrt.jar).
После подключения Runtime, можно встраивать любые JavaFX-компоненты с помощью класса JFXPanel. Дабавим текстовый редактор с помощью следующего кода:
package swingjavafx; import javax.swing.*; import java.awt.event.*; import javafx.embed.swing.*; import javafx.scene.*; import javafx.application.*; import javafx.scene.web.*; import javafx.scene.effect.*; public class SwingJavaFx extends JFrame { HTMLEditor edtr; public SwingJavaFx() { this.setSize(600, 400); this.setVisible(true); this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); this.setLayout(null); JButton b1 = new JButton("Read"); b1.setSize(150, 22); b1.setLocation(10, 10); b1.addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { JOptionPane.showMessageDialog(null, edtr.getHtmlText()); } }); this.add(b1); final JFXPanel jfx = new JFXPanel(); jfx.setSize(560, 300); jfx.setLocation(10, 40); this.add(jfx); Platform.runLater(new Runnable() { @Override public void run() { Group root = new Group(); Scene scene = new Scene(root, 400, 300); jfx.setScene(scene); javafx.scene.shape.Rectangle rctngl = new javafx.scene.shape.Rectangle(); rctngl.setTranslateX(20); rctngl.setTranslateY(30); rctngl.setWidth(500); rctngl.setHeight(250); rctngl.setEffect(new Shadow()); root.getChildren().add(rctngl); edtr = new HTMLEditor(); edtr.setHtmlText("Blablabla"); edtr.setTranslateX(20); edtr.setTranslateY(30); edtr.setPrefWidth(500); edtr.setPrefHeight(250); root.getChildren().add(edtr); } }); } public static void main(String[] args) { new SwingJavaFx(); } }
В результате получится такая форма:

Как видно по коду кнопки Read, мы в любой момент можем получить доступ к данным в текстовом редакторе. Для запуска такого приложения мы должны добавить в class-path путь к Runtime-библиотеки JavaFX, например так:
java -cp %cp%;"C:\Program Files\Oracle\JavaFX Runtime 2.0\lib\jfxrt.jar";SwingJavaFx.jar swingjavafx.SwingJavaFxВсё достаточно тривиально и не отличается от использования других библиотек в Java-проектах.
API JavaFX
С возможностями JavaFX можно ознакомиться на странице платформы. В API есть набор классов для рисования примитивов, графиков, набор компонентов GUI с собственным оформлением. Наконец-то появился компонент веб-браузера и текстовый редактор с возможностью форматирования.
На странице http://www.oracle.com/technetwork/java/javafx/overview/index.html можно запустить демо-приложение Ensemble (это аналог Flex Component Explorer).

Binding
Binding (связывание) предназначен для связывания свойств объектов. Например вы можете привязать координаты кнопки к ширине окна и в результате при изменении размеров формы, переместится и привязанная кнопка.
Рассмотрим на примере:
package bindingjavafx; import javafx.application.*; import javafx.event.*; import javafx.scene.*; import javafx.scene.control.*; import javafx.stage.*; import javafx.beans.binding.*; import javafx.beans.value.*; public class BindingJavaFX extends Application { public static void main(String[] args) { Application.launch(args); } @Override public void start(Stage primaryStage) { primaryStage.setTitle("Hello World"); Group root = new Group(); final Scene scene = new Scene(root, 300, 250); final Button btn = new Button(); btn.setLayoutX(100); btn.setLayoutY(80); btn.setText("Hello World"); btn.setOnAction(new EventHandler<ActionEvent>() { public void handle(ActionEvent event) { System.out.println("Hello World"); } }); final DoubleBinding db = scene.widthProperty().subtract(150); db.addListener(new javafx.beans.value.ChangeListener< Number>() { public void changed(ObservableValue<? extends Number> ov, Number t, Number t1) { btn.setLayoutX(db.getValue()); } }); root.getChildren().add(btn); primaryStage.setScene(scene); primaryStage.show(); } }
Результат:

Строка с кодом
final DoubleBinding db = scene.widthProperty().subtract(150);создаёт переменную db и привязывает её значение к ширине формы минус 150 пикселов. К сожалению мы не можем просто привязать координаты кнопки к этой переменной (типа btn.layoutX.bind(db); что было бы очевидно и удобно). Поэтому к созданной переменной надо добавить Listener, в котором и обновлять координаты кнопки:
db.addListener(new javafx.beans.value.ChangeListener< Number>() { public void changed(ObservableValue<? extends Number> ov, Number t, Number t1) { btn.setLayoutX(db.getValue()); } });
Кроме того, binding накладывает дополнительные ограничения — если вы привяжете свойство к переменной то не сможете изменить простым присваиванием, будет выброшено исключение во время работы приложения.
Сам код API чрезмерно раздут, например класс SimpleDoubleProperty имеет иерархию наследования из шести классов и имплементирует девять интерфейсов. Это новое понимание характеристики simple.
В результате хорошая идея выглядит не очень привлекательно. Много кода, неявные ошибки. Хотя простая и понятная реализация связывания есть для многих языков и библиотек, в том числе и для Swing.
Использование CSS
Оформление интерфейса можно задавать с помощью CSS. Для этого нужно создать описание стилей, например такое:
.cssbutton { -fx-font: 16px "Serif"; -fx-padding: 10; -fx-background-color: #CCFF99; -fx-effect: dropshadow( one-pass-box , black , 12 , 0.0 , 1 , 1 ); }
— стиль cssbutton задаёт шрифт, отступы, заливку и эффект тени. Далее нужно подключить файл стилей к сцене:
scene.getStylesheets().add("cssjavafx/style.css");и добавить стиль к компоненту по имени:
btn.getStyleClass().add("cssbutton");Кнопка будет выглядеть примерно так:

Описание доступных в CSS свойств можно прочитать на странице документации.
Не очень понятно зачем вообще нужно задавать оформление стилями. Всё что можно задать стилями можно сделать и в коде, в CSS нельзя использовать переменные или вычисления, нельзя создавать новые свойства, стили сложно отлаживать.
Использование FXML
Компоновку форм можно задать в отдельном файле FXML и при старте приложения загрузить из него. Такой подход используется во многих средах разработки.
В Netbeans для этих целей предлагается создать проект с типом JavaFX FXML Application. Будет создана начальная форма примерно такого содержания:
package fxmljavafx; import java.net.*; import java.util.*; import javafx.event.*; import javafx.fxml.*; import javafx.scene.control.*; public class Sample implements Initializable { @FXML private Label label; @FXML private void handleButtonAction(ActionEvent event) { System.out.println("You clicked me!"); label.setText("Hello World!"); } @Override public void initialize(URL url, ResourceBundle rb) { // } }
Аннотация @FXML служит для обозначения переменных и функций доступных при загрузке из FXML-файла. Сам файл выглядит примерно так:
<?xml version="1.0" encoding="UTF-8"?> <?import java.lang.*?> <?import javafx.scene.*?> <?import javafx.scene.control.*?> <?import javafx.scene.layout.*?> <AnchorPane id="AnchorPane" prefHeight="200" prefWidth="320" xmlns:fx="http://javafx.com/fxml" fx:controller="fxmljavafx.Sample"> <children> <Button id="button" layoutX="126" layoutY="90" text="Click Me!" onAction="#handleButtonAction" fx:id="button" /> <Label id="label" layoutX="126" layoutY="120" minHeight="16" minWidth="69" prefHeight="16" prefWidth="69" fx:id="label" /> </children> </AnchorPane>
Как видно в примере, и метод handleButtonAction и свойство label заданные в коде, используются внутри FXML-описания формы.
Загрузка FXML-формы при старте приложения вызывается примерно так:
Parent root = FXMLLoader.load(getClass().getResource("Sample.fxml")); stage.setScene(new Scene(root)); stage.show();
Каких-то преимуществ подобное создание GUI не даёт. Визуального редактора нет, а править руками одинаково затратно как Java-код, так и FXML-описание.
Визуальный редактор
Форморисовалка есть в Visual Studio для VB и C#, в Adobe Flex, в Oracle JBuilder есть прекрасный редактор для Swing. Для JavaFX визульный редактор был объявлен ещё 2 года назад. Можно даже видео с ним посмотреть на странице http://www.flickr.com/photos/douglasbullard/3609213630/in/photostream/.
Но пока публичного релиза не было и неизвестно когда будет.
Ложка мёда в бочке дёгтя
Общая оценка скорей всего неудовлетворительная. Демо-приложения тормозят, внешний вид компонентов уступает Macintosh Aqua, Adone Flex или Microsoft Metro. Первая версия JavaFX появилась ещё в далёком 2008 году но Sun так и не смог довести технологию до ума.
Тем не менее, на последней JavaOne о JavaFX говорилось очень много. Будем надеяться что Oracle железной рукой направит разработчиков в нужное русло и заставит выдать наконец-то рабочий инструмент весто обещаний.
