Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/vogel/injeqt

Dependency injection framework for Qt
https://github.com/vogel/injeqt

Last synced: 2 months ago
JSON representation

Dependency injection framework for Qt

Awesome Lists containing this project

README

        

injeqt
======

Simple dependency injection framework for Qt

Documentation
-------------

Documentation available at [github](https://github.com/vogel/injeqt/wiki).

Example
-------

Here is example of what can be done using injeqt:

#include
#include

#include
#include
#include
#include

class hello_service : public QObject
{
Q_OBJECT

public:
hello_service() {}
virtual ~hello_service() {}

std::string say_hello() const
{
return {"Hello"};
}
};

class world_service : public QObject
{
Q_OBJECT

public:
world_service() {}
virtual ~world_service() {}

std::string say_world() const
{
return {"World"};
}
};

class hello_factory : public QObject
{
Q_OBJECT

public:
Q_INVOKABLE hello_factory() {}
virtual ~hello_factory() {}

Q_INVOKABLE hello_service * create_service()
{
return new hello_service{};
}
};

class hello_client : public QObject
{
Q_OBJECT

public:
Q_INVOKABLE hello_client() : _s{nullptr}, _w{nullptr} {}
virtual ~hello_client() {}

std::string say() const
{
return _s->say_hello() + " " + _w->say_world() + "!";
}

private slots:
INJEQT_INIT void init()
{
std::cerr << "all services set" << std::endl;
}

INJEQT_DONE void done()
{
std::cerr << "ready for destruction" << std::endl;
}

INJEQT_SET void set_hello_service(hello_service *s)
{
_s = s;
}

INJEQT_SET void set_world_service(world_service *w)
{
_w = w;
}

private:
hello_service *_s;
world_service *_w;

};

class module : public injeqt::module
{
public:
explicit module()
{
_w = std::unique_ptr{new world_service{}};

add_type();
add_type();
add_factory();
add_ready_object(_w.get());
}

virtual ~module() {}

private:
std::unique_ptr _w;

};

int main()
{
auto modules = std::vector>{};
modules.emplace_back(std::unique_ptr{new module{}});

auto injector = injeqt::injector{std::move(modules)};
auto client = injector.get();
auto hello = client->say();

std::cout << hello << std::endl;
}

#include "hello-world.moc"

In that example we can see two main services names `hello_service` and `world_service`. There
is also client of these names `hello_client`. In `module` class we configure how we create
and access these instances.

`hello_client` is added using `add_type` function. It means that injeqt will try to create it
using default constructor. We provide that by declaration of `Q_INVOKABLE hello_client()`
(`Q_INVOKABLE` is requires by Qt's meta object system).

`hello_service` is added using `add_factory` function. It means that injeqt will first try to
create a `hello_factory` object, then it will look for a method of that objet that returns
`hello_service` object. It will find `Q_INVOKABLE hello_service * create_service()` and use it.

To be able to create `hello_factory` injeqt must know about it, so we also add it using `add_type`
method.

Last, `world_service`, is added as a ready object - provided from outside of injeqt scope.

In `main` method list of conifguration modules are passed to newly created `injector` instance.
From that moment, we can use `injector` to create and manage our services. Just one line below
an `hello_client` instance is required. This is what happens next:

* injeqt looks for dependencies of `hello_client` and found that it first needs to create `hello_factory`
* injeqt creates `hello_factory` without problems, as it does not have dependencies of its own
* injeqt adds new instance to object pool
* injeqt calls `hello_factory::create_service()` methods and adds its result to object pool
* now all dependencies of `hello_client` are available, so new instance of it is created with
default constructor and its added to objec tpool
* all methods of `hello_client` marked with `INJEQT_SET` are called with proper objects from pool
* all methods of `hello_client` marked with `INJEQT_INIT` are called
* this instance is returned to caller
* before injector is destructed, all methods of `hello_client` marked with `INJEQT_DONE` are called