Integrating RmlUi into your application
This guide will help you get started when you are ready to integrate RmlUi into your own application.
If you haven’t already done so, take a look at the sample applications in /Samples/
. There you can find a whole heap of useful examples of how to use and abuse RmlUi. Also in there you can find the shell library we used to develop all the samples (in /Samples/shell/
), which can be great if you want to write a quick and dirty application of your own.
Setting up the build environment
RmlUi is developed following the C++14 standard and can be used on the following platforms:
- Windows 32/64bit, compiling with Microsoft Visual Studio 2017+.
- MacOS 32/64bit, compiling with GCC 5+.
- Linux, compiling with GCC 5+.
Visual Studio
- Add the RmlUi include path
RmlUi/Include/
to your project’s include paths.- See
Project → Properties → C/C++ → General → Additional Include Directories
.
- See
- Add the RmlUi library path (
Debug/
andRelease/
under theRmlUi/Build/
directory as appropriate) and the FreeType library path to your library paths.- See
Project → Properties → Linker → General → Additional Library Directories
.
- See
- Link with
RmlCore.lib
andfreetype.lib
.- See
Project → Properties → Linker → Input → Additional Dependencies
.
- See
- If you have RmlUi built as a static library, add the following preprocessor definition:
RMLUI_STATIC_LIB
.- See
Project → Properties → C/C++ → Preprocessor → Preprocessor Definitions
.
- See
- If you have RmlUi built as a shared/dynamic library, copy the appropriate DLLs into the directory your executable will run from..
- That is,
Debug/RmlCore.dll
for debug builds andRelease/RmlCore.dll
for release builds from theRmlUi/Build/
folder.
- That is,
- If you have FreeType built as a shared/dynamic library, copy the
freetype.dll
file into the directory your executable will run from. - Add
#include <RmlUi/Core.h>
in a source or header file to start using RmlUi.
MacOS and Linux
- Add the RmlUi include path
RmlUi/Include/
and library pathRmlUi/Build/
to the paths in your build system. - Link with
RmlCore
andfreetype
. - Either copy the RmlUi libraries into your application’s working directory, or set a
LD_LIBRARY_PATH
(DYLD_LIBRARY_PATH
for MacOS) environment variable. - When the library is built as a static library, add
#define RMLUI_STATIC_LIB
before including the RmlUi headers. - Add
#include <RmlUi/Core.h>
in a source or header file to start using RmlUi.
Conan and vcpkg
If you used Conan or vcpkg to acquire RmlUi, then libraries and include paths should already have been setup for you. The only thing remaining:
- Add
#include <RmlUi/Core.h>
in a source or header file to start using RmlUi.
Initialising RmlUi
Before you can initialise RmlUi, you’ll need to set the interfaces that the library uses to interact with your application. There are two compulsory interfaces, the system interface and the render interface.
The system interface
The system interface is defined in <RmlUi/Core/SystemInterface.h>
. In order to create a valid system interface, you’ll need to create a class that inherits from Rml::SystemInterface
and provides the function:
virtual double GetElapsedTime();
The function should return the time (in seconds) since the start of the application. Install your system interface by calling Rml::SetSystemInterface()
with a pointer to the interface. Note that you must keep the system interface alive until after the call to Rml::Shutdown()
and destroy it afterwards. RmlUi won’t release your interfaces.
For more uses of the system interface, see the documentation.
The render interface
The render interface is defined in <RmlUi/Core/RenderInterface.h>
. It provides a way for RmlUi to send its geometry into your application’s rendering pipeline. If you want to get RmlUi up and running as quickly as possible in your own application, you can copy the render interface defined in the sample shell if your application is using OpenGL (you can find this at /Samples/shell/include/ShellRenderInterface.h
and /Samples/shell/src/ShellRenderInterface.cpp
).
Once you have a render interface for your application, install it into RmlUi by calling Rml::SetRenderInterface()
.
If you’d like to take an in-depth look at setting up your own render interface, please see the render interface documentation.
Initialising the library
Call the global function Rml::Initialise()
once you have installed the system and render interfaces and RmlUi will start up.
Creating a context
All elements within RmlUi are part of a context. You must have at least one context in order to load, manipulate and render and interface elements. To create a context, use the Rml::CreateContext()
function, passing in the name of the new context and its initial dimensions like so:
Rml::Context* context = Rml::CreateContext("default", Rml::Vector2i(myScreenWidth, myScreenHeight));
You can release the context when you’re done with it by calling Rml::RemoveContext(context->GetName())
. All contexts will automatically be destroyed on shutdown.
Updating and rendering
Your application will need to update and render each context it maintains, as appropriate. Call the Context::Update()
function on each context as often as necessary to update the context (usually after the frame’s input has been injected), and Context::Render()
at the appropriate place in your application’s render loop.
Loading fonts
RmlUi does not come integrated with any fonts (with the exception of the debugger plugin), they must be provided by the user. Font faces can be loaded through the Rml::LoadFontFace()
function.
bool success = Rml::LoadFontFace("assets/my_font_face.ttf");
Loading a document
Once you have a valid context, you can load a document into the context with the LoadDocument()
function. LoadDocument()
takes a single parameter, a string with the document’s file name. If the load is successful you’ll get a pointer to a Rml::ElementDocument
back; call Show()
on the document to make it visible.
Rml::ElementDocument* document = context->LoadDocument("../../assets/demo.rml");
if (document)
document->Show();
Unload the document by calling Close()
.
document->Close();
Note: event listeners attached to the document or any of its children must not be destroyed until the next call to Context::Update()
or Rml::Shutdown()
.
Injecting input
Once you’ve got a document loading and rendering, the next step is to get your input into RmlUi. The context object has a range of functions for sending mouse, keyboard and text input into the system:
// Sends a key down event into this context.
void ProcessKeyDown(Rml::Input::KeyIdentifier key_identifier, int key_modifier_state);
// Sends a key up event into this context.
void ProcessKeyUp(Rml::Input::KeyIdentifier key_identifier, int key_modifier_state);
// Sends a single unicode character (code point) as text input into this context.
void ProcessTextInput(Rml::Character character);
// Sends a string of UTF-8 text input into this context.
void ProcessTextInput(const Rml::String& string);
// Sends a mouse movement event into this context.
void ProcessMouseMove(int x, int y, int key_modifier_state);
// Sends a mouse-button down event into this context.
void ProcessMouseButtonDown(int button_index, int key_modifier_state);
// Sends a mouse-button up event into this context.
void ProcessMouseButtonUp(int button_index, int key_modifier_state);
// Sends a mouse-wheel movement event into this context.
void ProcessMouseWheel(float wheel_delta, int key_modifier_state);
Call the appropriate input functions to inject all relevant user input into your RmlUi context each frame, before you call Update()
. Note that RmlUi does not translate key presses into text; this is up to the application. For more information, see the chapter on user input.
Debugger
The RmlDebugger
plugin is a visual debugger for RmlUi elements, inspired by similar debuggers for web browsers. We strongly recommend you use this in your application during development!
To use RmlDebugger, include <RmlUi/Debugger.h>
in your application and link with RmlDebugger
. For usage details, see the documentation for the debugger plugin.
Where next?
Now that you’ve had a (very!) brief introduction to RmlUi, it is recommended you read the core overview to get an understanding of the composition of RmlUi. From there, either work your way through the documentation, or dive on into the code and consult it as necessary.