Don't include program-name with <Option name="color-file" />
[ardour.git] / gtk2_ardour / plugin_ui.h
1 /*
2     Copyright (C) 2000-2006 Paul Davis
3
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.
8
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.
13
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.
17
18 */
19
20 #ifndef __ardour_plugin_ui_h__
21 #define __ardour_plugin_ui_h__
22
23 #ifdef WAF_BUILD
24 #include "gtk2ardour-config.h"
25 #endif
26
27 #include <vector>
28 #include <map>
29 #include <list>
30
31 #include <sigc++/signal.h>
32
33 #include <gtkmm/button.h>
34 #include <gtkmm/box.h>
35 #include <gtkmm/table.h>
36 #include <gtkmm/eventbox.h>
37 #include <gtkmm/viewport.h>
38 #include <gtkmm/scrolledwindow.h>
39 #include <gtkmm/label.h>
40 #include <gtkmm/menu.h>
41 #include <gtkmm/image.h>
42 #include <gtkmm/adjustment.h>
43 #include <gtkmm/togglebutton.h>
44 #include <gtkmm/socket.h>
45 #include <gtkmm/socket.h>
46
47 #include "ardour/types.h"
48 #include "ardour/plugin.h"
49 #include "ardour/variant.h"
50
51 #include "ardour_button.h"
52 #include "ardour_dropdown.h"
53 #include "ardour_spinner.h"
54 #include "ardour_window.h"
55 #include "automation_controller.h"
56
57 namespace ARDOUR {
58         class PluginInsert;
59         class Plugin;
60         class WindowsVSTPlugin;
61         class LXVSTPlugin;
62         class IOProcessor;
63         class AUPlugin;
64         class Processor;
65 }
66
67 namespace PBD {
68         class Controllable;
69 }
70
71 namespace Gtkmm2ext {
72         class HSliderController;
73         class BarController;
74         class ClickBox;
75         class FastMeter;
76         class PixmapButton;
77 }
78
79 class LatencyGUI;
80 class ArdourWindow;
81 class PluginEqGui;
82 class VSTPluginUI;
83
84 class PlugUIBase : public virtual sigc::trackable, public PBD::ScopedConnectionList
85 {
86   public:
87         PlugUIBase (boost::shared_ptr<ARDOUR::PluginInsert>);
88         virtual ~PlugUIBase();
89
90         virtual gint get_preferred_height () = 0;
91         virtual gint get_preferred_width () = 0;
92         virtual bool resizable () { return true; }
93         virtual bool start_updating(GdkEventAny*) = 0;
94         virtual bool stop_updating(GdkEventAny*) = 0;
95
96         virtual void activate () {}
97         virtual void deactivate () {}
98
99         void update_preset_list ();
100         void update_preset ();
101
102         void latency_button_clicked ();
103
104         virtual bool on_window_show(const std::string& /*title*/) { return true; }
105         virtual void on_window_hide() {}
106
107         virtual void forward_key_event (GdkEventKey*) {}
108         virtual void grab_focus () {}
109   virtual bool non_gtk_gui() const { return false; }
110
111         sigc::signal<void,bool> KeyboardFocused;
112
113   protected:
114         boost::shared_ptr<ARDOUR::PluginInsert> insert;
115         boost::shared_ptr<ARDOUR::Plugin> plugin;
116
117         /* UI elements that can subclasses can add to their widgets */
118
119         /** a ComboBoxText which lists presets and manages their selection */
120         ArdourDropdown _preset_combo;
121         /** a label which has a * in if the current settings are different from the preset being shown */
122         Gtk::Label _preset_modified;
123         /** a button to add a preset */
124         ArdourButton add_button;
125         /** a button to save the current settings as a new user preset */
126         ArdourButton save_button;
127         /** a button to delete the current preset (if it is a user one) */
128         ArdourButton delete_button;
129         /** a button to delete the reset the plugin params */
130         ArdourButton reset_button;
131         /** a button to bypass the plugin */
132         ArdourButton bypass_button;
133         /** and self-explaining button :) */
134         ArdourButton pin_management_button;
135         /** a button to acquire keyboard focus */
136         Gtk::EventBox focus_button;
137         /** an expander containing the plugin description */
138         Gtk::Expander description_expander;
139         /** an expander containing the plugin analysis graph */
140         Gtk::Expander plugin_analysis_expander;
141         /** a button which, when clicked, opens the latency GUI */
142         ArdourButton latency_button;
143         /** a button which sets all controls' automation setting to Manual */
144         ArdourButton automation_manual_all_button;
145         /** a button which sets all controls' automation setting to Play */
146         ArdourButton automation_play_all_button;
147     /** a button which sets all controls' automation setting to Write */
148         ArdourButton automation_write_all_button;
149         /** a button which sets all controls' automation setting to Touch */
150         ArdourButton automation_touch_all_button;
151
152         void set_latency_label ();
153
154         LatencyGUI* latency_gui;
155         ArdourWindow* latency_dialog;
156
157         PluginEqGui* eqgui;
158
159         Gtk::Image* focus_out_image;
160         Gtk::Image* focus_in_image;
161         int _no_load_preset;
162
163         virtual void preset_selected (ARDOUR::Plugin::PresetRecord preset);
164         void add_plugin_setting ();
165         void save_plugin_setting ();
166         void delete_plugin_setting ();
167         void reset_plugin_parameters ();
168         void manage_pins ();
169         bool focus_toggled(GdkEventButton*);
170         bool bypass_button_release(GdkEventButton*);
171         void toggle_description ();
172         void toggle_plugin_analysis ();
173         void processor_active_changed (boost::weak_ptr<ARDOUR::Processor> p);
174         void plugin_going_away ();
175         void automation_state_changed ();
176         void preset_added_or_removed ();
177         void update_preset_modified ();
178
179         PBD::ScopedConnection death_connection;
180         PBD::ScopedConnection active_connection;
181         PBD::ScopedConnection preset_added_connection;
182         PBD::ScopedConnection preset_removed_connection;
183         PBD::ScopedConnectionList control_connections;
184 };
185
186 class GenericPluginUI : public PlugUIBase, public Gtk::VBox
187 {
188   public:
189         GenericPluginUI (boost::shared_ptr<ARDOUR::PluginInsert> plug, bool scrollable=false);
190         ~GenericPluginUI ();
191
192         gint get_preferred_height () { return prefheight; }
193         gint get_preferred_width () { return -1; }
194
195         bool start_updating(GdkEventAny*);
196         bool stop_updating(GdkEventAny*);
197
198   private:
199         Gtk::VBox main_contents;
200         Gtk::HBox settings_box;
201         Gtk::HBox hpacker;
202         Gtk::Menu* automation_menu;
203
204         gint prefheight;
205         bool is_scrollable;
206
207         struct MeterInfo {
208                 Gtkmm2ext::FastMeter *meter;
209
210                 float           min;
211                 float           max;
212                 bool            min_unbound;
213                 bool            max_unbound;
214                 bool packed;
215
216                 MeterInfo () {
217                         meter = 0;
218                         packed = false;
219                         min = 1.0e10;
220                         max = -1.0e10;
221                         min_unbound = false;
222                         max_unbound = false;
223                 }
224         };
225
226         /* FIXME: Unify with AutomationController */
227         struct ControlUI : public Gtk::HBox {
228
229                 const Evoral::Parameter parameter() const { return param; }
230
231                 Evoral::Parameter                            param;
232                 boost::shared_ptr<ARDOUR::AutomationControl> control;
233
234                 /* input */
235
236                 boost::shared_ptr<ARDOUR::ScalePoints>  scale_points;
237                 boost::shared_ptr<AutomationController> controller;
238
239                 ArdourButton                            automate_button;
240                 Gtk::Label                              label;
241                 ArdourDropdown*                         combo;
242                 Gtkmm2ext::ClickBox*                    clickbox;
243                 Gtk::FileChooserButton*                 file_button;
244                 ArdourSpinner*                          spin_box;
245
246                 bool                                    button;
247                 bool                                    update_pending;
248                 bool                                    ignore_change;
249
250                 /* output */
251
252                 Gtk::EventBox* display;
253                 Gtk::Label*    display_label;
254
255                 Gtk::HBox*     hbox;
256                 Gtk::VBox*     vbox;
257                 MeterInfo*     meterinfo;
258
259                 ControlUI (const Evoral::Parameter& param);
260                 ~ControlUI ();
261
262                 /* layout */
263                 Gtk::Table* knobtable;
264                 int x0, x1, y0, y1;
265         };
266
267         std::vector<ControlUI*>   input_controls; // workaround for preset load
268         std::vector<ControlUI*>   input_controls_with_automation;
269         std::vector<ControlUI*>   output_controls;
270
271         sigc::connection screen_update_connection;
272
273         void output_update();
274
275         void build ();
276         void automatic_layout (const std::vector<ControlUI *>& control_uis);
277         void custom_layout (const std::vector<ControlUI *>& control_uis);
278
279         ControlUI* build_control_ui (const Evoral::Parameter&                     param,
280                                      const ARDOUR::ParameterDescriptor&           desc,
281                                      boost::shared_ptr<ARDOUR::AutomationControl> mcontrol,
282                                      float                                        value,
283                                      bool                                         is_input,
284                                      bool                                         use_knob = false);
285
286         void ui_parameter_changed (ControlUI* cui);
287         void update_control_display (ControlUI* cui);
288         void update_input_displays (); // workaround for preset load
289         void control_combo_changed (ControlUI* cui, float value);
290
291         bool astate_button_event (GdkEventButton* ev, ControlUI*);
292         void automation_state_changed (ControlUI*);
293         void set_automation_state (ARDOUR::AutoState state, ControlUI* cui);
294         void set_all_automation (ARDOUR::AutoState state);
295
296         void knob_size_request(Gtk::Requisition* req, ControlUI* cui);
297
298         /* XXX: remove */
299         void print_parameter (char *buf, uint32_t len, uint32_t param);
300         bool integer_printer (char* buf, Gtk::Adjustment &, ControlUI *);
301         bool midinote_printer(char* buf, Gtk::Adjustment &, ControlUI *);
302
303         typedef std::map<uint32_t, Gtk::FileChooserButton*> FilePathControls;
304         FilePathControls _filepath_controls;
305         void set_path_property (const ARDOUR::ParameterDescriptor& desc,
306                                 Gtk::FileChooserButton*            widget);
307         void path_property_changed (uint32_t key, const ARDOUR::Variant& value);
308
309 };
310
311 class PluginUIWindow : public ArdourWindow
312 {
313   public:
314         PluginUIWindow (boost::shared_ptr<ARDOUR::PluginInsert> insert,
315                         bool scrollable=false,
316                         bool editor=true);
317         ~PluginUIWindow ();
318
319         PlugUIBase& pluginui() { return *_pluginui; }
320
321         void resize_preferred();
322         void set_parent (Gtk::Window*);
323         void set_title(const std::string& title);
324
325
326         bool on_key_press_event (GdkEventKey*);
327         bool on_key_release_event (GdkEventKey*);
328         void on_show ();
329         void on_hide ();
330
331   private:
332         std::string _title;
333         PlugUIBase* _pluginui;
334         PBD::ScopedConnection death_connection;
335         Gtk::Window* parent;
336         Gtk::VBox vbox;
337         bool was_visible;
338         bool _keyboard_focused;
339 #ifdef AUDIOUNIT_SUPPORT
340         int pre_deactivate_x;
341         int pre_deactivate_y;
342 #endif
343
344         void keyboard_focused (bool yn);
345
346         void app_activated (bool);
347         void plugin_going_away ();
348
349         bool create_windows_vst_editor (boost::shared_ptr<ARDOUR::PluginInsert>);
350         bool create_lxvst_editor(boost::shared_ptr<ARDOUR::PluginInsert>);
351         bool create_mac_vst_editor(boost::shared_ptr<ARDOUR::PluginInsert>);
352         bool create_audiounit_editor (boost::shared_ptr<ARDOUR::PluginInsert>);
353         bool create_lv2_editor (boost::shared_ptr<ARDOUR::PluginInsert>);
354 };
355
356 #ifdef MACVST_SUPPORT
357 /* this function has to be in a .mm file
358  * because MacVSTPluginUI has Cocoa members
359  */
360 extern VSTPluginUI* create_mac_vst_gui (boost::shared_ptr<ARDOUR::PluginInsert>);
361 #endif
362
363 #ifdef AUDIOUNIT_SUPPORT
364 /* this function has to be in a .mm file */
365 extern PlugUIBase* create_au_gui (boost::shared_ptr<ARDOUR::PluginInsert>, Gtk::VBox**);
366 #endif
367
368 #endif /* __ardour_plugin_ui_h__ */