Constraint-based layout manager for GTK+
Emeus is a constraint-based layout manager widget for GTK+,
written using the Cassowary constraint solving algorithm.
GTK+ has two different sorts of layout managers:
The first model works really well in ensuring that UIs are responsive to
size changes, by avoiding pixel-perfect positioning on the screen, as well
as ensuring that changing the font size or margins and paddings do not break
the user interface; its main down side is that it requires accurate, and
often verbose packing of widgets inside boxes, inside other boxes.
The second model allows a more efficient way to construct a user interface,
at the major costs of either “freezing” it, or requiring constant
recalculations of the relative position and size of each UI element.
Emeus provides a third layout management policy, based on constraints;
each UI element binds one of more of its attributes — like its width, or its
position — to other UI elements, in a way that is more natural to describe
from a UI building perspective, and hopefully more efficient that stacking
piles of boxes one inside another.
EmeusLayoutConstraint
is a GtkContainer that can
support multiple children; each child, in turn, is associated to one or moreEmeusConstraint
instances. Each constraint is the expression of a simple
linear equation:
item1.attr1 = item2.attr2 × multiplier + constant
Where:
item1
is the target widget, that is the widget we want to constraint;attr1
is an attribute of the target widget, like width
or end
,item2
is the source widget, that is the widget that provides the valueattr2
is an attribute of the source widget that provides the valuemultiplier
is a multiplication factor, expressed as a floating pointconstant
is an additional constant factor, expressed as a floatingUsing both notations, then, we can construct user interfaces like:
+-------------------------------------------+
| [ button 1 ] [ button 2 ] [ button 3] |
+-------------------------------------------+
By expressing the constraints between the UI elements. For instance, we can
center button2
within its parent and give it a minimum width of 250
logical pixels:
button2.width >= 250
button2.centerX = parent.centerX
button2.centerY = parent.centerY
Then, we can tie button1
and button3
to button2
, and ensure that the
width and height of all three buttons are the same:
button1.end = button2.start - 8
button1.width = button2.width
button1.height = button2.height
button3.start = button2.end + 8
button3.width = button2.width
button3.height = button2.height
The EmeusConstraintLayout
widget will attempt to resolve all the
constraints, and lay out its children according to them.
$ mkdir _build && cd _build
$ meson
$ ninja
$ ninja test
# ninja install
The API reference for Emeus is available online.
Emeus is released under the terms of the GNU Lesser General Public License,
either version 2.1 or, at your option, any later version.
The Cassowary simplex solving algorithm implementation is largely inspired
by the equivalent implementations written in various other language,
including:
Additionally, the automatic layout solving is inspired by
autolayout.js, a JavaScript automatic constraint-based
layout, which, in turn, is based on the Apple autolayout
layout manager.
You can check on the Overconstrained website for
additional Cassowary implementations in various languages.