2 #define _MIDI_MAPPER_H 1
4 // MIDIMapper is a class that gets incoming MIDI messages from mixer
5 // controllers (ie., it is not meant to be used with regular instruments)
6 // via MIDIDevice, interprets them according to a device-specific, user-defined
7 // mapping, and calls back into a receiver (typically the MainWindow).
8 // This way, it is possible to control audio functionality using physical
9 // pots and faders instead of the mouse.
20 #include "shared/midi_device.h"
22 class MIDIMappingProto;
24 // Interface for receiving interpreted controller messages.
25 class ControllerReceiver {
27 virtual ~ControllerReceiver() {}
29 // All values are [0.0, 1.0].
30 virtual void set_locut(float value) = 0;
31 virtual void set_limiter_threshold(float value) = 0;
32 virtual void set_makeup_gain(float value) = 0;
34 virtual void set_stereo_width(unsigned bus_idx, float value) = 0;
35 virtual void set_treble(unsigned bus_idx, float value) = 0;
36 virtual void set_mid(unsigned bus_idx, float value) = 0;
37 virtual void set_bass(unsigned bus_idx, float value) = 0;
38 virtual void set_gain(unsigned bus_idx, float value) = 0;
39 virtual void set_compressor_threshold(unsigned bus_idx, float value) = 0;
40 virtual void set_fader(unsigned bus_idx, float value) = 0;
42 virtual void toggle_mute(unsigned bus_idx) = 0;
43 virtual void toggle_locut(unsigned bus_idx) = 0;
44 virtual void toggle_auto_gain_staging(unsigned bus_idx) = 0;
45 virtual void toggle_compressor(unsigned bus_idx) = 0;
46 virtual void clear_peak(unsigned bus_idx) = 0;
47 virtual void toggle_limiter() = 0;
48 virtual void toggle_auto_makeup_gain() = 0;
50 // Signals to highlight controls to mark them to the user
51 // as MIDI-controllable (or not).
52 virtual void clear_all_highlights() = 0;
54 virtual void highlight_locut(bool highlight) = 0;
55 virtual void highlight_limiter_threshold(bool highlight) = 0;
56 virtual void highlight_makeup_gain(bool highlight) = 0;
58 virtual void highlight_stereo_width(unsigned bus_idx, bool highlight) = 0;
59 virtual void highlight_treble(unsigned bus_idx, bool highlight) = 0;
60 virtual void highlight_mid(unsigned bus_idx, bool highlight) = 0;
61 virtual void highlight_bass(unsigned bus_idx, bool highlight) = 0;
62 virtual void highlight_gain(unsigned bus_idx, bool highlight) = 0;
63 virtual void highlight_compressor_threshold(unsigned bus_idx, bool highlight) = 0;
64 virtual void highlight_fader(unsigned bus_idx, bool highlight) = 0;
66 virtual void highlight_mute(unsigned bus_idx, bool highlight) = 0;
67 virtual void highlight_toggle_locut(unsigned bus_idx, bool highlight) = 0;
68 virtual void highlight_toggle_auto_gain_staging(unsigned bus_idx, bool highlight) = 0;
69 virtual void highlight_toggle_compressor(unsigned bus_idx, bool highlight) = 0;
70 virtual void highlight_clear_peak(unsigned bus_idx, bool highlight) = 0;
71 virtual void highlight_toggle_limiter(bool highlight) = 0;
72 virtual void highlight_toggle_auto_makeup_gain(bool highlight) = 0;
74 // Raw events; used for the editor dialog only.
75 virtual void controller_changed(unsigned controller) = 0;
76 virtual void note_on(unsigned note) = 0;
79 class MIDIMapper : public MIDIReceiver {
81 MIDIMapper(ControllerReceiver *receiver);
82 virtual ~MIDIMapper();
83 void set_midi_mapping(const MIDIMappingProto &new_mapping);
85 const MIDIMappingProto &get_current_mapping() const;
87 // Overwrites and returns the previous value.
88 ControllerReceiver *set_receiver(ControllerReceiver *new_receiver);
90 void refresh_highlights();
91 void refresh_lights();
93 void set_has_peaked(unsigned bus_idx, bool has_peaked)
95 this->has_peaked[bus_idx] = has_peaked;
99 void controller_received(int controller, int value) override;
100 void note_on_received(int note) override;
101 void update_num_subscribers(unsigned num_subscribers) override;
104 void match_controller(int controller, int field_number, int bank_field_number, float value, std::function<void(unsigned, float)> func);
105 void match_button(int note, int field_number, int bank_field_number, std::function<void(unsigned)> func);
106 bool has_active_controller(unsigned bus_idx, int field_number, int bank_field_number); // Also works for buttons.
107 bool bank_mismatch(int bank_field_number);
109 void update_highlights();
111 void update_lights_lock_held();
112 void activate_lights_all_buses(int field_number, std::map<MIDIDevice::LightKey, uint8_t> *active_lights);
114 std::atomic<bool> should_quit{false};
116 std::atomic<bool> has_peaked[MAX_BUSES] {{ false }};
118 mutable std::mutex mu;
119 ControllerReceiver *receiver; // Under <mu>.
120 std::unique_ptr<MIDIMappingProto> mapping_proto; // Under <mu>.
121 int num_controller_banks; // Under <mu>.
122 std::atomic<int> current_controller_bank{0};
123 std::atomic<int> num_subscribed_ports{0};
125 MIDIDevice midi_device;
128 bool load_midi_mapping_from_file(const std::string &filename, MIDIMappingProto *new_mapping);
129 bool save_midi_mapping_to_file(const MIDIMappingProto &mapping_proto, const std::string &filename);
131 #endif // !defined(_MIDI_MAPPER_H)