Auto updating launcher for JavaFX Applications. Combined with JavaFX native packaging, you get a native installer with automatic app updates.
You can see the launcher in action in this Demo Application.
- Maven Example with pom.xml
- Gradle Example with build.gradle
Check out the changelog for a list of all updates to the project.
See the launcher in action in this short screencast. There is also a video about customizing the update user interface.
FXLauncher is a 18Kb jar that can be used to boot your application. It knows the location of your application repository where you host all the app resources.
See a manifest example here. FXLauncher will look up the remote repository to check for a newer version of the manifest on each startup.
After the manifest is retrieved, FXLauncher synchronizes every file mentioned in the manifest while providing the user with a progress indicator. After all resources are in sync, a classloader is initialized with all the resources from the manifest.
Lastly, the application entry point retrieved from the manifest is invoked. Everything happens in-JVM, no restarts needed.
Before each run, the launcher will synchronize all resources and seamlessly launch an always updated application.
See the QuickStart projects at the top of the README for information on integrating FXLauncher in your build system.
FXLauncher can also be used to launch an application at an arbitrary url by specifying the --app
parameter at startup:
java -jar fxlauncher.jar --app=http://remote/location/app.xml
Alternatively (or in combination with --app...
), you can override the uri attribute in the manifest (app.xml
) so that both app.xml
and all resources are loaded from the specified uri. This is especially useful for testing the complete setup locally or from a staging environment.
java -jar fxlauncher.jar --uri=http://remote/location/
The two parameters also work in tandem, allowing you to load a specified manifest from one URL and override its uri.
java -jar fxlauncher.jar --app=http://remote/location/app.xml --uri=http://remote/location/
Note: All parameters (including these) are passed on to your application. So please ensure that your parameters have a different name if they carry different data.
To load the application FXLauncher creates a new classloader fed with the classes of your manifest. This works as long as none of the classes uses the system classloader to load one of your classes or resources which is not aware of the application classes then.
One example is Java's java.net.URL
class which uses Class.forName
to load a custom handler, which uses the classloader of
the callee which is the system classloader then.
To workaround that problem you could setup a special FXLauncher classloader which allows a different strategy how to feed the classpath with additional classpath entries.
java -Djava.system.class.loader=fxlauncher.FxlauncherClassCloader -jar fxlauncher.jar --app=http://remote/location/app.xml
FXLauncher allows you to run in headless mode and thus be used not only for JavaFX applications, but also for e.g. services which run without any gui at all.
For this to work you could use fxlauncher.HeadlessMainLauncher
java -classpath fxlauncher.jar fxlauncher.HeadlessMainLauncher --app=http://remote/location/app.xml
Notice: WhatsNew
is not supported.
The app uri can be a file:// based url instead of an web uri if your application is hosted on a common network location for your users.
The native installer does not contain any application code, only the launcher. There is
no need to rebuild your native installer when you update your project, simply run the deploy-app
goal
and all users will run the newest version on their next startup. Point users to the fxlauncher.jar
or
to a native installer if you wish.
Check out these prebuilt installers for a more complex demo application
By default, the artifacts are downloaded to the current working directory. This is usually fine for native installers, but if you distribute your application via just the launcher jar, you might want to specify where the downloaded artifacts land. See the cache dir documentationfor more information.
It's worth noting that the two package alternatives for Windows, (EXE and MSI) have different install location defaults.
While EXE will default to %APPDATALOCAL%, the MSI installer will default to %ProgramFiles%. If you use the MSI installer you
might therefore need to specify the cache dir parameter as cacheDir 'USERLIB/MyApp'
to make sure that the launcher has
write access to download the artifacts for your application.
Read more about Java Packager in the official documentation:
https://docs.oracle.com/javase/8/docs/technotes/guides/deploy/self-contained-packaging.html
FXLauncher will not download a remote version if the local version is newer. This is controlled
by comparing a timestamp in the manifest. Specifying --accept-downgrades=true
as the last argument to CreateManifest will
allow you to make sure that the version you have published will always be used by your clients even if they have a newer version installed.
This option is also available in the Gradle plugin as acceptDowngrades
.
If you are behind a proxy that does a man in the middle attack to snoop on ssl connections, you can use the
--ignoressl
commandline option to ignore the ssl errors this will generate.
Normally FXlauncher will try to launch the application even if there were errors
during the update of the files from the server. Often this will work using the files
already in the local cache. With the option --stopOnUpdateErrors
this will not be done.
The error will be shown and the launcher will exit.
You can have FXLauncher show a whats new dialog. This dialog will
only be shown when FXLauncher has to download new files from the server. The html content will be shown in a WebView
. Specify --whats-new=filename.html
The file needs to be copied into the jar like so:
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<version>1.4.0</version>
<executions>
<execution>
<id>copy-whatsnewfile</id>
<phase>package</phase>
<goals>
<goal>exec</goal>
</goals>
<configuration>
<executable>jar</executable>
<workingDirectory>${app.dir}</workingDirectory>
<arguments>
<argument>uf</argument>
<argument>fxlauncher.jar</argument>
<argument>-C</argument>
<argument>${project.basedir}/src/main/resources</argument>
<argument>whatsnew.html</argument>
</arguments>
</configuration>
</execution>
</executions>
</plugin>
By default the update screen will remain visible until the primary stage is shown. If your application does not show anything on the primary
stage at startup, the update screen will stay visible until it does. You can pass in --lingering-update-screen=false
to hide it immediately
after the update process has completed.
It is also possible to embed the launchar jar in a native installer system like Advanced Installer - same approach as above, but without using javapackager. With this approach, you can choose whether to include a JRE or have the installer software preinstall it. Again, you are only distributing the launcher with the native installer, the rest is pulled down on demand.
FXLauncher uses a custom classloader to dynamically load the synchronized resources. This classloader is
then made available to the FXMLLoader
. You can access it via FXMLLoader.getDefaultClassLoader()
.
FXLauncher supports filtering of resources for the running platform. Any resource
that ends with -[mac|win|linux].jar
will only be downloaded and put on the classpath on the corresponding
platform. The manifest enforces this though the os
attribute in app.xml
.
If you need to load native libraries before the custom class loader kicks inn, specify the --preload-native-libraries=
parameter
to CreateManifest. It supports a comma separated list of libraries to load. Remember: No extensions, just the library name.
There are two ways to customize the appearance of the update UI. Either you can configure the supported style properties in the manifest, or you can provide a custom implementation of the UIProvider to completely customize the UI. Have a look at this Custom UI Demo Project for more information about customizing the updater.