diff options
author | Paul Davis <paul@linuxaudiosystems.com> | 2016-09-25 14:46:07 -0500 |
---|---|---|
committer | Paul Davis <paul@linuxaudiosystems.com> | 2016-09-27 14:59:32 -0500 |
commit | 85c1523611f401a318df4547e2b6f84e0b0d5af8 (patch) | |
tree | f4c4c194800c23f2190f09ec6e3cfa52c1fc3bb9 /libs/canvas/box.cc | |
parent | ba82e673fd4435065ae9689411d7ed0d232b8cde (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.cc | 310 |
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) +{ +} |