]> git.sesse.net Git - nageru/blob - futatabi/midi_mapping_dialog.h
Add a MIDI mapping editor for Futatabi.
[nageru] / futatabi / midi_mapping_dialog.h
1 #ifndef _MIDI_MAPPING_DIALOG_H
2 #define _MIDI_MAPPING_DIALOG_H
3
4 #include <stdbool.h>
5 #include <QDialog>
6 #include <QString>
7 #include <map>
8 #include <memory>
9 #include <string>
10 #include <utility>
11 #include <vector>
12
13 #include "midi_mapper.h"
14
15 class QEvent;
16 class QObject;
17
18 namespace Ui {
19 class MIDIMappingDialog;
20 }  // namespace Ui
21
22 class MIDIMappingProto;
23 class QComboBox;
24 class QSpinBox;
25 class QTreeWidgetItem;
26
27 class MIDIMappingDialog : public QDialog, public ControllerReceiver
28 {
29         Q_OBJECT
30
31 public:
32         MIDIMappingDialog(MIDIMapper *mapper);
33         ~MIDIMappingDialog();
34
35         // For use in midi_mapping_dialog.cpp only.
36         struct Control {
37                 std::string label;
38                 int field_number;  // In MIDIMappingBusProto.
39                 int bank_field_number;  // In MIDIMappingProto.
40         };
41
42         // ControllerReceiver interface. We only implement the raw events.
43         void preview() override {}
44         void queue() override {}
45         void play() override {}
46         void toggle_lock() override {}
47         void jog(int delta) override {}
48         void switch_camera(unsigned camera_idx) override {}
49         void set_master_speed(float speed) override {}
50         void cue_in() override {}
51         void cue_out() override {}
52
53         // Raw events; used for the editor dialog only.
54         void controller_changed(unsigned controller) override;
55         void note_on(unsigned note) override;
56
57 private:
58         void ok_clicked();
59         void cancel_clicked();
60         void save_clicked();
61         void load_clicked();
62
63         void add_bank_selector(QTreeWidgetItem *item, const MIDIMappingProto &mapping_proto, int bank_field_number);
64         
65         enum class ControlType { CONTROLLER, CONTROLLER_LIGHT, BUTTON, BUTTON_LIGHT, CAMERA_BUTTON, CAMERA_BUTTON_LIGHT };
66         void add_controls(const std::string &heading, ControlType control_type,
67                           const MIDIMappingProto &mapping_proto, const std::vector<Control> &controls);
68         void fill_controls_from_mapping(const MIDIMappingProto &mapping_proto);
69
70         std::unique_ptr<MIDIMappingProto> construct_mapping_proto_from_ui();
71
72         Ui::MIDIMappingDialog *ui;
73         MIDIMapper *mapper;
74         ControllerReceiver *old_receiver;
75
76         // All controllers actually laid out on the grid (we need to store them
77         // so that we can move values back and forth between the controls and
78         // the protobuf on save/load).
79         struct InstantiatedSpinner {
80                 QSpinBox *spinner;
81                 QSpinBox *spinner2;  // Value for button lights, min value for controller lights.
82                 QSpinBox *spinner3;  // Max value for controller lights.
83                 int field_number;  // In MIDIMappingBusProto.
84         };
85         struct InstantiatedComboBox {
86                 QComboBox *combo_box;
87                 int field_number;  // In MIDIMappingProto.
88         };
89         std::vector<InstantiatedSpinner> controller_spinners;
90         std::vector<InstantiatedSpinner> controller_light_spinners;
91         std::vector<InstantiatedSpinner> button_spinners;
92         std::vector<InstantiatedSpinner> button_light_spinners;
93         std::vector<InstantiatedSpinner> camera_button_spinners;  // One per camera.
94         std::vector<InstantiatedSpinner> camera_button_light_spinners;  // One per camera.
95         std::vector<InstantiatedComboBox> bank_combo_boxes;
96
97         // Keyed on field number.
98         std::map<unsigned, QSpinBox *> spinners;
99 };
100
101 #endif  // !defined(_MIDI_MAPPING_DIALOG_H)