Fix stacking of automation region views.
[ardour.git] / gtk2_ardour / automation_region_view.cc
1 /*
2     Copyright (C) 2007 Paul Davis
3     Author: Dave Robillard
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20 #include "pbd/memento_command.h"
21 #include "ardour/automation_control.h"
22 #include "ardour/event_type_map.h"
23 #include "ardour/session.h"
24 #include "ardour/source.h"
25 #include "automation_region_view.h"
26 #include "public_editor.h"
27
28 #include "i18n.h"
29
30 AutomationRegionView::AutomationRegionView(ArdourCanvas::Group*                      parent,
31                                            AutomationTimeAxisView&                   time_axis,
32                                            boost::shared_ptr<ARDOUR::Region>         region,
33                                            const Evoral::Parameter&                  param,
34                                            boost::shared_ptr<ARDOUR::AutomationList> list,
35                                            double                                    spu,
36                                            Gdk::Color const &                        basic_color)
37         : RegionView(parent, time_axis, region, spu, basic_color)
38         , _parameter(param)
39 {
40         if (list) {
41                 assert(list->parameter() == param);
42                 create_line(list);
43         }
44
45         group->signal_event().connect (mem_fun (this, &AutomationRegionView::canvas_event), false);
46         group->raise_to_top();
47 }
48
49 void
50 AutomationRegionView::init (Gdk::Color const & basic_color, bool /*wfd*/)
51 {
52         _enable_display = false;
53
54         RegionView::init(basic_color, false);
55
56         compute_colors (basic_color);
57
58         reset_width_dependent_items ((double) _region->length() / samples_per_unit);
59
60         set_height (trackview.current_height());
61
62         _region->StateChanged.connect (mem_fun(*this, &AutomationRegionView::region_changed));
63
64         set_colors ();
65
66         _enable_display = true;
67 }
68
69 void
70 AutomationRegionView::create_line (boost::shared_ptr<ARDOUR::AutomationList> list)
71 {
72         _line = boost::shared_ptr<AutomationLine>(new AutomationLine(
73                                 ARDOUR::EventTypeMap::instance().to_symbol(list->parameter()),
74                                 trackview, *get_canvas_group(), list, &_time_converter));
75         _line->set_colors();
76         _line->set_interpolation(list->interpolation());
77         _line->set_height ((uint32_t)rint(trackview.current_height() - NAME_HIGHLIGHT_SIZE));
78         _line->show();
79         _line->show_all_control_points();
80 }
81
82 bool
83 AutomationRegionView::canvas_event(GdkEvent* ev)
84 {
85         if (ev->type == GDK_BUTTON_RELEASE) {
86
87                 const nframes_t when = trackview.editor().pixel_to_frame((nframes_t)ev->button.x)
88                         - _region->position();
89                 add_automation_event(ev, when, ev->button.y);
90         }
91
92         return false;
93 }
94
95 void
96 AutomationRegionView::add_automation_event (GdkEvent* /*event*/, nframes_t when, double y)
97 {
98         if (!_line) {
99                 boost::shared_ptr<Evoral::Control> c = _region->control(_parameter, true);
100                 boost::shared_ptr<ARDOUR::AutomationControl> ac
101                                 = boost::dynamic_pointer_cast<ARDOUR::AutomationControl>(c);
102                 assert(ac);
103                 create_line(ac->alist());
104         }
105         assert(_line);
106
107         double x = 0;
108         AutomationTimeAxisView* const view = automation_view();
109
110         view->canvas_display()->w2i (x, y);
111
112         /* compute vertical fractional position */
113
114         const double h = trackview.current_height() - TimeAxisViewItem::NAME_HIGHLIGHT_SIZE - 2;
115         y = 1.0 - (y / h);
116
117         /* map using line */
118
119         _line->view_to_model_coord (x, y);
120
121         view->session().begin_reversible_command (_("add automation event"));
122         XMLNode& before = _line->the_list()->get_state();
123
124         _line->the_list()->add (when, y);
125
126         XMLNode& after = _line->the_list()->get_state();
127         view->session().commit_reversible_command (new MementoCommand<ARDOUR::AutomationList>(
128                         *_line->the_list(), &before, &after));
129
130         view->session().set_dirty ();
131 }
132
133 void
134 AutomationRegionView::set_height (double h)
135 {
136         RegionView::set_height(h);
137
138         if (_line)
139                 _line->set_height ((uint32_t)rint(h - NAME_HIGHLIGHT_SIZE));
140 }
141
142 bool
143 AutomationRegionView::set_position (nframes64_t pos, void* src, double* ignored)
144 {
145         return RegionView::set_position(pos, src, ignored);
146 }
147
148
149 void
150 AutomationRegionView::reset_width_dependent_items (double pixel_width)
151 {
152         RegionView::reset_width_dependent_items(pixel_width);
153
154         if (_line)
155                 _line->reset();
156 }
157
158
159 void
160 AutomationRegionView::region_resized (ARDOUR::Change what_changed)
161 {
162         RegionView::region_resized(what_changed);
163
164         if (_line)
165                 _line->reset();
166 }
167
168
169 void
170 AutomationRegionView::entered()
171 {
172         if (_line)
173                 _line->track_entered();
174 }
175
176
177 void
178 AutomationRegionView::exited()
179 {
180         if (_line)
181                 _line->track_exited();
182 }
183