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