summaryrefslogtreecommitdiff
path: root/libs/canvas/box.cc
diff options
context:
space:
mode:
authorPaul Davis <paul@linuxaudiosystems.com>2016-09-25 14:46:07 -0500
committerPaul Davis <paul@linuxaudiosystems.com>2016-09-27 14:59:32 -0500
commit85c1523611f401a318df4547e2b6f84e0b0d5af8 (patch)
treef4c4c194800c23f2190f09ec6e3cfa52c1fc3bb9 /libs/canvas/box.cc
parentba82e673fd4435065ae9689411d7ed0d232b8cde (diff)
add new Box container for Canvas.
API subject to change and improvement
Diffstat (limited to 'libs/canvas/box.cc')
-rw-r--r--libs/canvas/box.cc310
1 files changed, 310 insertions, 0 deletions
diff --git a/libs/canvas/box.cc b/libs/canvas/box.cc
new file mode 100644
index 0000000000..d7d0c90548
--- /dev/null
+++ b/libs/canvas/box.cc
@@ -0,0 +1,310 @@
+/*
+ Copyright (C) 2011-2014 Paul Davis
+ Author: Carl Hetherington <cth@carlh.net>
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+*/
+
+#include <algorithm>
+
+#include "canvas/box.h"
+#include "canvas/rectangle.h"
+
+using namespace ArdourCanvas;
+
+Box::Box (Canvas* canvas, Orientation o)
+ : Item (canvas)
+ , orientation (o)
+ , spacing (0)
+ , top_padding (0), right_padding (0), bottom_padding (0), left_padding (0)
+ , top_margin (0), right_margin (0), bottom_margin (0), left_margin (0)
+ , homogenous (false)
+{
+ self = new Rectangle (this);
+ self->set_outline (false);
+ self->set_fill (false);
+}
+
+Box::Box (Item* parent, Orientation o)
+ : Item (parent)
+ , orientation (o)
+ , spacing (0)
+ , top_padding (0), right_padding (0), bottom_padding (0), left_padding (0)
+ , top_margin (0), right_margin (0), bottom_margin (0), left_margin (0)
+{
+ self = new Rectangle (this);
+ self->set_outline (false);
+ self->set_fill (false);
+}
+
+
+Box::Box (Item* parent, Duple const & p, Orientation o)
+ : Item (parent, p)
+ , orientation (o)
+ , spacing (0)
+ , top_padding (0), right_padding (0), bottom_padding (0), left_padding (0)
+ , top_margin (0), right_margin (0), bottom_margin (0), left_margin (0)
+{
+ self = new Rectangle (this);
+ self->set_outline (false);
+ self->set_fill (false);
+}
+
+void
+Box::render (Rect const & area, Cairo::RefPtr<Cairo::Context> context) const
+{
+ Item::render_children (area, context);
+}
+
+void
+Box::compute_bounding_box () const
+{
+ _bounding_box = boost::none;
+
+ if (_items.empty()) {
+ _bounding_box_dirty = false;
+ return;
+ }
+
+ add_child_bounding_boxes (!collapse_on_hide);
+
+ if (_bounding_box) {
+ Rect r = _bounding_box.get();
+
+ _bounding_box = r.expand (top_padding + outline_width() + top_margin,
+ right_padding + outline_width() + right_margin,
+ bottom_padding + outline_width() + bottom_margin,
+ left_padding + outline_width() + left_margin);
+ }
+
+ _bounding_box_dirty = false;
+}
+
+void
+Box::set_spacing (double s)
+{
+ spacing = s;
+}
+
+void
+Box::set_padding (double t, double r, double b, double l)
+{
+ double last = t;
+
+ top_padding = t;
+
+ if (r >= 0) {
+ last = r;
+ }
+ right_padding = last;
+ if (b >= 0) {
+ last = b;
+ }
+ bottom_padding = last;
+ if (l >= 0) {
+ last = l;
+ }
+ left_padding = last;
+}
+
+void
+Box::set_margin (double t, double r, double b, double l)
+{
+ double last = t;
+ top_margin = t;
+ if (r >= 0) {
+ last = r;
+ }
+ right_margin = last;
+ if (b >= 0) {
+ last = b;
+ }
+ bottom_margin = last;
+ if (l >= 0) {
+ last = l;
+ }
+ left_margin = last;
+}
+
+void
+Box::reset_self ()
+{
+ if (_bounding_box_dirty) {
+ compute_bounding_box ();
+ }
+
+ if (!_bounding_box) {
+ self->hide ();
+ return;
+ }
+
+ Rect r (_bounding_box.get());
+
+ /* XXX need to shrink by margin */
+
+ self->set (r);
+}
+
+void
+Box::reposition_children ()
+{
+ Duple previous_edge (0, 0);
+ Distance largest_width = 0;
+ Distance largest_height = 0;
+
+ if (homogenous) {
+
+ for (std::list<Item*>::iterator i = _items.begin(); ++i != _items.end(); ++i) {
+ boost::optional<Rect> bb = (*i)->bounding_box();
+ if (bb) {
+ largest_height = std::max (largest_height, bb.get().height());
+ largest_width = std::max (largest_height, bb.get().width());
+ }
+ }
+ }
+
+ for (std::list<Item*>::iterator i = _items.begin(); ++i != _items.end(); ++i) {
+
+ (*i)->set_position (previous_edge);
+
+ if (orientation == Vertical) {
+
+ Distance shift = 0;
+
+ if (homogenous) {
+ shift = largest_height;
+ } else {
+ boost::optional<Rect> bb = (*i)->bounding_box();
+
+ if (!(*i)->visible()) {
+ /* invisible child */
+ if (!collapse_on_hide) {
+ /* still add in its size */
+ if (bb) {
+ shift += bb.get().height();
+ }
+ }
+ } else {
+ if (bb) {
+ shift += bb.get().height();
+ }
+ }
+ }
+
+ previous_edge = previous_edge.translate (Duple (0, spacing + shift));
+ } else {
+
+ Distance shift = 0;
+
+ if (homogenous) {
+ shift = largest_width;
+ } else {
+ boost::optional<Rect> bb = (*i)->bounding_box();
+
+ if (!(*i)->visible()) {
+ if (!collapse_on_hide) {
+ if (bb) {
+ shift += bb.get().width();
+ }
+ }
+ } else {
+ if (bb) {
+ shift += bb.get().width();
+ }
+ }
+ }
+
+ previous_edge = previous_edge.translate (Duple (0, spacing + shift));
+ }
+ }
+
+ _bounding_box_dirty = true;
+ reset_self ();
+}
+
+void
+Box::pack_end (Item* i, double extra_padding)
+{
+ if (!i) {
+ return;
+ }
+
+ /* prepend new child */
+ Item::add_front (i);
+ reposition_children ();
+}
+void
+Box::pack_start (Item* i, double extra_padding)
+{
+ if (!i) {
+ return;
+ }
+
+ /* append new child */
+ Item::add (i);
+ reposition_children ();
+}
+
+void
+Box::add (Item* i)
+{
+ pack_start (i);
+}
+
+void
+Box::child_changed ()
+{
+ /* catch visibility and size changes */
+
+ Item::child_changed ();
+ reposition_children ();
+}
+
+void
+Box::set_collapse_on_hide (bool yn)
+{
+ if (collapse_on_hide != yn) {
+ collapse_on_hide = yn;
+ reposition_children ();
+ }
+}
+
+/*----*/
+
+VBox::VBox (Canvas* c)
+ : Box (c, Vertical)
+{
+}
+VBox::VBox (Item* i)
+ : Box (i, Vertical)
+{
+}
+VBox::VBox (Item* i, Duple const & position)
+ : Box (i, position, Vertical)
+{
+}
+
+HBox::HBox (Canvas* c)
+ : Box (c, Horizontal)
+{
+}
+HBox::HBox (Item* i)
+ : Box (i, Horizontal)
+{
+}
+HBox::HBox (Item* i, Duple const & position)
+ : Box (i, position, Horizontal)
+{
+}