2 Copyright (C) 2016 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.
21 #include "ardour/debug.h"
22 #include "ardour/mute_control.h"
23 #include "ardour/session.h"
24 #include "ardour/solo_control.h"
25 #include "ardour/solo_isolate_control.h"
27 #include "launch_control_xl.h"
29 using namespace ArdourSurface;
30 using namespace ARDOUR;
35 LaunchControlXL::build_maps ()
41 #define MAKE_KNOB(i,cc, index) \
42 knob = new Knob ((i), (cc), (index), (*this)); \
43 cc_knob_map.insert (std::make_pair (knob->controller_number(), knob)); \
44 id_knob_map.insert (std::make_pair (knob->id(), knob))
46 for (uint8_t n = 0; n < 8; ++n) {
47 MAKE_KNOB (static_cast<KnobID>(n), (n + 13), n);
48 MAKE_KNOB (static_cast<KnobID>(n + 8), (n + 29), (n + 8));
49 MAKE_KNOB (static_cast<KnobID>(n + 16), (n + 49), (n + 16));
56 #define MAKE_FADER(i,cc) \
57 fader = new Fader ((i), (cc)); \
58 cc_fader_map.insert (std::make_pair (fader->controller_number(), fader)); \
59 id_fader_map.insert (std::make_pair (fader->id(), fader))
61 for (uint8_t n = 0; n < 8; ++n) {
62 MAKE_FADER (static_cast<FaderID>(n), (n + 77) );
67 ControllerButton *controller_button;
68 NoteButton *note_button;
71 #define MAKE_TRACK_BUTTON_PRESS(i,nn,index,color,p) \
72 note_button = new TrackButton ((i), (nn), (index), (color), (p), (*this)); \
73 nn_note_button_map.insert (std::make_pair (note_button->note_number(), note_button)); \
74 id_note_button_map.insert (std::make_pair (note_button->id(), note_button))
75 #define MAKE_SELECT_BUTTON_PRESS(i,cc,index,p) \
76 controller_button = new SelectButton ((i), (cc), (index), (p), (*this)); \
77 cc_controller_button_map.insert (std::make_pair (controller_button->controller_number(), controller_button)); \
78 id_controller_button_map.insert (std::make_pair (controller_button->id(), controller_button))
79 #define MAKE_TRACK_STATE_BUTTON_PRESS(i,nn,index,p) \
80 note_button = new TrackStateButton ((i), (nn), (index), (p), (*this)); \
81 nn_note_button_map.insert (std::make_pair (note_button->note_number(), note_button)); \
82 id_note_button_map.insert (std::make_pair (note_button->id(), note_button))
83 #define MAKE_TRACK_STATE_BUTTON_PRESS_RELEASE_LONG(i,nn,index, p,r,l) \
84 note_button = new TrackStateButton ((i), (nn), (index), (p), (r), (l), (*this)); \
85 nn_note_button_map.insert (std::make_pair (note_button->note_number(), note_button)); \
86 id_note_button_map.insert (std::make_pair (note_button->id(), note_button))
89 MAKE_TRACK_BUTTON_PRESS(Focus1, 41, 24, YellowLow, &LaunchControlXL::button_track_focus_1);
90 MAKE_TRACK_BUTTON_PRESS(Focus2, 42, 25, YellowLow, &LaunchControlXL::button_track_focus_2);
91 MAKE_TRACK_BUTTON_PRESS(Focus3, 43, 26, YellowLow, &LaunchControlXL::button_track_focus_3);
92 MAKE_TRACK_BUTTON_PRESS(Focus4, 44, 27, YellowLow, &LaunchControlXL::button_track_focus_4);
93 MAKE_TRACK_BUTTON_PRESS(Focus5, 57, 28, YellowLow, &LaunchControlXL::button_track_focus_5);
94 MAKE_TRACK_BUTTON_PRESS(Focus6, 58, 29, YellowLow, &LaunchControlXL::button_track_focus_6);
95 MAKE_TRACK_BUTTON_PRESS(Focus7, 59, 30, YellowLow, &LaunchControlXL::button_track_focus_7);
96 MAKE_TRACK_BUTTON_PRESS(Focus8, 60, 31, YellowLow, &LaunchControlXL::button_track_focus_8);
97 MAKE_TRACK_BUTTON_PRESS(Control1, 73, 32, AmberLow, &LaunchControlXL::button_track_control_1);
98 MAKE_TRACK_BUTTON_PRESS(Control2, 74, 33, AmberLow, &LaunchControlXL::button_track_control_2);
99 MAKE_TRACK_BUTTON_PRESS(Control3, 75, 34, AmberLow, &LaunchControlXL::button_track_control_3);
100 MAKE_TRACK_BUTTON_PRESS(Control4, 76, 35, AmberLow, &LaunchControlXL::button_track_control_4);
101 MAKE_TRACK_BUTTON_PRESS(Control5, 89, 36, AmberLow, &LaunchControlXL::button_track_control_5);
102 MAKE_TRACK_BUTTON_PRESS(Control6, 90, 37, AmberLow, &LaunchControlXL::button_track_control_6);
103 MAKE_TRACK_BUTTON_PRESS(Control7, 91, 38, AmberLow, &LaunchControlXL::button_track_control_7);
104 MAKE_TRACK_BUTTON_PRESS(Control8, 92, 39, AmberLow, &LaunchControlXL::button_track_control_8);
106 MAKE_SELECT_BUTTON_PRESS(SelectUp, 104, 44, &LaunchControlXL::button_select_up);
107 MAKE_SELECT_BUTTON_PRESS(SelectDown, 105, 45, &LaunchControlXL::button_select_down);
108 MAKE_SELECT_BUTTON_PRESS(SelectLeft, 106, 46, &LaunchControlXL::button_select_left);
109 MAKE_SELECT_BUTTON_PRESS(SelectRight, 107, 47, &LaunchControlXL::button_select_right);
111 MAKE_TRACK_STATE_BUTTON_PRESS_RELEASE_LONG(Device, 105, 40, &LaunchControlXL::relax, &LaunchControlXL::button_device, &LaunchControlXL::button_device_long_press);;
112 MAKE_TRACK_STATE_BUTTON_PRESS(Mute, 106, 41, &LaunchControlXL::button_mute);
113 MAKE_TRACK_STATE_BUTTON_PRESS(Solo, 107, 42, &LaunchControlXL::button_solo);
114 MAKE_TRACK_STATE_BUTTON_PRESS(Record, 108, 43, &LaunchControlXL::button_record);
119 LaunchControlXL::button_name_by_id (ButtonID id)
133 return "Select Down";
135 return "Select Right";
137 return "Select Left";
178 LaunchControlXL::knob_name_by_id (KnobID id)
237 LaunchControlXL::fader_name_by_id (FaderID id)
263 LaunchControlXL::TrackButton*
264 LaunchControlXL::track_button_by_range(uint8_t n, uint8_t first, uint8_t middle)
266 NNNoteButtonMap::iterator b;
268 b = nn_note_button_map.find (first + n);
270 b = nn_note_button_map.find (middle + n - 4);
273 TrackButton* button = 0;
275 if (b != nn_note_button_map.end()) {
276 button = static_cast<TrackButton*>(b->second);
284 LaunchControlXL::update_track_focus_led(uint8_t n)
286 TrackButton* b = focus_button_by_column(n);
293 if ( stripable[n]->is_selected() ) {
294 b->set_color(YellowFull);
296 b->set_color(AmberLow);
302 write (b->state_msg());
306 LaunchControlXL::button_track_focus(uint8_t n)
308 if (buttons_down.find(Device) != buttons_down.end()) {
309 DEBUG_TRACE (DEBUG::LaunchControlXL, "DEVICE BUTTON HOLD\n");
310 stripable[n]->solo_isolate_control()->set_value (!stripable[n]->solo_isolate_control()->get_value(), PBD::Controllable::UseGroup);
315 if ( stripable[n]->is_selected() ) {
316 ControlProtocol::RemoveStripableFromSelection (stripable[n]);
318 ControlProtocol::AddStripableToSelection (stripable[n]);
327 boost::shared_ptr<AutomationControl>
328 LaunchControlXL::get_ac_by_state(uint8_t n) {
329 boost::shared_ptr<AutomationControl> ac;
331 switch(track_mode()) {
333 ac = stripable[n]->mute_control();
337 ac = stripable[n]->solo_control();
341 ac = stripable[n]->rec_enable_control();
350 LaunchControlXL::Knob**
351 LaunchControlXL::knobs_by_column(uint8_t col, Knob** knob_col)
353 for (uint8_t n = 0; n < 3; ++n) {
354 knob_col[n] = id_knob_map.find(static_cast<KnobID>(col+n*8))->second;
361 LaunchControlXL::update_knob_led(uint8_t n)
365 uint32_t absolute_strip_num = (n + bank_start) % 8;
368 switch (absolute_strip_num) {
371 if (stripable[n]->is_selected()) {
380 if (stripable[n]->is_selected()) {
389 if (stripable[n]->is_selected()) {
398 if (stripable[n]->is_master()) {
401 if (stripable[n]->is_selected()) {
411 knobs_by_column(n, knobs_col);
413 for (uint8_t s = 0; s < 3; ++s)
416 knobs_col[s]->set_color(color);
418 knobs_col[s]->set_color(Off);
420 write (knobs_col[s]->state_msg());
425 LaunchControlXL::update_track_control_led(uint8_t n)
427 TrackButton* b = control_button_by_column(n);
434 boost::shared_ptr<AutomationControl> ac = get_ac_by_state(n);
436 switch(track_mode()) {
438 if (ac->get_value()) {
439 b->set_color(YellowFull);
441 b->set_color(AmberLow);
445 if (ac && !(stripable[n]->is_master())) {
446 if (ac->get_value()) {
447 b->set_color(GreenFull);
449 b->set_color(GreenLow);
457 if (ac->get_value()) {
458 b->set_color(RedFull);
460 b->set_color(RedLow);
474 write (b->state_msg());
478 LaunchControlXL::solo_mute_rec_changed(uint32_t n) {
482 update_track_control_led(n);
486 LaunchControlXL::button_track_control(uint8_t n) {
491 if (buttons_down.find(Device) != buttons_down.end()) {
492 DEBUG_TRACE (DEBUG::LaunchControlXL, "DEVICE BUTTON HOLD\n");
494 if (stripable[n] != master) {
495 DEBUG_TRACE (DEBUG::LaunchControlXL, "MIXBUS Master Assign\n");
496 stripable[n]->master_send_enable_controllable()->set_value (!stripable[n]->master_send_enable_controllable()->get_value(), PBD::Controllable::UseGroup);
499 /* something useful for Ardour */
504 boost::shared_ptr<AutomationControl> ac = get_ac_by_state(n);
507 session->set_control (ac, !ac->get_value(), PBD::Controllable::UseGroup);
512 LaunchControlXL::button_track_mode(TrackMode state)
514 set_track_mode(state);
515 for (uint8_t n = 0; n < 8; ++n) {
516 update_track_control_led(n);
519 TrackStateButton* mute = static_cast<TrackStateButton*>(id_note_button_map[Mute]);
520 TrackStateButton* solo = static_cast<TrackStateButton*>(id_note_button_map[Solo]);
521 TrackStateButton* record = static_cast<TrackStateButton*>(id_note_button_map[Record]);
523 write(mute->state_msg((state == TrackMute)));
524 write(solo->state_msg((state == TrackSolo)));
525 write(record->state_msg((state == TrackRecord)));
529 LaunchControlXL::button_select_left()
531 switch_bank (max (0, bank_start - (7 + (LaunchControlXL::use_fader8master))));
535 LaunchControlXL::button_select_right()
537 switch_bank (max (0, bank_start + 7 + (LaunchControlXL::use_fader8master)));
541 LaunchControlXL::button_select_up()
547 LaunchControlXL::button_select_down()
553 LaunchControlXL::button_device()
559 LaunchControlXL::button_device_long_press()
565 LaunchControlXL::button_mute()
567 if (buttons_down.find(Device) != buttons_down.end()) {
568 access_action ("Editor/track-mute-toggle");
570 button_track_mode(TrackMute);
575 LaunchControlXL::button_solo()
577 if (buttons_down.find(Device) != buttons_down.end()) {
578 access_action ("Editor/track-solo-toggle");
580 button_track_mode(TrackSolo);
585 LaunchControlXL::button_record()
587 if (buttons_down.find(Device) != buttons_down.end()) {
588 access_action ("Editor/track-record-enable-toggle");
590 button_track_mode(TrackRecord);
595 LaunchControlXL::button_long_press_timeout (ButtonID id, Button* button)
597 if (buttons_down.find (id) != buttons_down.end()) {
598 DEBUG_TRACE (DEBUG::LaunchControlXL, string_compose ("long press timeout for %1, invoking method\n", id));
599 (this->*button->long_press_method) ();
601 DEBUG_TRACE (DEBUG::LaunchControlXL, string_compose ("long press timeout for %1, expired/cancelled\n", id));
602 /* release happened and somehow we were not cancelled */
605 /* whichever button this was, we've used it ... don't invoke the
608 consumed.insert (id);
610 return false; /* don't get called again */
615 LaunchControlXL::start_press_timeout (Button* button, ButtonID id)
617 Glib::RefPtr<Glib::TimeoutSource> timeout = Glib::TimeoutSource::create (500); // milliseconds
618 button->timeout_connection = timeout->connect (sigc::bind (sigc::mem_fun (*this, &LaunchControlXL::button_long_press_timeout), id, button));
619 timeout->attach (main_loop()->get_context());