ardour/gtk2_ardour/mac_vst_plugin_ui.mm

254 lines
6.9 KiB
Plaintext

/*
* Copyright (C) 2016-2017 Robin Gareus <robin@gareus.org>
*
* 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.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include <gtkmm.h>
#include <gtk/gtk.h>
#include <gdk/gdkquartz.h>
#include "gui_thread.h"
#include "ardour/plugin_insert.h"
#include "ardour/mac_vst_plugin.h"
#include "ardour/vst_types.h"
#include "mac_vst_plugin_ui.h"
#include "pbd/i18n.h"
using namespace Gtk;
using namespace ARDOUR;
using namespace PBD;
struct ERect{
short top;
short left;
short bottom;
short right;
};
@implementation ResizeNotificationObject
- (ResizeNotificationObject*) initWithPluginUI: (MacVSTPluginUI*) vstui
{
self = [ super init ];
plugin_ui = vstui;
return self;
}
- (void)viewResized:(NSNotification *)notification
{
(void) notification; // unused
plugin_ui->view_resized();
}
@end
VSTPluginUI*
create_mac_vst_gui (std::shared_ptr<PlugInsertBase> pib)
{
/* PluginUIWindow::create_mac_vst_editor assures this cast works */
std::shared_ptr<MacVSTPlugin> mvst = std::dynamic_pointer_cast<MacVSTPlugin> (pib->plugin());
return new MacVSTPluginUI (pib, mvst);
}
MacVSTPluginUI::MacVSTPluginUI (std::shared_ptr<PlugInsertBase> pib, std::shared_ptr<VSTPlugin> vst)
: VSTPluginUI (pib, vst)
, _ns_view (0)
{
low_box.add_events (Gdk::VISIBILITY_NOTIFY_MASK | Gdk::EXPOSURE_MASK);
low_box.signal_realize().connect (mem_fun (this, &MacVSTPluginUI::lower_box_realized));
low_box.signal_visibility_notify_event ().connect (mem_fun (this, &MacVSTPluginUI::lower_box_visibility_notify));
low_box.signal_size_request ().connect (mem_fun (this, &MacVSTPluginUI::lower_box_size_request));
low_box.signal_size_allocate ().connect (mem_fun (this, &MacVSTPluginUI::lower_box_size_allocate));
low_box.signal_map ().connect (mem_fun (this, &MacVSTPluginUI::lower_box_map));
low_box.signal_unmap ().connect (mem_fun (this, &MacVSTPluginUI::lower_box_unmap));
pack_start (low_box, true, true);
low_box.show ();
vst->LoadPresetProgram.connect (_program_connection, invalidator (*this), boost::bind (&MacVSTPluginUI::set_program, this), gui_context());
_ns_view = [[NSView new] retain];
AEffect* plugin = _vst->state()->plugin;
plugin->dispatcher (plugin, effEditOpen, 0, 0, _ns_view, 0.0f);
_idle_connection = Glib::signal_idle().connect (sigc::mem_fun (*this, &MacVSTPluginUI::idle));
_resize_notifier = [[ResizeNotificationObject alloc] initWithPluginUI:this];
[[NSNotificationCenter defaultCenter] addObserver:_resize_notifier
selector:@selector(viewResized:) name:NSViewFrameDidChangeNotification
object:_ns_view];
NSArray* subviews = [_ns_view subviews];
for (unsigned long i = 0; i < [subviews count]; ++i) {
NSView* subview = [subviews objectAtIndex:i];
[[NSNotificationCenter defaultCenter] addObserver:_resize_notifier
selector:@selector(viewResized:) name:NSViewFrameDidChangeNotification
object:subview];
break; /* only watch first subview (if any) */
}
}
MacVSTPluginUI::~MacVSTPluginUI ()
{
[[NSNotificationCenter defaultCenter] removeObserver:_resize_notifier];
[_resize_notifier release];
[_ns_view removeFromSuperview];
[_ns_view release];
AEffect* plugin = _vst->state()->plugin;
plugin->dispatcher (plugin, effEditClose, 0, 0, 0, 0.0f);
_idle_connection.disconnect();
}
NSWindow*
MacVSTPluginUI::get_nswindow ()
{
Gtk::Container* toplevel = get_toplevel();
if (!toplevel || !toplevel->is_toplevel()) {
error << _("MacVSTPluginUI: no top level window!") << endmsg;
return 0;
}
NSWindow* true_parent = gdk_quartz_window_get_nswindow (toplevel->get_window()->gobj());
if (!true_parent) {
error << _("MacVSTPluginUI: no top level window!") << endmsg;
return 0;
}
return true_parent;
}
int
MacVSTPluginUI::package (Gtk::Window& win)
{
VSTPluginUI::package (win);
return 0;
}
void
MacVSTPluginUI::forward_key_event (GdkEventKey* ev)
{
NSEvent* nsevent = gdk_quartz_event_get_nsevent ((GdkEvent*)ev);
if (_ns_view && nsevent) {
/* filter on nsevent type here because GDK massages FlagsChanged
* messages into GDK_KEY_{PRESS,RELEASE} but Cocoa won't
* handle a FlagsChanged message as a keyDown or keyUp
*/
if ([nsevent type] == NSKeyDown) {
[[[_ns_view window] firstResponder] keyDown:nsevent];
} else if ([nsevent type] == NSKeyUp) {
[[[_ns_view window] firstResponder] keyUp:nsevent];
} else if ([nsevent type] == NSFlagsChanged) {
[[[_ns_view window] firstResponder] flagsChanged:nsevent];
}
}
}
void
MacVSTPluginUI::lower_box_realized ()
{
NSWindow* win = get_nswindow ();
if (!win) {
return;
}
[win setAutodisplay:1]; // turn off GTK stuff for this window
NSView* view = gdk_quartz_window_get_nsview (low_box.get_window()->gobj());
[view addSubview:_ns_view];
low_box.queue_resize ();
}
bool
MacVSTPluginUI::lower_box_visibility_notify (GdkEventVisibility* ev)
{
return false;
}
void
MacVSTPluginUI::lower_box_map ()
{
[_ns_view setHidden:0];
}
void
MacVSTPluginUI::lower_box_unmap ()
{
[_ns_view setHidden:1];
}
void
MacVSTPluginUI::lower_box_size_request (GtkRequisition* requisition)
{
struct ERect* er = NULL;
AEffect* plugin = _vst->state()->plugin;
plugin->dispatcher (plugin, effEditGetRect, 0, 0, &er, 0 );
if (er) {
requisition->width = er->right - er->left;
requisition->height = er->bottom - er->top;
} else {
requisition->width = 600;
requisition->height = 400;
}
}
void
MacVSTPluginUI::lower_box_size_allocate (Gtk::Allocation& allocation)
{
gint xx, yy;
gtk_widget_translate_coordinates(
GTK_WIDGET(low_box.gobj()),
GTK_WIDGET(low_box.get_parent()->gobj()),
8, 6, &xx, &yy);
[_ns_view setFrame:NSMakeRect (xx, yy, allocation.get_width (), allocation.get_height ())];
NSArray* subviews = [_ns_view subviews];
for (unsigned long i = 0; i < [subviews count]; ++i) {
NSView* subview = [subviews objectAtIndex:i];
[subview setFrame:NSMakeRect (0, 0, allocation.get_width (), allocation.get_height ())];
break; /* only resize first subview */
}
}
void
MacVSTPluginUI::view_resized ()
{
low_box.queue_resize ();
}
int
MacVSTPluginUI::get_XID ()
{
return _vst->state()->xid; // unused
}
bool
MacVSTPluginUI::idle ()
{
AEffect* plugin = _vst->state()->plugin;
_vst->state()->wantIdle = plugin->dispatcher (plugin, effEditIdle, 0, 0, NULL, 0);
return true; // _vst->state()->wantIdle;
}
void
MacVSTPluginUI::set_program ()
{
vststate_maybe_set_program (_vst->state());
}