Important Notice: This document is outdated and subject to change.
See Qt Creator IDE.
It's being developed in QML language.
Sources are in the assets/qml
directory.
Start the game and edit the code.
Changes apply immediately after any QML file is saved.
The by-filename URLs have the following form: image://by-filename/<filename>.<subid>
.
They are resolved relative to the asset directory.
Example:
image://by-filename/gaben.png.2
The by-id URLs are like: image://by-graphic-id/<texture-id>.<subid>
or image://by-terrain-id/<texture-id>.<subid>
.
Example:
image://by-graphic-id/7231.5
Components are adapted by writing QObject counterparts for them.
Look for the examples in the libopenage/renderer/gui
directory.
There is a property-based approach and a model-based extension to it.
Let's suppose we have a class ResourceAmount
in libopenage/economy
, and we want to be able to use it in the GUI.
In order to do that:
- A class
ResourceAmountLink
must be created in thelibopenage/renderer/gui
. It must derive fromGuiItemQObject
andGuiItem<ResourceAmountLink>
. It must be registered in the QML type system using a usual Qt approach:
qmlRegisterType<ResourceAmountLink>("yay.sfttech.openage", 1, 0, "ResourceAmount");
- Specializations
struct Wrap<ResourceAmount>
andstruct Unwrap<ResourceAmountLink>
must be defined:
namespace qtgui {
template<>
struct Wrap<ResourceAmount> {
using Type = ResourceAmountLink;
};
template<>
struct Unwrap<ResourceAmountLink> {
using Type = ResourceAmount;
};
} // namespace qtgui
- Also ResourceAmount needs a public member to be added:
public:
qtgui::GuiItemLink *gui_link
- Declare and implement needed properties and signals in the
ResourceAmountLink
using Qt property syntax.
There is a class GeneratorParameters
in libopenage/
directory.
It has a big list of parameters of different types like generation_seed
, player_radius
, player_names
, etc.
So, we're not going to write a Qt property for each one:
-
GeneratorParameters
must derive from theqtgui::GuiPropertyMap
. -
GeneratorParameters
should set its initial values like so:
this->setv("generation_seed", 4321);
this->setv("player_radius", 10);
this->set_csv("player_names", std::vector<std::string>{"name1", "name2"});
- A class
GeneratorParametersLink
must be created in thelibopenage/renderer/gui
. It must derive fromQObject
andGuiItemListModel<GeneratorParametersLink>
. It must be registered in the QML type system using a usual Qt approach:
qmlRegisterType<GeneratorParametersLink>("yay.sfttech.openage", 1, 0, "GeneratorParameters");
- Specializations
struct Wrap<GeneratorParameters>
andstruct Unwrap<GeneratorParametersLink>
must be defined:
namespace qtgui {
template<>
struct Wrap<GeneratorParameters> {
using Type = GeneratorParametersLink;
};
template<>
struct Unwrap<GeneratorParametersLink> {
using Type = GeneratorParameters;
};
} // namespace qtgui
That results into a ListModel
-like QML type with display
and edit
roles.
Basically, a database table with two columns: display
and edit
.
Qt properties can be added to the model-based class just like to the property-based.
Since the GUI and the game logic may be in different threads, additional care is needed.
It's done by the GuiItem::i()
function (GuiItem
is a base of Link
classes).
For example, forwarding of a clear()
member function call from GameMainLink
to GameMain
:
void GameMainLink::clear() {
static auto f = [] (GameMain *_this) {
_this->clear();
};
this->i(f, this);
}
Returning a value synchronously from such call isn't possible. See section about signals for that functionality.
The Link classes act like caches. So, the properties that are needed to be available in QML should be declared as members. The Q_PROPERTY should be used (see Qt docs).
Setters of the properties that may receive constant values must be implemented using the GuiItem::s()
or GuiItem::sf()
functions.
Create a class with needed signals, EditorModeSignals
for example:
class EditorModeSignals : public QObject {
Q_OBJECT
public:
signals:
void toggle();
};
Create a member of this type in the game class EditorMode
.
Then connect its signals in the corresponding EditorModeLink
class by overriding its on_core_adopted()
:
void EditorModeLink::on_core_adopted() {
QObject::connect(&unwrap(this)->gui_signals, &EditorModeSignals::toggle, this, &EditorModeLink::toggle);
}
The files that are outside of the libopenage/renderer/gui/
are allowed to include only the headers from the libopenage/renderer/gui/guisys/public
and libopenage/renderer/gui/integration/public
.
The subsystem resides in the libopenage/renderer/gui
.
- random files in the directory - bindings for the game components
guisys/
- non-openage-specific partguisys/public/
- pimpl wrappersguisys/private/
- implementationguisys/link/
- binding-related classes
integration/
- openage-specific part, notably the image providersqml/
- QML code
The C++ code must be written in a such way that none of the errors in the QML code could provoke crash or trigger a C++ assert (C++ asserts are verifying C++ code, not QML).
There are several exceptional cases when the initialization of the QML environment has no choice but to fail (calls qFatal
).
Files from guisys/
are not allowed to include any game headers that are outside of guisys/
.
Headers from guisys/public/
and integration/public/
are not allowed to include any Qt headers directly or through other headers.
- gui: - QML code
- guilink: - binding of the game logic or other subsystems (like image providers) to the GUI
- guiinit: - for the code in the game that creates and uses the GUI renderer
- guisys: - for the code in the
guisys/
directory