/* Copyright (C) 2004 Paul Davis This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include #include #include #include #include "plugin_ui.h" #include using namespace Gtk; using namespace ARDOUR; using namespace PBD; VSTPluginUI::VSTPluginUI (boost::shared_ptr pi, boost::shared_ptr vp) : PlugUIBase (pi), vst (vp) { create_preset_store (); fst_run_editor (vst->fst()); preset_box.set_spacing (6); preset_box.set_border_width (6); preset_box.pack_end (bypass_button, false, false, 10); preset_box.pack_end (save_button, false, false); preset_box.pack_end (vst_preset_combo, false, false); vst_preset_combo.signal_changed().connect (mem_fun (*this, &VSTPluginUI::preset_chosen)); bypass_button.set_active (!insert->active()); pack_start (preset_box, false, false); pack_start (socket, true, true); } VSTPluginUI::~VSTPluginUI () { // plugin destructor destroys the custom GUI, via Windows fun-and-games, // and then our PluginUIWindow does the rest } void VSTPluginUI::preset_chosen () { // we can't dispatch directly here, too many plugins only expects one GUI thread. vst->fst()->want_program = vst_preset_combo.get_active_row_number (); socket.grab_focus (); } int VSTPluginUI::get_preferred_height () { return vst->fst()->height; } int VSTPluginUI::get_preferred_width () { return vst->fst()->width; } int VSTPluginUI::package (Gtk::Window& win) { /* forward configure events to plugin window */ win.signal_configure_event().connect (bind (mem_fun (*this, &VSTPluginUI::configure_handler), &socket), false); /* this assumes that the window's owner understands the XEmbed protocol. */ socket.add_id (fst_get_XID (vst->fst())); fst_move_window_into_view (vst->fst()); return 0; } bool VSTPluginUI::configure_handler (GdkEventConfigure* ev, Gtk::Socket *socket) { XEvent event; gint x, y; GdkWindow* w; if (socket == 0 || ((w = socket->gobj()->plug_window) == 0)) { return false; } event.xconfigure.type = ConfigureNotify; event.xconfigure.event = GDK_WINDOW_XWINDOW (w); event.xconfigure.window = GDK_WINDOW_XWINDOW (w); /* The ICCCM says that synthetic events should have root relative * coordinates. We still aren't really ICCCM compliant, since * we don't send events when the real toplevel is moved. */ gdk_error_trap_push (); gdk_window_get_origin (w, &x, &y); gdk_error_trap_pop (); event.xconfigure.x = x; event.xconfigure.y = y; event.xconfigure.width = GTK_WIDGET(socket->gobj())->allocation.width; event.xconfigure.height = GTK_WIDGET(socket->gobj())->allocation.height; event.xconfigure.border_width = 0; event.xconfigure.above = None; event.xconfigure.override_redirect = False; gdk_error_trap_push (); XSendEvent (GDK_WINDOW_XDISPLAY (w), GDK_WINDOW_XWINDOW (w), False, StructureNotifyMask, &event); gdk_error_trap_pop (); return false; } void VSTPluginUI::create_preset_store () { FST *fst = vst->fst(); int vst_version = fst->plugin->dispatcher (fst->plugin, effGetVstVersion, 0, 0, NULL, 0.0f); preset_model = ListStore::create (preset_columns); for (int i = 0; i < fst->plugin->numPrograms; ++i) { char buf[100]; TreeModel::Row row = *(preset_model->append()); snprintf (buf, 90, "preset %d", i); if (vst_version >= 2) { fst->plugin->dispatcher (fst->plugin, 29, i, 0, buf, 0.0); } row[preset_columns.name] = buf; row[preset_columns.number] = i; } if (fst->plugin->numPrograms > 0) { fst->plugin->dispatcher( fst->plugin, effSetProgram, 0, 0, NULL, 0.0 ); } vst_preset_combo.set_model (preset_model); CellRenderer* renderer = manage (new CellRendererText()); vst_preset_combo.pack_start (*renderer, true); vst_preset_combo.add_attribute (*renderer, "text", 0); if (vst->fst()->current_program != -1) { vst_preset_combo.set_active (vst->fst()->current_program); } else { vst_preset_combo.set_active (0); } } typedef int (*error_handler_t)( Display *, XErrorEvent *); static Display *the_gtk_display; static error_handler_t wine_error_handler; static error_handler_t gtk_error_handler; static int fst_xerror_handler( Display *disp, XErrorEvent *ev ) { if (disp == the_gtk_display) { printf ("relaying error to gtk\n"); return gtk_error_handler (disp, ev); } else { printf( "relaying error to wine\n" ); return wine_error_handler (disp, ev); } } void gui_init (int *argc, char **argv[]) { wine_error_handler = XSetErrorHandler (NULL); gtk_init (argc, argv); the_gtk_display = gdk_x11_display_get_xdisplay (gdk_display_get_default()); gtk_error_handler = XSetErrorHandler( fst_xerror_handler ); }