JavaFx —— 创建项目与打包

一、IDEA 创建 JavaFX 项目

  使用 IDEA 新建项目,Generators 选择 JavaFx,此处选择 kotlin 作为开发语言,Maven 进行项目构建;点击 Next 进行下一步选择额外的一些依赖库,不需要勾选,直接点击 Finish 即可。

  项目创建成功之后,对应的文件结构如下,下面依次介绍下这些文件中的内容。

.
├── pom.xml
└── src
    └── main
        ├── kotlin
        │   ├── com
        │   │   └── jinyx
        │   │       └── javafxdemo
        │   │           ├── HelloApplication.kt
        │   │           └── HelloController.kt
        │   └── module-info.java
        └── resources
            └── com
                └── jinyx
                    └── javafxdemo
                        └── hello-view.fxml

pom.xml

  导入 javafx 依赖,jdk8 之后,javafx 从 jdk 中移除,因此在高版本的 java 中使用 javafx 需要导入相关依赖库。

<dependencies>
    <dependency>
        <groupId>org.openjfx</groupId>
        <artifactId>javafx-controls</artifactId>
        <version>19.0.2.1</version>
    </dependency>
    <dependency>
        <groupId>org.openjfx</groupId>
        <artifactId>javafx-fxml</artifactId>
        <version>19.0.2.1</version>
    </dependency>
</dependencies>

  build 插件,用于打包 javafx 项目;关于其配置参数可以查看javafx:jlink options说明。

<build>
    <plugins>
        <plugin>
            <groupId>org.openjfx</groupId>
            <artifactId>javafx-maven-plugin</artifactId>
            <version>0.0.8</version>
            <configuration>
                <mainClass>com.jinyx.javafxdemo/com.jinyx.javafxdemo.HelloApplication</mainClass>
                <launcher>appfx</launcher>
                <jlinkZipName>app</jlinkZipName>
                <jlinkImageName>app</jlinkImageName>
                <noManPages>true</noManPages>
                <stripDebug>true</stripDebug>
                <noHeaderFiles>true</noHeaderFiles>
                <compress>1</compress>
            </configuration>
        </plugin>
    </plugins>
</build>

HelloApplication.kt

  应用程序主入口,继承自 javafx.application.Application,并重写 start 方法显示主窗口;在 main 方法中需要调用 launch 方法启动应用。

import javafx.application.Application
import javafx.fxml.FXMLLoader
import javafx.scene.Parent
import javafx.scene.Scene
import javafx.stage.Stage

class HelloApplication : Application() {
    override fun start(stage: Stage) {
        val parent = FXMLLoader.load<Parent>(HelloApplication::class.java.getResource("hello-view.fxml"))
        val scene = Scene(parent, 320.0, 240.0)
        stage.title = "Hello!"
        stage.scene = scene
        stage.show()
    }
}

fun main(args: Array<String>) {
    Application.launch(HelloApplication::class.java, *args)
}

  注意 kotlin 需要调用 launch(Class<? extends Application> clazz, String... args) 方法,而调用 launch(String... args) 将会出现如下错误;java 代码正常。

Error: class com.jinyx.javafxdemo.HelloApplicationKt is not a subclass of javafx.application.Application

  重写的 start 方法中,通过 FXMLLoader 加载 fxml 页面,在通过 stage.show 完成主窗口的显示

hello-info.fxml

  页面布局如下,采用 xml 标签语言,格式和 Android 类似,不过对应组件的样式,需要配合使用 css;后续学习组件中再深入。

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.geometry.Insets?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.layout.VBox?>

<?import javafx.scene.control.Button?>
<VBox alignment="CENTER" spacing="20.0" xmlns:fx="http://javafx.com/fxml"
      fx:controller="com.jinyx.javafxdemo.HelloController">
    <padding>
        <Insets bottom="20.0" left="20.0" right="20.0" top="20.0"/>
    </padding>

    <Label fx:id="welcomeText" style="-fx-font-family: 'PingFang SC'"/>
    <Button text="Hello!" style="-fx-font-family: 'PingFang SC'" onAction="#onHelloButtonClick"/>
</VBox>

  fx:controller="com.jinyx.javafxdemo.HelloController" 指定需要绑定的 Controller,可以在控制器里面处理业务逻辑

HelloController.kt

  使用 @FXML 注解将组件和点击事件与 fxml 之间进行绑定,从而获取到 Label 组件以及提供 Button 点击事件

package com.jinyx.javafxdemo

import javafx.fxml.FXML
import javafx.scene.control.Label

class HelloController {
    @FXML
    private lateinit var welcomeText: Label

    @FXML
    private fun onHelloButtonClick() {
        welcomeText.text = "Welcome to JavaFX Application!"
    }
}

module-info.java

module com.jinyx.javafxdemo {
    requires javafx.controls;
    requires javafx.fxml;
    requires kotlin.stdlib;


    opens com.jinyx.javafxdemo to javafx.fxml;
    exports com.jinyx.javafxdemo;
}

  使用关键字 module,其属于 java9 的新特性;模块名的命名规则同包名,如下的 com.jinyx.javafxdemo;生命的模块主题可以为空,也可以包含如下指令:

  • exports: 声明指定的包可以被其他模块访问,也可以增加 to 指定其他模块访问
  • requires: 指定需要依赖的模块,例如上面指定需要的 javafx.controlsjavafx.fxmlkotlin.stdlib
  • provides
  • uses
  • opens: 对比 exports,允许在运行时可以使用反射访问;而 exports 编译期、运行期都只能访问 public 成员,但是使用 opens 需要 exports 配合使用,否则引用模块在编译期找不到该包

二、运行程序

  打开右侧 Maven 面板,双击 Pugins —> javafx —> javafx:run 或者命令行运行 mvn javafx:run 即可启动程序;
  如果出现如左边的乱码问题,可以在 fxml 中为组件指定字体,例如上面 fxml 代码中的 style="-fx-font-family: 'PingFang SC'"

三、打包应用

  打开右侧 Maven 面板,双击 Pugins —> javafx —> javafx:jlink 或者命令行运行 mvn javafx:jlink;打包成功之后,将会在 target/app/bin 目录下生成可执行程序 appfx; 此处的 app 和 appfx 的名称,分别对应在 pom.xml 中,javafx-maven-plugin 插件定义的 jlinkImageNamelauncher

  但是此时打包生成的 appfx 可执行程序,运行时会打开一个终端窗口,因此需要进一步将其打包成对应平台的安装包或应用程序,例如 Windows 平台的 exe 程序、macOS 平台的 dmg 安装包;此处使用 jdk14 之后自带的 jpackage 进行打包,打包指令如下:

jpackage --name appName --module com.jinyx/com.jinyx.DesktopAppKt --runtime-image ./target/app/ --icon /Users/jinyx/img_logo.icns --type app-image
  • --name: 打包的应用程序名
  • --module: 对应 javafx-maven-plugin 中配置的 mainClass
  • --runtime-image: jlink 打出的包
  • --icon: 指定应用图标,macOS 测试需要 icns 类型图标;png 转 icns
  • --type: 指定打包类型;例如我当前系统可选的值有 {"app-image", "dmg", "pkg"}

  上面的参数以及其他参数说明都可以通过 jpackage -h 进行查看

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容