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