Java-GI is a tool for generating GObject-Introspection bindings for Java. The generated bindings use the Foreign Function & Memory API (JEP 454) to access native resources from Java, with wrapper classes based on GObject-Introspection to offer an elegant API. Java-GI version 0.11.* generates bindings to develop Java applications for libraries, based of the versions in GNOME Platform 47:
Library | Version |
---|---|
GLib | 2.82 |
GTK | 4.16 |
LibAdwaita | 1.6 |
GStreamer | 1.24 |
GtkSourceview | 5.14 |
WebkitGtk | 2.46 |
You can generate Java bindings for other libraries with the java-gi
command-line tool on the GIR (introspection data) file.
Please note that Java-GI is still under active development, and the API is subject to unannounced changes. Feedback is welcome!
For more information, visit the Java-GI website.
Let's write a small "Hello World" GTK application:
import org.gnome.gtk.*;
import org.gnome.gio.ApplicationFlags;
public class HelloWorld {
public static void main(String[] args) {
new HelloWorld(args);
}
private final Application app;
public HelloWorld(String[] args) {
app = new Application("my.example.HelloApp", ApplicationFlags.DEFAULT_FLAGS);
app.onActivate(this::activate);
app.run(args);
}
public void activate() {
var window = new ApplicationWindow(app);
window.setTitle("GTK from Java");
window.setDefaultSize(300, 200);
var box = Box.builder()
.setOrientation(Orientation.VERTICAL)
.setHalign(Align.CENTER)
.setValign(Align.CENTER)
.build();
var button = Button.withLabel("Hello world!");
button.onClicked(window::close);
box.append(button);
window.setChild(box);
window.present();
}
}
Add the Gtk dependency to your build script, for example with Gradle:
repositories {
mavenCentral()
}
dependencies {
implementation 'io.github.jwharm.javagi:gtk:0.11.2'
}
The result:
You can find some examples here. Each example can be separately built and run with gradle run
:
Web Browser | Peg Solitaire | Calculator | Notepad |
Nearly all types, functions and parameters defined in the GIR files are supported by Java-GI. Even complex function signatures with combinations of arrays, callbacks, out-parameters and varargs are available in Java.
Some interesting features of the bindings that Java-GI generates:
Memory management of native resources is automatically taken care of. Java-GI uses GObject toggle references to dispose the native object when the Java instance is garbage-collected, and releases all other memory allocations (for strings, arrays and structs) after use.
All API docstrings are translated into Javadoc, so they are directly available in your IDE.
As an example, the generated documentation of gtk_button_get_icon_name
contains links to other methods, and specifies the return value. This is all translated to valid Javadoc.
The Javadoc is published online.
GObject classes are available as Java classes (obviously). The GObject TypeClass definition is an inner class in the Java class.
Interfaces are mapped to Java interfaces, using default
interface methods to call native methods.
Type aliases (typedef
s in C) for classes, records and interfaces are represented in Java with a subclass of the original type. Aliases for primitive types such as int
or float
are represented by simple wrapper classes.
Enumerations are represented as Java enum
types, and flag parameters are mapped to EnumSet
.
Most classes have one or more constructors. However, constructors in GTK are often overloaded, and the name contains valuable information for the user. Java-GI therefore maps constructors named "new" to regular Java constructors, and generates static factory methods for all other constructors:
// gtk_button_new
var button1 = new Button();
// gtk_button_new_with_label
var button2 = Button.withLabel("Open...");
// gtk_button_new_from_icon_name
var button3 = Button.fromIconName("document-open");
Some struct types (called "records" in GObject-Introspection) don't have constructors, because in C these are meant to be stack-allocated. An example is Gdk.RGBA
. Java-GI adds constructors that will allocate a new struct in an Arena of your choice. You can either allocate an empty struct (var color = new RGBA();
) and fill in the values later, or pass the values immediately: var purple = new RGBA(0.9f, 0.1f, 0.9f, 1.0f);
Signals are mapped to type-safe methods and objects in Java. (Detailed signals like notify
have an extra String
parameter.) A signal can be connected to a lambda expression or method reference:
var button = Button.withLabel("Close");
button.onClicked(window::close);
For every signal, a method to connect (e.g. onClicked
) and emit the signal (emitClicked
) is included in the API. New signal connections return a SignalConnection
object, that allows you to disconnect, block and unblock a signal, or check whether the signal is still connected.
Functions with callback parameters are supported too. The generated Java bindings contain @FunctionalInterface
definitions for all callback functions to ensure type safety.
Closures are marshaled to Java methods using reflection.
You can easily register a Java class as a GType:
public class MyWidget extends Widget {
public static Type gtype = Types.register(MyWidget.class);
// Construct new instance
public static MyWidget newInstance() {
return GObject.newInstance(gtype);
}
// Default constructor, used by Java-GI for marshaling
public MyWidget(MemorySegment address) {
super(address);
}
}
You can define custom GObject properties and signals using annotations. The following example defines an int
property named "lives"
and a "game-over"
signal with a String
parameter:
@Property
public int getLives() {
return lives;
}
@Property
public void setLives(int value) {
this.lives = value;
if (value == 0)
emit("game-over", player.name());
}
@Signal
public interface GameOver {
void apply(String playerName);
}
Java classes can implement interfaces and override methods without any additional effort. When implementing methods from an interface (or overriding virtual methods from a parent class), Java-GI will register it in the GObject type system, so native code will call your Java method too. See for example this implementation of the ListModel
interface, or read the Java-GI documentation for an overview of all the possibilities.
A class with a @GtkTemplate
annotation will be registered as a Gtk composite template class:
@GtkTemplate(name="HelloWindow", ui="/my/example/hello-window.ui")
public class HelloWindow extends ApplicationWindow {
private static Type gtype = TemplateTypes.register(HelloWindow.class);
@GtkChild(name="header_bar")
public HeaderBar header;
@GtkChild
public Label label;
@GtkCallback
public void buttonClicked() {
...
}
...
In the above example, the header
and label
fields and the buttonClicked
callback function are all declared the hello-window.ui
file.
You can read more about template classes in the documentation.
Java-GI takes care of marshaling Java values from and to native values. When working with arrays, Java-GI will automatically copy native array contents from and to a Java array, marshaling the contents to the correct types along the way. A null
terminator is added where applicable. You also don't need to specify the array length as a separate parameter.
Nullability of parameters (as defined in the GObject-introspection attributes) is indicated with @Nullable
and @NotNull
attributes, and checked at runtime. The nullability attributes are imported from Jetbrains Annotations (as a compile-time-only dependency).
Variadic functions (varargs) are supported too:
Dialog d = Dialog.withButtons(
"Test dialog",
window,
DialogFlags.MODAL,
"Accept",
ResponseType.ACCEPT,
"Cancel",
ResponseType.CANCEL,
null
);
d.show();
Out-parameters are mapped to a simple Out<T>
container-type in Java, that offers typesafe get()
and set()
methods to retrieve or modify the value.
File file = ...
Out<byte[]> contents = new Out<byte[]>();
file.loadContents(null, contents, null));
System.out.printf("Read %d bytes%n", contents.get().length);
You can construct an object with properties using a Builder pattern. In the "Hello World" app above, it's used to create a Box
. It can be used for any other type too:
var window = ApplicationWindow.builder()
.setApplication(this)
.setTitle("Window")
.setDefaultWidth(300)
.setDefaultHeight(200)
.build();
Java-GI generates builders for all classes. In a builder, you can set the properties of the class, its parents, and all implemented interfaces.
GError
parameters are mapped to Java GErrorException
s.
try {
file.replaceContents(contents, null, false, FileCreateFlags.NONE, null, null);
} catch (GErrorException e) {
... // handle exception
}
The published bindings are cross-platform: You can use the same jar on all supported operating systems (64-bit Linux, Windows and MacOS) provided that the native libraries are installed. Platform-specific types and methods (like Gtk.PrintUnixDialog
) check the operating system at runtime and throw an UnsupportedPlatformException
when necessary.
Java-GI offers a command-line utility to generate bindings for any library that supports GObject-Introspection. It is documented here.
Java-GI is still under active development. The most notable issues and missing features are currently:
- Java does not distinguish between signed and unsigned data types. Be extra careful when native code returns, for example, a
guint
. - Java-GI makes heavy use of Cleaners to free memory or decrease an object's refcount. Cleaners are triggered during garbage collection of a Java instance. However, Java doesn't guarantee when, and if, the GC will run, and what it will clean, leading to memory leaks.