从 0 到 1 搞一个 Compose Desktop 版本的玩天气之打包
大家好,前两篇文章大概介绍了下上手 Compose Desktop
和自定义绘制时遇到的一些问题,项目的最终实现效果如下:
视频
代码写好了,该弄的动画也弄了,该请求的网络数据也请求了,该实现的效果也都实现好了,但是!!!咱们得打包出来啊!不打包出来别人如何使用呢?难道说别人想用你开发的桌面应用,结果你给他说你先下载一个 IntelliJ Idae
,然后下载下我的源码,之后把环境配置好,最后运行就可以了!如果下次再想用的时候再运行一次就好了!
这说的是人话嘛😂,肯定不能这样,所以一定要打包!由于 Compose Desktop
不止可以运行在 Mac
中,还可以运行在 Windows
和 Linux
中,所以需要打多个包。那使用 Compose Desktop
应该如何打包呢?且听我慢慢道来!
基本用法
插件中的基本配置单元是一个 application
,application
是什么呢?在第一篇文章中也提到了,就是在 build.gradle.kts
文件中的代码,咱们再来看下:
compose.desktop {
application {
mainClass = "MainKt"
nativeDistributions {
targetFormats(TargetFormat.Dmg, TargetFormat.Msi, TargetFormat.Deb)
packageName = "Demo"
packageVersion = "1.0.0"
}
}
}
一个 application
定义了一组最终二进制文件的共享配置。换句话说,application
DSL 允许将一堆文件连同 JDK 分发包打包成一组各种格式(.dmg
、.deb
、.msi
、.exe
等)的压缩二进制安装程序。
该插件创建以下任务:
package<FormatName>
(例如packageDmg
或packageMsi
)用于将应用程序打包成相应的格式。这块需要注意的是,目前没有交叉编译支持,因此只能使用特定操作系统构建格式(例如,要构建.dmg
就必须使用 macOS)。默认情况下会跳过与当前操作系统不兼容的任务。packageDistributionForCurrentOS
是一个生命周期任务,聚合了应用程序的所有包任务。packageUberJarForCurrentOS
用于创建单个 jar 文件,其中包含当前操作系统的所有依赖项。run
用于在本地运行应用程序。需要定义一个mainClass
包含该main
函数的类。请注意,run
将启动具有完整运行时的非打包 JVM 应用程序。这比创建具有最小运行时间的紧凑二进制映像更快、更容易调试。要运行最终的二进制图像,需要改用runDistributable
。createDistributable
用于在不创建安装程序的情况下创建预打包的应用程序映像和最终应用程序映像。runDistributable
用于运行预先打包的应用程序映像。- 只有在脚本中使用
block
或property
时才会创建任务。
光这么说其实有点懵,来一张图大家就明白我说的是什么了!
是不是有点恍然大明白的感觉!直接点击 IntelliJ IDEA
右侧边栏的 Gradle
,就会出现这个侧边栏,然后点击 Task
中的 compose desktop
就会出现上面描述的那些任务。
打包配置
Compose Desktop
打包有很多的配置项,下面来分别看下。
配置包含的 JDK 模块
Gradle 插件使用 jlink 通过仅包含必要的 JDK 模块来最小化可分发的大小。
此时,Gradle 插件不会自动确定必要的 JDK 模块。未能提供必要的模块不会导致编译问题,但会导致在运行时出现 ClassNotFoundException
的错误。
如果在运行打包的应用程序或任务时遇到 ClassNotFoundException
,可以使用DSL 方法runDistributable
来配置包含额外的 JDK 模块,需要使用 modules
来配置。
可以通过手动或运行 suggestModules
任务来确定哪些模块是必需的。suggestModules
使用 jdeps 静态分析工具来确定可能缺少的模块。
如果安装包的大小不重要的话,可以使用 includeAllModules
DSL 属性简单地包括所有运行时模块作为替代。
compose.desktop {
application {
nativeDistributions {
modules("java.sql")
// alternatively: includeAllModules = true
}
}
}
这块在我打包的时候搜了好久!最后在 Issue
中找到了解决方案!
可用格式
以下格式可用于支持的操作系统:
- macOS —
.dmg
(TargetFormat.Dmg
)、.pkg
(TargetFormat.Pkg
) - Windows —
.exe
(TargetFormat.Exe
)、.msi
(TargetFormat.Msi
) - Linux —
.deb
(TargetFormat.Deb
)、.rpm
(TargetFormat.Rpm
)
指定包版本
由于可以打多种不同的包,也有可能需要区分不同的版本,所以可以指定包的版本。如何指定的呢?来看代码:
compose.desktop {
application {
nativeDistributions {
packageVersion = "1.0.0"
linux {
packageVersion = "1.0.0"
debPackageVersion = "1.0.0"
rpmPackageVersion = "1.0.0"
}
macOS {
packageVersion = "1.1.0"
dmgPackageVersion = "1.1.0"
pkgPackageVersion = "1.1.0"
packageBuildVersion = "1.1.0"
dmgPackageBuildVersion = "1.1.0"
pkgPackageBuildVersion = "1.1.0"
}
windows {
packageVersion = "1.2.0"
msiPackageVersion = "1.2.0"
exePackageVersion = "1.2.0"
}
}
}
}
必须为本机分发包指定包版本,还可以使用以下 DSL 属性(按优先级降序排列):
nativeDistributions.<os>.<packageFormat>PackageVersion
指定单个包格式的版本;nativeDistributions.<os>.packageVersion
指定单个目标操作系统的版本;nativeDistributions.packageVersion
指定所有包的版本;
对于 macOS,还可以使用以下 DSL 属性指定构建版本(按优先级降序排列):
nativeDistributions.macOS.<packageFormat>PackageBuildVersion
指定单一包格式的构建版本;nativeDistributions.macOS.packageBuildVersion
为所有 macOS 包指定构建版本。
需要注意的是,版本必须遵循以下规则:
-
dmg
和pkg
:格式为 MAJOR.MINOR.PATCH其中:
MAJOR
是一个 > 0 的整数;MINOR
是一个可选的非负整数;PATCH
是一个可选的非负整数; -
msi
和exe
:格式为 MAJOR.MINOR.BUILD其中:
MAJOR
是一个非负整数,最大值为255;MINOR
是一个非负整数,最大值为255;BUILD
是一个非负整数,最大值为65535; -
rpm
:版本不得包含-
(破折号)字符。 -
deb
:格式为 EPOCH:UPSTREAM_VERSION-DEBIAN_REVISION其中:
EPOCH
是一个可选的非负整数;UPSTREAM_VERSION
只包含字母数字和字符.
,+
,-
,~
,必须以数字开头;DEBIAN_REVISION
是可选的,可能只包含字母数字和字符.
,+
,~
。
自定义 JDK 版本
由于该插件使用jpackage
,所以最低得使用 JDK 15。
JAVA_HOME
环境变量指向兼容的 JDK 版本。javaHome
通过 DSL 设置:
compose.desktop {
application {
javaHome = System.getenv("JDK_15")
}
}
自定义输出目录
Compose Desktop
默认的打包路径在 /build/compose/binaries/main/app 中,如果想修改下打包路径的话,需要修改下配置:
compose.desktop {
application {
nativeDistributions {
outputBaseDir.set(project.buildDir.resolve("customOutputDir"))
}
}
}
自定义基本数据
DSL 块中提供以下属性 nativeDistributions
:
packageName
— 应用程序名称(默认值:Gradle 项目名称);version
— 应用程序的版本(默认值:Gradle 项目的版本);description
— 应用程序的描述(默认值:无);copyright
— 应用程序的版权(默认值:无);vendor
— 应用程序的供应商(默认值:无);licenseFile
— 应用程序的许可证(默认值:无)。
compose.desktop {
application {
nativeDistributions {
packageName = "PlayWeather"
version = "1.1.0"
description = "PlayWeather"
copyright = "© 2022 My Name. All rights reserved."
vendor = "Example vendor"
licenseFile.set(project.file("LICENSE.txt"))
}
}
}
这块大家可以根据需求来定义这些数据,如不需要不写即可。
特定平台选项
需要使用相应的 DSL 块设置特定于平台的选项,使用方法就是上面 maxOS
、windows
、linux
,不同平台可配置的选项都不太一样!
-
所有平台
-
iconFile.set(File("PATH_TO_ICON"))
— 应用程序特定于平台的图标的路径。 -
packageVersion = "1.0.0"
— 特定于平台的包版本。 -
installationPath = "PATH_TO_INSTALL_DIR"
默认安装目录的绝对或相对路径;在 Windows 上dirChooser = true
,可用于启用在安装过程中自定义路径。
-
-
Linux
-
packageName = "custom-package-name"
覆盖默认的应用程序名称; -
debMaintainer = "maintainer@example.com"
— deb 包维护者的电子邮件; -
menuGroup = "my-example-menu-group"
— 应用程序的菜单组; -
appRelease = "1"
— rpm 包的发布值,或 deb 包的修订值; -
appCategory = "CATEGORY"
— rpm 包的组值,或 deb 包的部分值; -
rpmLicenseType = "TYPE_OF_LICENSE"
— rpm 包的一种许可证; -
debPackageVersion = "DEB_VERSION"``Specifying package version
— 特定于 deb 的包版本; -
rpmPackageVersion = "RPM_VERSION"``Specifying package version
— 特定于 rpm 的软件包版本;
-
-
MacOS
-
bundleID — 唯一的应用标识符;
- 只能包含字母数字字符 (
A-Z
,a-z
,0-9
)、连字符 (-
) 和句点 (.
) 字符; com.mycompany.myapp
建议使用反向 DNS 表示法(例如);
- 只能包含字母数字字符 (
-
packageName
— 应用名称; -
dockName
— 显示在菜单栏、“关于”菜单项、停靠栏等中的应用程序名称。packageName
等于默认情况下的名称; -
signing
,notarization
,provisioningProfile
, 和runtimeProvisioningProfile
— 详见相应教程; -
appStore = true
— 为 Apple App Store 构建和签名。至少需要 JDK 17; -
appCategory
— Apple App Store 的应用类别。默认值是public.app-category.utilities
; -
entitlementsFile.set(File("PATH_TO_ENTITLEMENTS"))
— 包含签名时使用的权利的文件路径; -
runtimeEntitlementsFile.set(File("PATH_TO_RUNTIME_ENTITLEMENTS"))
— 包含在签署 JVM 运行时时使用的权利的文件路径; -
dmgPackageVersion = "DMG_VERSION"
— 一个特定于 dmg 的包版本(详见参考资料部分); -
pkgPackageVersion = "PKG_VERSION"
— 特定于 pkg 的包版本(详情请参阅参考资料部分); -
packageBuildVersion = "DMG_VERSION"
— 包构建版本(详见参考资料部分); -
dmgPackageBuildVersion = "DMG_VERSION"
— 特定于 dmg 的软件包构建版本(详情请参阅参考资料部分); -
pkgPackageBuildVersion = "PKG_VERSION"
— 特定于 pkg 的包构建版本; -
infoPlist
— 链接到别的程序。
-
-
Linux
console = true
为应用程序添加一个控制台启动器;dirChooser = true
允许在安装过程中自定义安装路径;perUserInstall = true
允许在每个用户的基础上安装应用程序menuGroup = "start-menu-group"
将应用程序添加到指定的开始菜单组;upgradeUuid = "UUID"
— 一个唯一的 ID,当更新的版本比安装的版本更新时,它使用户能够通过安装程序更新应用程序。对于单个应用程序,该值必须保持不变;msiPackageVersion = "MSI_VERSION"
— 特定于 msi 的软件包版本;exePackageVersion = "EXE_VERSION"
— 特定于 pkg 的包版本
修改应用图标
这块为什么要单独拿出一大块来说呢?因为这个问题中困扰了我好久。。。所以才。。。。
还记得之前文章中说了 Window
可组合项中可以设定 Icon
么,但当时说的时候专门说了此 Icon
并非应用程序的图标!因为应用程序图标需要以特定于操作系统的格式提供:
.icns
对于 macOS.ico
适用于 Windows.png
对于 Linux
看下代码吧:
compose.desktop {
application {
nativeDistributions {
macOS {
iconFile.set(project.file("icon.icns"))
}
windows {
iconFile.set(project.file("icon.ico"))
}
linux {
iconFile.set(project.file("icon.png"))
}
}
}
}
Linux
中的 png
格式的图片我们很常见,但是 Mac
和 Windows
中的格式是什么鬼。。。没见过啊!
如果知道这两种文件格式的话大家直接跳过后面的部分即可,这里还需要注意的是这里的文件路径指的是项目根目录。
ICNS 文件
- 什么是
.icns
?
.icns
是苹果的 macOS 操作系统的 App 图标文件的扩展名,大家在 macOS 的 Desktop 桌面
、Finder 访达
、Dock 程序坞
等看到应用程序的外观就是由一个内置在此 App 内部的扩展名为.icns
的文件实现的。
可以通过鼠标“右键”点击 App - “显示包内容” - 进入 Contents
目录 - 进入Resources
目录,然后在目录下可以找到名为 Appicon.icns
或其他后缀为 .icns
的一个图标文件。
- 如何创建
.icns
扩展名的图标文件?
a. 准备一张图片,重命名为 icon.png
,其他大小尺寸可以通过终端命令生成;
b. 通过鼠标右键或者命令,创建一个名为 icons.iconset
的文件夹
mkdir icons.iconset
c. 通过”终端“来快速创建各种不同尺寸要求的图片文件
sips -z 512 512 icon.png -o icons.iconset/icon_512x512.png
d. ”终端“中运行下面的命令,就可以获得名为icon.icns
的图标文件了
iconutil -c icns icons.iconset -o icon.icns
注意:icon.png
图片文件和 icons.iconset
文件夹要保存在同一级目录下,”终端“启动后切换到相同目录。
ICO 文件
ico
文件是 Windows
系统的应用图标格式,我也不会制作,但找到了一个制作 ico
的网站:
https://www.butterpig.top/icopro/
需要的话可以进去制作。
混淆
从 Compose Desktop
1.2 版本开始,Compose Gradle
插件支持开箱即用的 ProGuard。ProGuard
是一个众所周知的用于缩小和混淆的开源工具Guardsquare。
Gradle插件为每个对应的默认打包任务提供了发布任务:
默认任务(没有 ProGuard) | 发布任务(带 ProGuard) | 描述 |
---|---|---|
createDistributable | createReleaseDistributable | 使用捆绑的 JDK 和资源创建应用程序映像 |
runDistributable | runReleaseDistributable | 使用捆绑的 JDK 和资源运行应用程序映像 |
run | runRelease | jar 使用 Gradle JDK运行非打包应用程序 |
package<FORMAT_NAME> | packageRelease<FORMAT_NAME> | 将应用程序映像打包到<FORMAT_NAME> 文件中 |
packageForCurrentOS | packageReleaseForCurrentOS | 将应用程序映像打包成与当前操作系统兼容的格式 |
notarize<FORMAT_NAME> | notarizeRelease<FORMAT_NAME> | 上传<FORMAT_NAME> 用于公证的应用程序图像(仅限 macOS) |
checkNotarizationStatus | checkReleaseNotarizationStatus | 检查公证是否成功(仅限 macOS) |
默认配置添加了一些 ProGuard 规则:
- 缩小应用程序图像,即删除未使用的类;
compose.desktop.application.mainClass
用作入口点;keep
避免破坏Compose
运行时的一些规则。
在许多情况下,获得缩小的 Compose
应用程序不需要任何额外的配置。但是,有时 ProGuard
可能无法跟踪字节码中的某些用法(例如,如果通过反射使用类,则可能会发生这种情况)。如果遇到仅在 ProGuard
处理后才会发生的问题,可能需要添加自定义规则。为此,通过 DSL 指定配置文件:
compose.desktop {
application {
buildTypes.release.proguard {
configurationFiles.from(project.file("compose-desktop.pro"))
}
}
}
可以参考 Guardsquare 关于 ProGuard 规则和配置选项的综合手册。
默认情况下禁用混淆。要启用它,请通过 Gradle DSL 设置以下属性:
compose.desktop {
application {
buildTypes.release.proguard {
obfuscate.set(true)
}
}
}
混淆需要根据需要使用了,如果需要控制包体积的话尽量还是打开,可以减小包体积,还会减小代码泄漏的可能性;反之打不打开都行!
实际操作
上面都是理论知识,咱们得实操啊!有基础知识之后实操就很简单了,先来看下 build.gradle.kts
中的配置项吧:
compose.desktop {
application {
mainClass = "MainKt"
nativeDistributions {
targetFormats(TargetFormat.Dmg, TargetFormat.Exe, TargetFormat.Msi, TargetFormat.Deb)
packageName = "PlayWeather"
packageVersion = "1.0.0"
description = "Play Weather App"
copyright = "© 2022 My Name. All rights reserved."
vendor = "Lenovo"
licenseFile.set(project.file("LICENSE.txt"))
modules("java.instrument", "java.management", "java.naming", "java.sql", "jdk.unsupported")
linux {
packageVersion = "1.0.0"
debPackageVersion = "1.0.0"
rpmPackageVersion = "1.0.0"
// 设置图标
iconFile.set(project.file("launcher/icon.png"))
}
macOS {
packageVersion = "1.1.0"
dmgPackageVersion = "1.1.0"
pkgPackageVersion = "1.1.0"
dockName = "PlayWeather"
packageBuildVersion = "1.1.0"
dmgPackageBuildVersion = "1.1.0"
pkgPackageBuildVersion = "1.1.0"
// 设置图标
iconFile.set(project.file("launcher/icon.icns"))
}
windows {
packageVersion = "1.2.0"
msiPackageVersion = "1.2.0"
exePackageVersion = "1.2.0"
// 设置图标
iconFile.set(project.file("launcher/icon.ico"))
}
}
buildTypes.release.proguard {
obfuscate.set(false)
configurationFiles.from(project.file("proguard-rules.pro"))
}
}
}
这里的配置项就不多说了,上面都有过介绍,下面来打包吧!
点击上图中蓝色箭头标注的进行打包,上面说过了,不能跨系统打包,Mac
只能打 Mac
中使用的包。。双击执行 packageDmg
任务:
没问题的话大概会出现上图的样子,由于没有配置自定义包路径,所以还在默认文件中,按照上面所描述的路径进行查看:
复制文件路径,在访达中打开:
双击进行安装即可:
然后在资源库中找到应用双击打开应该会遇到下面的错误:
这时点击取消,然后打开设置 -> 隐私与安全性,往下滑:
点击箭头标注的“仍要打开”按钮,会让你输入电脑的密码,输入完成后会弹出下面的对话框:
点击打开,这时应用就能正常使用了。
苹果端就不打 release
包了,还需要苹果的开发者账号那一大堆。。。目前先能正常在 Mac
中运行吧!
总结
本文大概总结了下使用 Compose Desktop
如何进行打包,此项目所有代码都放到了 Github
中。
Github
地址:https://github.com/zhujiang521/PlayWeather/tree/desktop
如果文中写的有误,欢迎在评论区提出,咱们一起探讨。
文章如果能帮助到大家,哪怕是一点,我也非常高兴,先这样。