diff options
author | crupest <crupest@outlook.com> | 2018-09-25 13:37:33 +0800 |
---|---|---|
committer | crupest <crupest@outlook.com> | 2018-09-25 13:37:33 +0800 |
commit | ccbc293c0158d4450c0db344474193f17925403f (patch) | |
tree | 1ecde3132b3299ab2f272f541a76bced67f983b1 /src/ui/controls/margin_container.cpp | |
parent | 184c3b2b39d3fa34f9349a7d7fbebe49bc62f7fc (diff) | |
parent | ea4b0966d8eb5a8e76dfbe4d833a07a4797a3284 (diff) | |
download | cru-ccbc293c0158d4450c0db344474193f17925403f.tar.gz cru-ccbc293c0158d4450c0db344474193f17925403f.tar.bz2 cru-ccbc293c0158d4450c0db344474193f17925403f.zip |
Merge branch 'master' into issue4-dev
Diffstat (limited to 'src/ui/controls/margin_container.cpp')
-rw-r--r-- | src/ui/controls/margin_container.cpp | 63 |
1 files changed, 63 insertions, 0 deletions
diff --git a/src/ui/controls/margin_container.cpp b/src/ui/controls/margin_container.cpp new file mode 100644 index 00000000..8f9101b2 --- /dev/null +++ b/src/ui/controls/margin_container.cpp @@ -0,0 +1,63 @@ +#include "margin_container.h" + +namespace cru::ui::controls +{ + inline float AtLeast0(const float value) + { + return value < 0 ? 0 : value; + } + + inline Size AtLeast0(const Size& size) + { + return Size(AtLeast0(size.width), AtLeast0(size.height)); + } + + MarginContainer::MarginContainer(const Thickness& margin) + : Control(true), margin_(margin) + { + } + + void MarginContainer::SetMargin(const Thickness& margin) + { + margin_ = margin; + Relayout(); + } + + Size MarginContainer::OnMeasure(const Size& available_size) + { + const auto margin_size = Size(margin_.left + margin_.right, margin_.top + margin_.bottom); + const auto coerced_available_size = AtLeast0(available_size - margin_size); + return Control::OnMeasure(coerced_available_size) + margin_size; + } + + void MarginContainer::OnLayout(const Rect& rect) + { + const auto anchor = Point(margin_.left, margin_.top); + const auto margin_size = Size(margin_.left + margin_.right, margin_.top + margin_.bottom); + ForeachChild([anchor, margin_size, rect](Control* control) + { + const auto layout_params = control->GetLayoutParams(); + const auto size = control->GetDesiredSize(); + + auto&& calculate_anchor = [](const float anchor, const Alignment alignment, const float layout_length, const float control_length) -> float + { + switch (alignment) + { + case Alignment::Center: + return anchor + (layout_length - control_length) / 2; + case Alignment::Start: + return anchor; + case Alignment::End: + return anchor + layout_length - control_length; + default: + UnreachableCode(); + } + }; + + control->Layout(Rect(Point( + calculate_anchor(anchor.x, layout_params->width.alignment, rect.width - margin_size.width, size.width), + calculate_anchor(anchor.y, layout_params->height.alignment, rect.height - margin_size.height, size.height) + ), size)); + }); + } +} |