Now we’ve drawn some text to the screen, we can move on to slightly more complicated layouts.
The library has a stack of “drawing states”. Each drawing state holds various drawing parameters such as current viewport, colour, font, and so on.
Let’s look at a simple example, where we specify the size of a button.
First, we draw a button, we call control_button()
with the text to include as the button text:
@gfx
ui_start();
control_button("Press me");
The initial viewport is the whole available screen. This means that our button will fill the whole screen:
Let’s change the size of the button. To do this, we change the viewport.
You don’t generally change the viewport of the current drawing state. Instead, you push a new drawing state onto the stack which has the modifications you need. When you are done, you pop that drawing state off the stack, and the viewport (and other parameters) return to their previous state.
Here, we use ui_push_height()
to push a new drawing state to the stack, where the height of the new viewport is set to be 50 pixels:
ui_push_height(50);
control_button("Press me");
ui_pop();
(Indenting the code like this is useful to keep track of how deep the drawing-state stack currently is.)
To set the width as well, we use ui_push_width()
. We’re also going to use ui_debug()
, which illustrates the drawing stack by outlining and numbering the viewport at each level in the stack:
ui_push_height(50);
ui_push_width(100);
control_button("Press me");
ui_debug();
ui_pop();
ui_pop();
Here, you can see the stack of drawing states at the time that ui_debug()
is called. The top level (1) has a viewport that covers the entire available area. The second level (2) in the stack has the correct height, and the third (3) has the correct width and height.
Now, what happens if we move ui_debug()
one line down, so it’s below ui_pop()
?
ui_push_height(50);
ui_push_width(100);
control_button("Press me");
ui_pop();
ui_debug();
ui_pop();
As you can see, there are only two layers in the drawing-state stack at the point. The drawing-state which was previously labelled “3” has been forgotten by the time ui_debug()
is called.
Often, once you’ve drawn the UI in an area of the screen, you want to avoid using that area of the screen for anything else. However, with the previous code, when we called ui_pop()
the viewport returned to its previous state, which is inconvenient.
So, we have a family of functions: ui_split_top()
, ui_split_bottom()
, ui_split_left()
and ui_split_right()
. These functions push a new viewport to the stack, but they also modify the old viewport to avoid this area. Let’s look at an example:
ui_split_bottom(50);
ui_debug();
ui_pop();
Here, when we call ui_debug()
the stack is two levels deep. However, while the first level (1) was originally filling the whole screen, it is now smaller, so that it no longer overlaps the second level (2).
Let’s see how this looks using a button and some text:
ui_split_bottom(50);
control_button("Press me");
ui_pop();
ui_text("Hello, world!");
As you can see, the “Hello, world!” text is now vertically centred in the remaining space above the button.
ui_split_next()
Now, let’s introduce another very useful function: ui_split_next()
.
This function pops the current viewport off from the stack, and then re-splits using the same height/width. This is very useful for creating stacks of elements:
ui_split_bottom(50);
control_button("Press me");
ui_split_next();
control_button("Then press me");
ui_pop();
ui_text("Hello, world!");
So, the second button is exactly the same height. If you’re not sure what the viewport stack looks like at any point during this, then add ui_debug()
to see what it’s doing.
Often instead of splitting with an exact height/width, you might want to split either as a proportion (e.g. 50% of the height), or to fit a certain piece of text (e.g. just enough width to fit some text in your button).
For these, you can use the variants: ui_split_leftratio()
or ui_split_lefttext()
(and similar for right/top/bottom). Let’s take a look at this code which uses both techniques:
ui_split_topratio(0.5);
ui_text("This section takes up 50% of the available vertical space");
ui_pop();
ui_split_lefttext("Auto-sized button");
control_button("Auto-sized button");
ui_pop();
ui_text("remaining space");
The ui_push_height()
and ui_push_width()
functions also have these variants (ui_push_heightratio()
and so on).
In all the above examples, text is always displayed centred in the available space. Similarly, in earlier examples when we used ui_push_height()
, the new viewport was centred within the previous one.
This is because the horizontal/vertical alignment is central by default. You can change the alignment using ui_align(horizontal, vertical)
. Both arguments are values between 0 and 1, where 0 is top/left, and 1 is bottom/right. If you specify -1
for either of these, it preserves the current value.
ui_align(0, -1);
ui_text("left");
ui_align(1, 1);
ui_text("bottom-right");
The alignment is a property of the drawing state. This means that if you change the alignment in the current drawing state, then it will be reset when you clear that drawing state using ui_pop()
.
ui_push_above()
and othersIf you have drawn part of your UI using a specific height/width and alignment, you might want to then fit something else into the space around it.
For this, you can use the functions ui_push_above()
, ui_push_below()
, ui_push_leftof()
and ui_push_rightof()
. They take the same arguments as ui_push_height()
, and have the same ratio/text variants (e.g. ui_push_aboveratio()
and ui_push_abovetext()
).
For example, here we draw a button right in the middle of the screen. We then fit some text above and below this button:
ui_push_height(50);
control_button("Press me");
ui_pop();
ui_push_above(50);
ui_text("text above the button");
ui_pop();
ui_push_below(50);
ui_text("text below the button");
ui_pop();