Native Bind Factories
Native bind factories are used to add instances of a class to the system to allow it to be injected into the constructor that contains a native method.
The following is an example of using injected instances:
public class MyLibraryFunctions {
private final EntityInitializer entityInitializer;
private final SongTypeFactory songTypeFactory;
public MyLibraryFunctions(EntityInitializer entityInitializer, SongTypeFactory songTypeFactory) {
this.entityInitializer = entityInitializer;
this.songTypeFactory = songTypeFactory;
}
}
The order of the parameters don't matter, nor do they necessarily need to be set to a class member variable, but it is recommended to. It is not recommended to execute code in the constructor, that should be reserved for methods annotated with @BeforeAnyInvocation as outlined here.
Predefined Injectable Classes
Below is a list of all classes that are injectable by default, with a brief description of their purpose.
Class | Description |
---|---|
MusicPopulator | Often ran in a @BeforeAnyInvocation method, this force-populates a given music type with its service provider data, as lazy loading of types may allow there to be no lookup of the type yet |
EntityDefinitionManager | Handles lookups and defining entities |
EntityInitializer | Initializes entities by name or definition with constructor args, supporting auto-conversion of types |
ListInitializer | Creates ListTypes from values, either Qilletni or Java. |
SongTypeFactory | Creates a SongType from a Track |
CollectionTypeFactory | Creates a CollectionType from a Playlist |
AlbumTypeFactory | Creates an AlbumType from an Album |
FunctionInvoker | Invokes a Qilletni function, allowing for callbacks |
TypeConverter | Converts Qilletni types to Java types and back |
DynamicProvider | Manages the current service provider, with getters for internal |
PackageConfig | A scoped instance (see section below) of the current library's config |
BackgroundTaskExecutor | Allow for async Qilletni callbacks, outlined in the Background Tasks page |
Adding Custom Injectable Classes
Sometimes, you have your own factories or other classes that you want injected into classes with native methods. To do this, first you must implement the NativeFunctionBindingFactory class. Then, add it to the qilletni_info.yml file, such as:
# ...
native_bind_factory: dev.qilletni.lib.spotify.SpotifyNativeFunctionBindingFactory
The implementation of this interface may look something like this:
public class SpotifyNativeFunctionBindingFactory implements NativeFunctionBindingFactory {
@Override
public void applyNativeFunctionBindings(NativeFunctionClassInjector nativeFunctionClassInjector) {
nativeFunctionClassInjector.addInjectableInstance(new PlaylistCreator(SpotifyApiSingleton.getSpotifyAuthorizer()));
}
}
This will now allow the spotify library's class, PlaylistCreator
to be used in classes containing native methods, such as:
public class PlaylistToolsFunctions {
private final PlaylistCreator playlistCreator;
public PlaylistToolsFunctions(PlaylistCreator playlistCreator) {
this.playlistCreator = playlistCreator;
}
}
Scoped Injectable Classes
Not all injectable instances may need to be injected into all classes. Qilletni offers the ability to scope the injectable classes via class name, allowing more restricted access to them. This also lets you have multiple native method classes taking in different instances of the same class or interface. Below is an example of setting a scoped injectable class, and two classes trying to use it.
public class CatNativeFunctionsBindingFactory implements NativeFunctionBindingFactory {
@Override
public void applyNativeFunctionBindings(NativeFunctionClassInjector nativeFunctionClassInjector) {
nativeFunctionClassInjector.addScopedInjectableInstance(new CatSitter(), List.of(CatFunctions.class));
}
}