Skip to content

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:

qilletni_info.yml
# ...
native_bind_factory: dev.qilletni.lib.spotify.SpotifyNativeFunctionBindingFactory

The implementation of this interface may look something like this:

SpotifyNativeFunctionBindingFactory
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:

PlaylistToolsFunctions.java
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.

CatNativeFunctionsBindingFactory.java
public class CatNativeFunctionsBindingFactory implements NativeFunctionBindingFactory {

    @Override
    public void applyNativeFunctionBindings(NativeFunctionClassInjector nativeFunctionClassInjector) {
        nativeFunctionClassInjector.addScopedInjectableInstance(new CatSitter(), List.of(CatFunctions.class));
    }
}
CatFunctions.java
public class CatFunctions {
    private final CatSitter catSitter;

    public CatFunctions(CatSitter catSitter) {  // Works
        this.catSitter = catSitter;
    }
}
LitterBoxFunctions.java
public class LitterBoxFunctions {
    private final CatSitter catSitter;

    public LitterBoxFunctions(CatSitter catSitter) {  // Fails; Can't find injectable CatFunctions instance
        this.catSitter = catSitter;
    }
}