Plugin analysis numerics
[ardour.git] / gtk2_ardour / vst_plugin_ui.cc
1 /*
2     Copyright (C) 2000-2010 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 #include <gtkmm.h>
21 #include "ardour/vst_plugin.h"
22 #include "ardour/vst_types.h"
23 #include "ardour/plugin_insert.h"
24 #include "vst_plugin_ui.h"
25
26 #ifdef PLATFORM_WINDOWS
27 #include <gdk/gdkwin32.h>
28 #elif defined __APPLE__
29 // TODO
30 #else
31 #include <gdk/gdkx.h>
32 #endif
33
34 VSTPluginUI::VSTPluginUI (boost::shared_ptr<ARDOUR::PluginInsert> insert, boost::shared_ptr<ARDOUR::VSTPlugin> plugin)
35         : PlugUIBase (insert)
36         , _vst (plugin)
37 {
38         Gtk::HBox* box = manage (new Gtk::HBox);
39         box->set_spacing (6);
40         box->set_border_width (6);
41         box->pack_end (focus_button, false, false);
42         box->pack_end (bypass_button, false, false, 4);
43         if (insert->controls().size() > 0) {
44                 box->pack_end (reset_button, false, false, 4);
45         }
46         box->pack_end (delete_button, false, false);
47         box->pack_end (save_button, false, false);
48         box->pack_end (add_button, false, false);
49         box->pack_end (_preset_combo, false, false);
50         box->pack_end (_preset_modified, false, false);
51         box->pack_end (pin_management_button, false, false);
52
53         bypass_button.set_active (!insert->active ());
54
55         pack_start (*box, false, false);
56         box->signal_size_allocate().connect (sigc::mem_fun (*this, &VSTPluginUI::top_box_allocated));
57 #ifdef GDK_WINDOWING_X11
58         pack_start (_socket, true, true);
59         _socket.set_border_width (0);
60 #endif
61 }
62
63 VSTPluginUI::~VSTPluginUI ()
64 {
65
66 }
67
68 void
69 VSTPluginUI::preset_selected (ARDOUR::Plugin::PresetRecord preset)
70 {
71 #ifdef GDK_WINDOWING_X11
72         _socket.grab_focus ();
73 #endif
74         PlugUIBase::preset_selected (preset);
75 }
76
77 int
78 VSTPluginUI::get_preferred_height ()
79 {
80         return _vst->state()->height + _vst->state()->voffset;
81 }
82
83 int
84 VSTPluginUI::get_preferred_width ()
85 {
86         return _vst->state()->width + _vst->state()->hoffset;
87 }
88
89 int
90 VSTPluginUI::package (Gtk::Window& win)
91 {
92 #ifdef GDK_WINDOWING_X11
93         /* Forward configure events to plugin window */
94         win.signal_configure_event().connect (sigc::mem_fun (*this, &VSTPluginUI::configure_handler), false);
95
96         /* This assumes that the window's owner understands the XEmbed protocol */
97         _socket.add_id (get_XID ());
98         _socket.set_size_request(
99                         _vst->state()->width + _vst->state()->hoffset,
100                         _vst->state()->height + _vst->state()->voffset);
101 #endif
102
103         return 0;
104 }
105
106 bool
107 VSTPluginUI::on_window_show(const std::string& title)
108 {
109         _vst->state()->gui_shown = 1;
110         return PlugUIBase::on_window_show(title);
111 }
112
113 void
114 VSTPluginUI::on_window_hide()
115 {
116         _vst->state()->gui_shown = 0;
117         PlugUIBase::on_window_hide();
118 }
119
120
121 bool
122 VSTPluginUI::configure_handler (GdkEventConfigure*)
123 {
124 #ifdef GDK_WINDOWING_X11
125         XEvent event;
126         gint x, y;
127         GdkWindow* w;
128         Window xw = _vst->state()->linux_plugin_ui_window;
129
130         if ((w = _socket.gobj()->plug_window) == 0) {
131                 return false;
132         }
133
134         event.xconfigure.type = ConfigureNotify;
135         event.xconfigure.event = GDK_WINDOW_XWINDOW (w);
136         event.xconfigure.window = GDK_WINDOW_XWINDOW (w);
137
138         /* The ICCCM says that synthetic events should have root relative
139          * coordinates. We still aren't really ICCCM compliant, since
140          * we don't send events when the real toplevel is moved.
141          */
142         gdk_error_trap_push ();
143         gdk_window_get_origin (w, &x, &y);
144         gdk_error_trap_pop ();
145
146         event.xconfigure.x = x;
147         event.xconfigure.y = y;
148         event.xconfigure.width = GTK_WIDGET (_socket.gobj())->allocation.width;
149         event.xconfigure.height = GTK_WIDGET (_socket.gobj())->allocation.height;
150
151         event.xconfigure.border_width = 0;
152         event.xconfigure.above = None;
153         event.xconfigure.override_redirect = False;
154
155         gdk_error_trap_push ();
156         XSendEvent (GDK_WINDOW_XDISPLAY (w), GDK_WINDOW_XWINDOW (w), False, StructureNotifyMask, &event);
157         /* if the plugin does adds itself to the parent,
158          * but ardour re-parents it, we have a pointer to
159          * the socket's child and need to resize the
160          * child window (e.g. JUCE, u-he)
161          */
162         if (xw) {
163                 XMoveResizeWindow (GDK_WINDOW_XDISPLAY (w), xw,
164                                 0, 0, _vst->state()->width, _vst->state()->height);
165                 XMapRaised (GDK_WINDOW_XDISPLAY (w), xw);
166                 XFlush (GDK_WINDOW_XDISPLAY (w));
167         }
168         gdk_error_trap_pop ();
169 #endif
170
171         return false;
172 }