/* This file is part of Ingen. Copyright 2007-2015 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 "PortMenu.hpp" #include "App.hpp" #include "ObjectMenu.hpp" #include "ingen/Atom.hpp" #include "ingen/Forge.hpp" #include "ingen/Interface.hpp" #include "ingen/Properties.hpp" #include "ingen/Resource.hpp" #include "ingen/URIs.hpp" #include "ingen/client/BlockModel.hpp" #include "ingen/client/GraphModel.hpp" // IWYU pragma: keep #include "ingen/client/ObjectModel.hpp" #include "ingen/client/PortModel.hpp" #include "ingen/paths.hpp" #include "raul/Path.hpp" #include "raul/Symbol.hpp" #include #include #include #include #include #include #include #include #include namespace ingen { using client::BlockModel; using client::GraphModel; using client::PortModel; namespace gui { PortMenu::PortMenu(BaseObjectType* cobject, const Glib::RefPtr& xml) : ObjectMenu(cobject, xml) , _internal_graph_port(false) { xml->get_widget("object_menu", _port_menu); xml->get_widget("port_set_min_menuitem", _set_min_menuitem); xml->get_widget("port_set_max_menuitem", _set_max_menuitem); xml->get_widget("port_reset_range_menuitem", _reset_range_menuitem); xml->get_widget("port_expose_menuitem", _expose_menuitem); } void PortMenu::init(App& app, const std::shared_ptr& port, bool internal_graph_port) { const URIs& uris = app.uris(); ObjectMenu::init(app, port); _internal_graph_port = internal_graph_port; _set_min_menuitem->signal_activate().connect( sigc::mem_fun(this, &PortMenu::on_menu_set_min)); _set_max_menuitem->signal_activate().connect( sigc::mem_fun(this, &PortMenu::on_menu_set_max)); _reset_range_menuitem->signal_activate().connect( sigc::mem_fun(this, &PortMenu::on_menu_reset_range)); _expose_menuitem->signal_activate().connect( sigc::mem_fun(this, &PortMenu::on_menu_expose)); const bool is_control(app.can_control(port.get()) && port->is_numeric()); const bool is_on_graph(std::dynamic_pointer_cast(port->parent())); const bool is_input(port->is_input()); if (!is_on_graph) { _polyphonic_menuitem->set_sensitive(false); _rename_menuitem->set_sensitive(false); _destroy_menuitem->set_sensitive(false); } if (port->is_a(uris.atom_AtomPort)) { _polyphonic_menuitem->hide(); } _reset_range_menuitem->set_visible(is_input && is_control && !is_on_graph); _set_max_menuitem->set_visible(is_input && is_control); _set_min_menuitem->set_visible(is_input && is_control); _expose_menuitem->set_visible(!is_on_graph); _learn_menuitem->set_visible(is_input && is_control); _unlearn_menuitem->set_visible(is_input && is_control); if (!is_control && is_on_graph) { _separator_menuitem->hide(); } _enable_signal = true; } void PortMenu::on_menu_disconnect() { if (_internal_graph_port) { _app->interface()->disconnect_all( _object->parent()->path(), _object->path()); } else { _app->interface()->disconnect_all( _object->parent()->path().parent(), _object->path()); } } void PortMenu::on_menu_set_min() { const URIs& uris = _app->uris(); auto model = std::dynamic_pointer_cast(_object); const Atom& value = model->get_property(uris.ingen_value); if (value.is_valid()) { _app->set_property(_object->uri(), uris.lv2_minimum, value); } } void PortMenu::on_menu_set_max() { const URIs& uris = _app->uris(); auto model = std::dynamic_pointer_cast(_object); const Atom& value = model->get_property(uris.ingen_value); if (value.is_valid()) { _app->set_property(_object->uri(), uris.lv2_maximum, value); } } void PortMenu::on_menu_reset_range() { const URIs& uris = _app->uris(); auto model = std::dynamic_pointer_cast(_object); // Remove lv2:minimum and lv2:maximum properties Properties remove; remove.insert({uris.lv2_minimum, Property(uris.patch_wildcard)}); remove.insert({uris.lv2_maximum, Property(uris.patch_wildcard)}); _app->interface()->delta(_object->uri(), remove, Properties()); } void PortMenu::on_menu_expose() { const URIs& uris = _app->uris(); auto port = std::dynamic_pointer_cast(_object); auto block = std::dynamic_pointer_cast(port->parent()); const std::string label = block->label() + " " + block->port_label(port); const Raul::Path path = Raul::Path(block->path() + Raul::Symbol("_" + port->symbol())); ingen::Resource r(*_object); r.remove_property(uris.lv2_index, uris.patch_wildcard); r.set_property(uris.lv2_symbol, _app->forge().alloc(path.symbol())); r.set_property(uris.lv2_name, _app->forge().alloc(label.c_str())); // TODO: Pretty kludgey coordinates const float block_x = block->get_property(uris.ingen_canvasX).get(); const float block_y = block->get_property(uris.ingen_canvasY).get(); const float x_off = (label.length() * 16.0f) * (port->is_input() ? -1 : 1); const float y_off = port->index() * 32.0f; r.set_property(uris.ingen_canvasX, _app->forge().make(block_x + x_off)); r.set_property(uris.ingen_canvasY, _app->forge().make(block_y + y_off)); _app->interface()->put(path_to_uri(path), r.properties()); if (port->is_input()) { _app->interface()->connect(path, _object->path()); } else { _app->interface()->connect(_object->path(), path); } } } // namespace gui } // namespace ingen