Merge remote-tracking branch 'remotes/origin/exportvis' into windows+cc
[ardour.git] / gtk2_ardour / note_base.cc
1 /*
2     Copyright (C) 2007 Paul Davis
3     Author: David 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 <iostream>
21
22 #include "gtkmm2ext/keyboard.h"
23
24 #include "canvas/text.h"
25
26 #include "note_base.h"
27 #include "midi_region_view.h"
28 #include "public_editor.h"
29 #include "editing_syms.h"
30 #include "keyboard.h"
31
32 using namespace std;
33 using namespace Gtkmm2ext;
34 using ARDOUR::MidiModel;
35 using namespace ArdourCanvas;
36
37 PBD::Signal1<void,NoteBase*> NoteBase::NoteBaseDeleted;
38
39 /// dividing the hue circle in 16 parts, hand adjusted for equal look, courtesy Thorsten Wilms
40 const uint32_t NoteBase::midi_channel_colors[16] = {
41           0xd32d2dff,  0xd36b2dff,  0xd3972dff,  0xd3d12dff,
42           0xa0d32dff,  0x7dd32dff,  0x2dd45eff,  0x2dd3c4ff,
43           0x2da5d3ff,  0x2d6fd3ff,  0x432dd3ff,  0x662dd3ff,
44           0x832dd3ff,  0xa92dd3ff,  0xd32dbfff,  0xd32d67ff
45         };
46
47 NoteBase::NoteBase(MidiRegionView& region, bool with_events, const boost::shared_ptr<NoteType> note)
48         : _region(region)
49         , _item (0)
50         , _text(0)
51         , _state(None)
52         , _note(note)
53         , _with_events (with_events)
54         , _selected(false)
55         , _valid (true)
56         , _mouse_x_fraction (-1.0)
57         , _mouse_y_fraction (-1.0)
58 {
59 }
60
61 NoteBase::~NoteBase()
62 {
63         NoteBaseDeleted (this);
64
65         delete _text;
66 }
67
68 void
69 NoteBase::set_item (Item* item)
70 {
71         _item = item;
72         _item->set_data ("notebase", this);
73
74         if (_with_events) {
75                 _item->Event.connect (sigc::mem_fun (*this, &NoteBase::event_handler));
76         }
77 }
78
79 void
80 NoteBase::invalidate ()
81 {
82         _valid = false;
83 }
84
85 void
86 NoteBase::validate ()
87 {
88         _valid = true;
89 }
90
91 void
92 NoteBase::show_velocity()
93 {
94         if (!_text) {
95                 _text = new Text (_item->parent ());
96                 _text->set_ignore_events (true);
97                 _text->set_color (ARDOUR_UI::config()->get_canvasvar_MidiNoteVelocityText());
98                 _text->set_alignment (Pango::ALIGN_CENTER);
99         }
100
101         _text->set_x_position ((x0() + x1()) / 2);
102         _text->set_y_position ((y0() + y1()) / 2);
103         ostringstream velo(ios::ate);
104         velo << int(_note->velocity());
105         _text->set (velo.str ());
106         _text->show();
107         _text->raise_to_top();
108 }
109
110 void
111 NoteBase::hide_velocity()
112 {
113         delete _text;
114         _text = 0;
115 }
116
117 void
118 NoteBase::on_channel_selection_change(uint16_t selection)
119 {
120         // make note change its color if its channel is not marked active
121         if ( (selection & (1 << _note->channel())) == 0 ) {
122                 set_fill_color(ARDOUR_UI::config()->get_canvasvar_MidiNoteInactiveChannel());
123                 set_outline_color(calculate_outline(ARDOUR_UI::config()->get_canvasvar_MidiNoteInactiveChannel()));
124         } else {
125                 // set the color according to the notes selection state
126                 set_selected(_selected);
127         }
128         // this forces the item to update..... maybe slow...
129         _item->hide();
130         _item->show();
131 }
132
133 void
134 NoteBase::on_channel_change(uint8_t channel)
135 {
136         _region.note_selected(this, true);
137         _region.change_channel(channel);
138 }
139
140 void
141 NoteBase::set_selected(bool selected)
142 {
143         if (!_note) {
144                 return;
145         }
146
147         _selected = selected;
148         set_fill_color (base_color());
149         
150         if (_selected) {
151                 set_outline_color(calculate_outline(ARDOUR_UI::config()->get_canvasvar_MidiNoteSelected()));
152         } else {
153                 set_outline_color(calculate_outline(base_color()));
154         }
155
156 }
157
158 #define SCALE_USHORT_TO_UINT8_T(x) ((x) / 257)
159
160 uint32_t
161 NoteBase::base_color()
162 {
163         using namespace ARDOUR;
164
165         ColorMode mode = _region.color_mode();
166
167         const uint8_t min_opacity = 15;
168         uint8_t       opacity = std::max(min_opacity, uint8_t(_note->velocity() + _note->velocity()));
169
170         switch (mode) {
171         case TrackColor:
172         {
173                 Gdk::Color color = _region.midi_stream_view()->get_region_color();
174                 return UINT_INTERPOLATE (RGBA_TO_UINT(
175                                                  SCALE_USHORT_TO_UINT8_T(color.get_red()),
176                                                  SCALE_USHORT_TO_UINT8_T(color.get_green()),
177                                                  SCALE_USHORT_TO_UINT8_T(color.get_blue()),
178                                                  opacity), 
179                                          ARDOUR_UI::config()->get_canvasvar_MidiNoteSelected(), 0.5);
180         }
181
182         case ChannelColors:
183                 return UINT_INTERPOLATE (UINT_RGBA_CHANGE_A (NoteBase::midi_channel_colors[_note->channel()],
184                                                              opacity), 
185                                          ARDOUR_UI::config()->get_canvasvar_MidiNoteSelected(), 0.5);
186
187         default:
188                 return meter_style_fill_color(_note->velocity(), selected());
189         };
190
191         return 0;
192 }
193
194 void
195 NoteBase::set_mouse_fractions (GdkEvent* ev)
196 {
197         double ix, iy;
198         bool set_cursor = false;
199
200         switch (ev->type) {
201         case GDK_MOTION_NOTIFY:
202                 ix = ev->motion.x;
203                 iy = ev->motion.y;
204                 set_cursor = true;
205                 break;
206         case GDK_ENTER_NOTIFY:
207                 ix = ev->crossing.x;
208                 iy = ev->crossing.y;
209                 set_cursor = true;
210                 break;
211         case GDK_BUTTON_PRESS:
212         case GDK_BUTTON_RELEASE:
213                 ix = ev->button.x;
214                 iy = ev->button.y;
215                 break;
216         default:
217                 _mouse_x_fraction = -1.0;
218                 _mouse_y_fraction = -1.0;
219                 return;
220         }
221
222         boost::optional<ArdourCanvas::Rect> bbox = _item->bounding_box ();
223         assert (bbox);
224
225         _item->canvas_to_item (ix, iy);
226         /* XXX: CANVAS */
227         /* hmm, something wrong here. w2i should give item-local coordinates
228            but it doesn't. for now, finesse this.
229         */
230         ix = ix - bbox.get().x0;
231         iy = iy - bbox.get().y0;
232
233         /* fraction of width/height */
234         double xf;
235         double yf;
236         bool notify = false;
237
238         xf = ix / bbox.get().width ();
239         yf = iy / bbox.get().height ();
240
241         if (xf != _mouse_x_fraction || yf != _mouse_y_fraction) {
242                 notify = true;
243         }
244
245         _mouse_x_fraction = xf;
246         _mouse_y_fraction = yf;
247
248         if (notify) {
249                 if (big_enough_to_trim()) {
250                         _region.note_mouse_position (_mouse_x_fraction, _mouse_y_fraction, set_cursor);
251                 } else {
252                         /* pretend the mouse is in the middle, because this is not big enough
253                            to trim right now.
254                         */
255                         _region.note_mouse_position (0.5, 0.5, set_cursor);
256                 }
257         }
258 }
259
260 bool
261 NoteBase::event_handler (GdkEvent* ev)
262 {
263         if (!_region.get_time_axis_view().editor().internal_editing()) {
264                 return false;
265         }
266
267         switch (ev->type) {
268         case GDK_ENTER_NOTIFY:
269                 set_mouse_fractions (ev);
270                 _region.note_entered (this);
271                 break;
272
273         case GDK_LEAVE_NOTIFY:
274                 set_mouse_fractions (ev);
275                 _region.note_left (this);
276                 break;
277
278         case GDK_MOTION_NOTIFY:
279                 set_mouse_fractions (ev);
280                 break;
281
282         case GDK_BUTTON_PRESS:
283                 set_mouse_fractions (ev);
284                 if (ev->button.button == 3 && Keyboard::no_modifiers_active (ev->button.state) && _selected) {
285                         _region.get_time_axis_view().editor().edit_notes (_region);
286                         return true;
287                 }
288                 break;
289
290         case GDK_BUTTON_RELEASE:
291                 set_mouse_fractions (ev);
292                 if (ev->button.button == 3 && Keyboard::no_modifiers_active (ev->button.state)) {
293                         return true;
294                 }
295                 break;
296
297         default:
298                 break;
299         }
300
301         return _region.get_time_axis_view().editor().canvas_note_event (ev, _item);
302 }
303
304 bool
305 NoteBase::mouse_near_ends () const
306 {
307         return (_mouse_x_fraction >= 0.0 && _mouse_x_fraction < 0.25) ||
308                 (_mouse_x_fraction >= 0.75 && _mouse_x_fraction < 1.0);
309 }
310
311 bool
312 NoteBase::big_enough_to_trim () const
313 {
314         return (x1() - x0()) > 10;
315 }
316