]> git.sesse.net Git - nageru/blob - midi_mapper.h
2b59b9ff53428cb1a02656049313014ee04548b4
[nageru] / midi_mapper.h
1 #ifndef _MIDI_MAPPER_H
2 #define _MIDI_MAPPER_H 1
3
4 // MIDIMapper is a class that listens for incoming MIDI messages from
5 // mixer controllers (ie., it is not meant to be used with regular
6 // instruments), 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.
10
11 #include <atomic>
12 #include <functional>
13 #include <memory>
14 #include <string>
15 #include <thread>
16
17 class MIDIMappingProto;
18 typedef struct snd_seq_addr snd_seq_addr_t;
19 typedef struct snd_seq_event snd_seq_event_t;
20 typedef struct _snd_seq snd_seq_t;
21
22 // Interface for receiving interpreted controller messages.
23 class ControllerReceiver {
24 public:
25         // All values are [0.0, 1.0].
26         virtual void set_locut(float value) = 0;
27         virtual void set_limiter_threshold(float value) = 0;
28         virtual void set_makeup_gain(float value) = 0;
29
30         virtual void set_treble(unsigned bus_idx, float value) = 0;
31         virtual void set_mid(unsigned bus_idx, float value) = 0;
32         virtual void set_bass(unsigned bus_idx, float value) = 0;
33         virtual void set_gain(unsigned bus_idx, float value) = 0;
34         virtual void set_compressor_threshold(unsigned bus_idx, float value) = 0;
35         virtual void set_fader(unsigned bus_idx, float value) = 0;
36
37         virtual void toggle_locut(unsigned bus_idx) = 0;
38         virtual void toggle_auto_gain_staging(unsigned bus_idx) = 0;
39         virtual void toggle_compressor(unsigned bus_idx) = 0;
40         virtual void clear_peak(unsigned bus_idx) = 0;
41 };
42
43 class MIDIMapper {
44 public:
45         MIDIMapper(ControllerReceiver *receiver);
46         virtual ~MIDIMapper();
47         void set_midi_mapping(const MIDIMappingProto &new_mapping);
48         void start_thread();
49         const MIDIMappingProto &get_current_mapping() const { return *mapping_proto; }
50
51 private:
52         void thread_func();
53         void handle_event(snd_seq_t *seq, snd_seq_event_t *event);
54         void subscribe_to_port(snd_seq_t *seq, const snd_seq_addr_t &addr);
55         void match_controller(int controller, int field_number, int bank_field_number, float value, std::function<void(unsigned, float)> func);
56         void match_button(int note, int field_number, int bank_field_number, std::function<void(unsigned)> func);
57         bool bank_mismatch(int bank_field_number);
58
59         ControllerReceiver *receiver;
60         std::atomic<bool> should_quit{false};
61         int should_quit_fd;
62
63         std::unique_ptr<MIDIMappingProto> mapping_proto;
64         int num_controller_banks;
65         int current_controller_bank = 0;
66
67         std::thread midi_thread;
68 };
69
70 bool load_midi_mapping_from_file(const std::string &filename, MIDIMappingProto *new_mapping);
71
72 #endif  // !defined(_MIDI_MAPPER_H)