Layout
The UI library provides an integration with the layout
library to allow defining layouts for windows easily. For
this purpose, the UI library provides three keywords that are used to define windows: window
,
frame
, and dialog
. They work the same, except that they create classes that are derived from
Window
, Frame
, and Dialog
respectively.
Inside a class defined using one of these keywords it is possible to include a layout
block that
specifies the layout of the window. The contents of the layout block follows the domain specific
language from the layout library. However, the UI library
extends the syntax to also allow declaring variables inside the layout syntax. Declared variables
will be added as members of the class, and will be initialized in the constructor. The UI library
also wraps the entire layout in a border.
For example, a frame can be declared as follows:
use ui; use core:geometry; // Creates an actor that inherits from ui.Frame. frame MyFrame { layout Grid { expandCol: 0; Button a("A") {} Button b("B") { rowspan: 2; } nextLine; Button c("C") {} nextLine; Button d("D") {} Button e("E") {} Button f("F") { row: 3; col: 2; } Button g("G") { row: 4; col: 1; colspan: 2; } } init() { init("My Window", Size(200, 200)); create(); a.onClick = () => print("Hello!"); } } void main() { MyFrame frame; frame.waitForClose(); }
Note that the UI library integrates computations of minimum size from the layout library. As such, the system will not allow the window to become smaller than its contents.
The UI library also adds special syntax for nesting containers. This is useful for placing certain
parts of the layout as a child to another component, such as when using ui.TabView
,
ui.Group
, or ui.ScrollWindow
.
The syntax looks a lot like a regular layout
declaration, except that the container
keyword is
used instead. This construct produces an expression (of type Container
) that can be passed as a
parameter to other components, as shown below:
use ui; use core:geometry; frame MyFrame { layout Grid { expandCol: 0; wrapCols: 1; TabView { add: "Tab 1", container Grid { expandCol: 0; Button b1("Button 1") {} }; add: "Tab 2", container Grid { expandCol: 0; Label l1("Label 1") {} }; } } init() { init("My Window", Size(200, 200)); create(); } } // ...
The UI library automatically inserts a border at the root of each layout block. This can be
suppressed by adding the text without border
after either layout
or container
.