Remove idiocy.
[ardour2.git] / gtk2_ardour / port_matrix_component.cc
blob7b83e5d10f6c8adb6220488b2d152a9041e4afea
1 /*
2 Copyright (C) 2002-2009 Paul Davis
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
20 #include "port_matrix_component.h"
21 #include "port_matrix.h"
22 #include "port_matrix_body.h"
24 using namespace std;
26 /** Constructor.
27 * @param p Port matrix that we're in.
29 PortMatrixComponent::PortMatrixComponent (PortMatrix* m, PortMatrixBody* b)
30 : _matrix (m),
31 _body (b),
32 _pixmap (0),
33 _render_required (true),
34 _dimension_computation_required (true)
39 /** Destructor */
40 PortMatrixComponent::~PortMatrixComponent ()
42 if (_pixmap) {
43 gdk_pixmap_unref (_pixmap);
47 void
48 PortMatrixComponent::setup ()
50 _dimension_computation_required = true;
51 _render_required = true;
54 GdkPixmap *
55 PortMatrixComponent::get_pixmap (GdkDrawable *drawable)
57 if (_render_required) {
59 if (_dimension_computation_required) {
60 compute_dimensions ();
61 _dimension_computation_required = false;
62 _body->component_size_changed ();
65 /* we may be zero width or height; if so, just
66 use the smallest allowable pixmap */
67 if (_width == 0) {
68 _width = 1;
70 if (_height == 0) {
71 _height = 1;
74 /* make a pixmap of the right size */
75 if (_pixmap) {
76 gdk_pixmap_unref (_pixmap);
78 _pixmap = gdk_pixmap_new (drawable, _width, _height, -1);
80 /* render */
81 cairo_t* cr = gdk_cairo_create (_pixmap);
82 render (cr);
83 cairo_destroy (cr);
85 _render_required = false;
88 return _pixmap;
91 void
92 PortMatrixComponent::set_source_rgb (cairo_t *cr, Gdk::Color const & c)
94 cairo_set_source_rgb (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p());
97 void
98 PortMatrixComponent::set_source_rgba (cairo_t *cr, Gdk::Color const & c, double a)
100 cairo_set_source_rgba (cr, c.get_red_p(), c.get_green_p(), c.get_blue_p(), a);
103 pair<uint32_t, uint32_t>
104 PortMatrixComponent::dimensions ()
106 if (_dimension_computation_required) {
107 compute_dimensions ();
108 _dimension_computation_required = false;
109 _body->component_size_changed ();
112 return make_pair (_width, _height);
115 Gdk::Color
116 PortMatrixComponent::background_colour ()
118 return _matrix->get_style()->get_bg (Gtk::STATE_NORMAL);
121 /** @param g Group.
122 * @return Visible size of the group in grid units, taking visibility and show_only_bundles into account.
124 uint32_t
125 PortMatrixComponent::group_size (boost::shared_ptr<const PortGroup> g) const
127 uint32_t s = 0;
129 if (g->visible()) {
130 PortGroup::BundleList const & bundles = g->bundles ();
131 if (_matrix->show_only_bundles()) {
132 s = bundles.size();
133 } else {
134 for (PortGroup::BundleList::const_iterator i = bundles.begin(); i != bundles.end(); ++i) {
135 s += i->bundle->nchannels();
138 } else {
139 s = 1;
142 return s;
145 /** @param bc Channel.
146 * @param groups List of groups.
147 * @return Position of bc in groups in grid units, taking visibility and show_only_bundles into account.
149 uint32_t
150 PortMatrixComponent::channel_to_position (ARDOUR::BundleChannel bc, PortGroupList const * groups) const
152 uint32_t p = 0;
154 for (PortGroupList::List::const_iterator i = groups->begin(); i != groups->end(); ++i) {
156 PortGroup::BundleList const & bundles = (*i)->bundles ();
158 for (PortGroup::BundleList::const_iterator j = bundles.begin(); j != bundles.end(); ++j) {
160 if (j->bundle == bc.bundle) {
162 /* found the bundle */
164 if (_matrix->show_only_bundles() || !(*i)->visible()) {
165 return p;
166 } else {
167 return p + bc.channel;
172 if ((*i)->visible()) {
174 /* move past this bundle */
176 if (_matrix->show_only_bundles()) {
177 p += 1;
178 } else {
179 p += j->bundle->nchannels ();
184 if (!(*i)->visible()) {
185 /* if this group isn't visible we won't have updated p, so do it now */
186 p += 1;
190 return 0;
194 pair<boost::shared_ptr<PortGroup>, ARDOUR::BundleChannel>
195 PortMatrixComponent::position_to_group_and_channel (uint32_t p, PortGroupList const * groups) const
197 PortGroupList::List::const_iterator i = groups->begin ();
199 while (i != groups->end()) {
201 uint32_t const gs = group_size (*i);
203 if (p < gs) {
205 /* it's in this group */
207 if (!(*i)->visible()) {
208 return make_pair (*i, ARDOUR::BundleChannel (boost::shared_ptr<ARDOUR::Bundle> (), 0));
211 PortGroup::BundleList const & bundles = (*i)->bundles ();
212 for (PortGroup::BundleList::const_iterator j = bundles.begin(); j != bundles.end(); ++j) {
214 if (_matrix->show_only_bundles()) {
216 if (p == 0) {
217 return make_pair (*i, ARDOUR::BundleChannel (j->bundle, 0));
218 } else {
219 p -= 1;
222 } else {
224 uint32_t const s = j->bundle->nchannels ();
225 if (p < s) {
226 return make_pair (*i, ARDOUR::BundleChannel (j->bundle, p));
227 } else {
228 p -= s;
235 } else {
237 p -= gs;
241 ++i;
244 return make_pair (boost::shared_ptr<PortGroup> (), ARDOUR::BundleChannel (boost::shared_ptr<ARDOUR::Bundle> (), 0));