/*
This file is part of Ingen.
Copyright 2007-2016 David Robillard
Ingen is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or any later version.
Ingen 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 Affero General Public License for details.
You should have received a copy of the GNU Affero General Public License
along with Ingen. If not, see .
*/
#include "NodeModule.hpp"
#include "App.hpp"
#include "GraphCanvas.hpp"
#include "GraphWindow.hpp"
#include "NodeMenu.hpp"
#include "Port.hpp"
#include "RenameWindow.hpp"
#include "Style.hpp"
#include "SubgraphModule.hpp"
#include "WidgetFactory.hpp"
#include "WindowFactory.hpp"
#include "ingen_config.h"
#include "ingen/Atom.hpp"
#include "ingen/Configuration.hpp"
#include "ingen/Interface.hpp"
#include "ingen/Log.hpp"
#include "ingen/client/BlockModel.hpp"
#include "ingen/client/GraphModel.hpp"
#include "ingen/client/PluginModel.hpp"
#include "ingen/client/PluginUI.hpp"
#include "lv2/atom/util.h"
#include
#include
#include
#include
namespace ingen {
using client::BlockModel;
using client::GraphModel;
using client::PluginModel;
using client::PortModel;
namespace gui {
NodeModule::NodeModule(GraphCanvas& canvas,
const std::shared_ptr& block)
: Ganv::Module(canvas, block->path().symbol(), 0, 0, true)
, _block(block)
, _gui_widget(nullptr)
, _gui_window(nullptr)
, _initialised(false)
{
block->signal_new_port().connect(
sigc::mem_fun(this, &NodeModule::new_port_view));
block->signal_removed_port().connect(
sigc::hide_return(sigc::mem_fun(this, &NodeModule::delete_port_view)));
block->signal_property().connect(
sigc::mem_fun(this, &NodeModule::property_changed));
block->signal_moved().connect(
sigc::mem_fun(this, &NodeModule::rename));
signal_event().connect(
sigc::mem_fun(this, &NodeModule::on_event));
signal_moved().connect(
sigc::mem_fun(this, &NodeModule::store_location));
signal_selected().connect(
sigc::mem_fun(this, &NodeModule::on_selected));
const auto* plugin = dynamic_cast(block->plugin());
if (plugin) {
plugin->signal_changed().connect(
sigc::mem_fun(this, &NodeModule::plugin_changed));
}
for (const auto& p : block->properties()) {
property_changed(p.first, p.second);
}
if (_block->has_property(app().uris().ingen_uiEmbedded,
app().uris().forge.make(true))) {
// Schedule idle callback to embed GUI once ports arrive
Glib::signal_timeout().connect(
sigc::mem_fun(*this, &NodeModule::idle_init), 25, G_PRIORITY_DEFAULT_IDLE);
} else {
_initialised = true;
}
}
NodeModule::~NodeModule()
{
delete _gui_widget;
delete _gui_window;
}
bool
NodeModule::idle_init()
{
if (_block->ports().empty()) {
return true; // Need to embed GUI, but ports haven't shown up yet
}
// Ports have arrived, embed GUI and deregister this callback
embed_gui(true);
_initialised = true;
return false;
}
bool
NodeModule::show_menu(GdkEventButton* ev)
{
WidgetFactory::get_widget_derived("object_menu", _menu);
if (!_menu) {
app().log().error("Failed to load object menu widget\n");
return false;
}
_menu->init(app(), _block);
_menu->signal_embed_gui.connect(
sigc::mem_fun(this, &NodeModule::on_embed_gui_toggled));
_menu->signal_popup_gui.connect(
sigc::hide_return(sigc::mem_fun(this, &NodeModule::popup_gui)));
_menu->popup(ev->button, ev->time);
return true;
}
NodeModule*
NodeModule::create(GraphCanvas& canvas,
const std::shared_ptr& block,
bool human)
{
auto graph = std::dynamic_pointer_cast(block);
NodeModule* ret = (graph)
? new SubgraphModule(canvas, graph)
: new NodeModule(canvas, block);
for (const auto& p : block->properties()) {
ret->property_changed(p.first, p.second);
}
for (const auto& p : block->ports()) {
ret->new_port_view(p);
}
ret->set_stacked(block->polyphonic());
if (human) {
ret->show_human_names(human); // FIXME: double port iteration
}
return ret;
}
App&
NodeModule::app() const
{
return static_cast(canvas())->app();
}
void
NodeModule::show_human_names(bool b)
{
const URIs& uris = app().uris();
if (b) {
set_label(block()->label().c_str());
} else {
set_label(block()->symbol().c_str());
}
for (auto* p : *this) {
auto* const port = dynamic_cast(p);
Glib::ustring label(port->model()->symbol().c_str());
if (b) {
const Atom& name_property = port->model()->get_property(uris.lv2_name);
if (name_property.type() == uris.forge.String) {
label = name_property.ptr();
} else {
Glib::ustring hn = block()->plugin_model()->port_human_name(
port->model()->index());
if (!hn.empty()) {
label = hn;
}
}
}
port->set_label(label.c_str());
}
}
void
NodeModule::port_activity(uint32_t index, const Atom& value)
{
const URIs& uris = app().uris();
if (!_plugin_ui) {
return;
}
if (_block->get_port(index)->is_a(uris.atom_AtomPort)) {
_plugin_ui->port_event(index,
lv2_atom_total_size(value.atom()),
uris.atom_eventTransfer,
value.atom());
}
}
void
NodeModule::port_value_changed(uint32_t index, const Atom& value)
{
const URIs& uris = app().uris();
if (!_plugin_ui) {
return;
}
if (value.type() == uris.atom_Float &&
_block->get_port(index)->is_numeric()) {
_plugin_ui->port_event(index, sizeof(float), 0, value.ptr());
} else {
_plugin_ui->port_event(index,
lv2_atom_total_size(value.atom()),
uris.atom_eventTransfer,
value.atom());
}
}
void
NodeModule::plugin_changed()
{
for (auto* p : *this) {
dynamic_cast(p)->update_metadata();
}
}
void
NodeModule::on_embed_gui_toggled(bool embed)
{
embed_gui(embed);
app().set_property(_block->uri(),
app().uris().ingen_uiEmbedded,
app().forge().make(embed));
}
void
NodeModule::embed_gui(bool embed)
{
if (embed) {
if (_gui_window) {
app().log().warn("LV2 GUI already popped up, cannot embed\n");
return;
}
if (!_plugin_ui) {
_plugin_ui = _block->plugin_model()->ui(app().world(), _block);
}
if (_plugin_ui) {
_plugin_ui->signal_property_changed().connect(
sigc::mem_fun(app(), &App::set_property));
if (!_plugin_ui->instantiate()) {
app().log().error("Failed to instantiate LV2 UI\n");
} else {
auto* c_widget =
static_cast(_plugin_ui->get_widget());
_gui_widget = Glib::wrap(c_widget);
Gtk::Container* container = new Gtk::EventBox();
container->set_name("IngenEmbeddedUI");
container->set_border_width(4.0);
container->add(*_gui_widget);
Ganv::Module::embed(container);
}
} else {
app().log().error("Failed to create LV2 UI\n");
}
if (_gui_widget) {
_gui_widget->show_all();
set_control_values();
}
} else { // un-embed
Ganv::Module::embed(nullptr);
_plugin_ui.reset();
}
}
void
NodeModule::rename()
{
if (app().world().conf().option("port-labels").get() &&
!app().world().conf().option("human-names").get()) {
set_label(_block->path().symbol());
}
}
void
NodeModule::new_port_view(const std::shared_ptr& port)
{
Port::create(app(), *this, port);
port->signal_value_changed().connect(
sigc::bind<0>(sigc::mem_fun(this, &NodeModule::port_value_changed),
port->index()));
port->signal_activity().connect(
sigc::bind<0>(sigc::mem_fun(this, &NodeModule::port_activity),
port->index()));
}
Port*
NodeModule::port(const std::shared_ptr& model)
{
for (auto* p : *this) {
auto* const port = dynamic_cast(p);
if (port->model() == model) {
return port;
}
}
return nullptr;
}
void
NodeModule::delete_port_view(const std::shared_ptr& model)
{
Port* const p = port(model);
if (!p) {
app().log().warn("Failed to find port %1% on module %2%\n",
model->path(), _block->path());
}
delete p;
}
bool
NodeModule::popup_gui()
{
if (_block->plugin() && app().uris().lv2_Plugin == _block->plugin_model()->type()) {
if (_plugin_ui) {
_gui_window->present();
return true;
}
const auto* const plugin =
dynamic_cast(_block->plugin());
assert(plugin);
_plugin_ui = plugin->ui(app().world(), _block);
if (_plugin_ui) {
_plugin_ui->signal_property_changed().connect(
sigc::mem_fun(app(), &App::set_property));
if (!_plugin_ui->instantiated() && !_plugin_ui->instantiate()) {
app().log().error("Failed to instantiate LV2 UI\n");
return false;
}
auto* c_widget = static_cast(_plugin_ui->get_widget());
_gui_widget = Glib::wrap(c_widget);
_gui_window = new Gtk::Window();
if (!_plugin_ui->is_resizable()) {
_gui_window->set_resizable(false);
}
_gui_window->set_title(_block->path() + " UI - Ingen");
_gui_window->set_role("plugin_ui");
_gui_window->add(*_gui_widget);
_gui_widget->show_all();
set_control_values();
_gui_window->signal_unmap().connect(
sigc::mem_fun(this, &NodeModule::on_gui_window_close));
_gui_window->present();
return true;
} else {
app().log().warn("No LV2 GUI for %1%\n", _block->path());
}
}
return false;
}
void
NodeModule::on_gui_window_close()
{
delete _gui_window;
_gui_window = nullptr;
_plugin_ui.reset();
_gui_widget = nullptr;
}
void
NodeModule::set_control_values()
{
uint32_t index = 0;
for (const auto& p : _block->ports()) {
if (app().can_control(p.get()) && p->value().is_valid()) {
port_value_changed(index, p->value());
}
++index;
}
}
bool
NodeModule::on_double_click(GdkEventButton* event)
{
popup_gui();
return true;
}
bool
NodeModule::on_event(GdkEvent* ev)
{
if (ev->type == GDK_BUTTON_PRESS && ev->button.button == 3) {
return show_menu(&ev->button);
} else if (ev->type == GDK_2BUTTON_PRESS) {
return on_double_click(&ev->button);
} else if (ev->type == GDK_ENTER_NOTIFY) {
GraphBox* const box = app().window_factory()->graph_box(
std::dynamic_pointer_cast(_block->parent()));
if (box) {
box->object_entered(_block.get());
}
} else if (ev->type == GDK_LEAVE_NOTIFY) {
GraphBox* const box = app().window_factory()->graph_box(
std::dynamic_pointer_cast(_block->parent()));
if (box) {
box->object_left(_block.get());
}
}
return false;
}
void
NodeModule::store_location(double ax, double ay)
{
const URIs& uris = app().uris();
const Atom x(app().forge().make(static_cast(ax)));
const Atom y(app().forge().make(static_cast(ay)));
if (x != _block->get_property(uris.ingen_canvasX) ||
y != _block->get_property(uris.ingen_canvasY))
{
app().interface()->put(_block->uri(), {{uris.ingen_canvasX, x},
{uris.ingen_canvasY, y}});
}
}
void
NodeModule::property_changed(const URI& key, const Atom& value)
{
const URIs& uris = app().uris();
if (value.type() == uris.forge.Float) {
if (key == uris.ingen_canvasX) {
move_to(static_cast(value.get()), get_y());
} else if (key == uris.ingen_canvasY) {
move_to(get_x(), static_cast(value.get()));
}
} else if (value.type() == uris.forge.Bool) {
if (key == uris.ingen_polyphonic) {
set_stacked(value.get());
} else if (key == uris.ingen_uiEmbedded && _initialised) {
if (value.get() && !_gui_widget) {
embed_gui(true);
} else if (!value.get() && _gui_widget) {
embed_gui(false);
}
} else if (key == uris.ingen_enabled) {
if (value.get()) {
set_dash_length(0.0);
} else {
set_dash_length(5.0);
}
}
} else if (value.type() == uris.forge.String) {
if (key == uris.lv2_name
&& app().world().conf().option("human-names").get()) {
set_label(value.ptr());
}
}
}
bool
NodeModule::on_selected(gboolean selected)
{
GraphWindow* win = app().window_factory()->parent_graph_window(block());
if (!win) {
return true;
}
if (selected && win->documentation_is_visible()) {
std::string doc;
bool html = false;
#ifdef HAVE_WEBKIT
html = true;
#endif
if (block()->plugin_model()) {
doc = block()->plugin_model()->documentation(html);
}
win->set_documentation(doc, html);
}
return true;
}
} // namespace gui
} // namespace ingen