Immediate Mode GUI
I am making my own ImGui library to learn a bit how the mainstream Dear ImGui library actually works. The inner workings of Dear ImGui wasn't documented or explained anywhere else, so I might just do a shallow investigation into their code base and breifly talk about how it inspires my own implementation.
Background
Immediate Mode GUI was initially suggested by Casey Muratori1 back in 2005. It's often being compared to the traditional retained mode GUI, e.g. Qt. ImGui on the other hand is declarative, using logical flow in the program to determine what to draw in a frame. This enables faster prototyping and writing throw-away code without any hassle.
State Storage
Since immediate mode means everything are declarative, there will be no states recorded. So how are we suppose to have states shared across frames? The answer is of course, putting them in a storage that is managed by the context object.
For example, an expandable element is clicked open on frame 1. Before we draw the expandable at frame 2, it needs to query the storage about the state of the expandable in order to determine whether to draw its expanded contents or not. But how to write the state of the expandable?
Element ID
The state storage itself shouldn't be complicated. It can be just a hash map to a state struct. The problem is how to address immediate mode UI elements? They are not guaranteed to be consistent across frames. Meaning we cannot give each elements with an incremental counter.
Luckily, the element stack stays relatively stable across frames. By saying "stack", they are usually pushed with Begin*()
and poped with End*()
(or manually by PushID()
and PopID()
). This essentially wraps a "namespace" around the stack elements, such that duplicated names are allowed. For example:
• label1
> myExpandable
• label2
v folder
• label3
> myExpandable
Here, the first myExpandable
at root level has a hash value already. To prevent collision, the second level myExpandable
will have its hash seeded by its parent folder
's hash. Now the name collision issue should be resolved. In case of inevitable collisions, Dear ImGui solves this by allowing custom IDs with <name>###<customID>
to override the hashing input.
And as a record, Dear ImGui chose CRC32 as the hashing algorithm. Although there's a FIXME in the code saying using FNV1a can potentially speed up the hashing process as there's no need to do random access to a look up table, proper measurements are needed to conclude its effectiveness.
Clip Rect
There's a term called "Clip Rect" in Dear ImGui, mainly used to prevent child elements being drawn outside the window. This can be easily achieved by scissor test in the rasterize pipeline. For complex shapes, stencil test is required, but it's an overkill for simple axis-aligned UI elements. For now, a simple scissor test should be sufficient.
Calling PushClipRect()
and PopClipRect()
will push a rect to the stack, then the renderer will set the topmost clipping region as the scissor testing region. This can effectively discard any pixels that exceeded the boundary.
Cursors
As anybody knows about cursors, there are two types, hardware and software. Hardware cursor movement wasn't limited by the frame rate of the application, usually it's done by calling platform specific functions (like SetCursor
in Windows, and NSCursor
in OSX). Dear ImGui appears to use the corresponding API provided by the backends, which is mainly hardware/OS cursors.
My Version
As I said in the Prelude section, my goal is to write minimal code to support both retained and immediate mode UI elements. // TODO: Write more when I'm confident and consolidated the current architecture design.
-
Casey Muratori (2005). Immediate-Mode Graphical User Interfaces. https://caseymuratori.com/blog_0001 ↩