summaryrefslogtreecommitdiffstats
path: root/src/gui
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2017-03-09 08:33:21 +0100
committerDavid Robillard <d@drobilla.net>2017-03-09 08:33:21 +0100
commit83d366452af8e93f0722658d730528d699f21e2b (patch)
treedb66b0653d57ca2e2939a1352b3d45aeada4689e /src/gui
parente360392489fe62dbae1f0c28b7f5fb839851f5f6 (diff)
downloadingen-groups.tar.gz
ingen-groups.tar.bz2
ingen-groups.zip
Preliminary port groups workgroups
Diffstat (limited to 'src/gui')
-rw-r--r--src/gui/GraphCanvas.cpp185
-rw-r--r--src/gui/GraphCanvas.hpp10
-rw-r--r--src/gui/GraphPortModule.cpp90
-rw-r--r--src/gui/GraphPortModule.hpp26
-rw-r--r--src/gui/ingen_gui.ui14
5 files changed, 264 insertions, 61 deletions
diff --git a/src/gui/GraphCanvas.cpp b/src/gui/GraphCanvas.cpp
index 6d22f407..1a07f0c6 100644
--- a/src/gui/GraphCanvas.cpp
+++ b/src/gui/GraphCanvas.cpp
@@ -16,6 +16,7 @@
#include <algorithm>
#include <cassert>
+#include <iterator>
#include <map>
#include <set>
#include <string>
@@ -34,6 +35,7 @@
#include "ingen/client/BlockModel.hpp"
#include "ingen/client/ClientStore.hpp"
#include "ingen/client/GraphModel.hpp"
+#include "ingen/client/GroupModel.hpp"
#include "ingen/client/PluginModel.hpp"
#include "ingen/ingen.h"
#include "lv2/lv2plug.in/ns/ext/atom/atom.h"
@@ -93,6 +95,7 @@ GraphCanvas::GraphCanvas(App& app,
Glib::RefPtr<Gtk::Builder> xml = WidgetFactory::create("canvas_menu");
xml->get_widget("canvas_menu", _menu);
+ xml->get_widget("canvas_menu_add_group_input", _add_input_group_menu);
xml->get_widget("canvas_menu_add_audio_input", _menu_add_audio_input);
xml->get_widget("canvas_menu_add_audio_output", _menu_add_audio_output);
xml->get_widget("canvas_menu_add_cv_input", _menu_add_cv_input);
@@ -109,7 +112,22 @@ GraphCanvas::GraphCanvas(App& app,
const URIs& uris = _app.uris();
+ LilvNodes* group_types = lilv_world_find_nodes(
+ app.world()->lilv_world(), NULL, uris.rdfs_subClassOf, uris.pg_DiscreteGroup);
+ LILV_FOREACH(nodes, t, group_types) {
+ const LilvNode* gtype = lilv_nodes_get(group_types, t);
+ _add_input_group_menu->items().push_back(
+ Gtk::Menu_Helpers::MenuElem(lilv_node_as_string(gtype)));
+ Gtk::MenuItem* menu_item = &(_add_input_group_menu->items().back());
+ menu_item->signal_activate().connect(
+ sigc::bind(sigc::mem_fun(this, &GraphCanvas::menu_add_group),
+ "group", "Group", Raul::URI(lilv_node_as_string(gtype)), false));
+ }
+
// Add port menu items
+ // _menu_add_audio_input->signal_activate().connect(
+ // sigc::bind(sigc::mem_fun(this, &GraphCanvas::menu_add_group),
+ // "stereo_in", "Stereo In", uris.pg_StereoGroup, false));
_menu_add_audio_input->signal_activate().connect(
sigc::bind(sigc::mem_fun(this, &GraphCanvas::menu_add_port),
"audio_in", "Audio In", uris.lv2_AudioPort, false));
@@ -147,6 +165,10 @@ GraphCanvas::GraphCanvas(App& app,
sigc::mem_fun(this, &GraphCanvas::add_block));
_graph->signal_removed_block().connect(
sigc::mem_fun(this, &GraphCanvas::remove_block));
+ _graph->signal_new_group().connect(
+ sigc::mem_fun(this, &GraphCanvas::add_group));
+ _graph->signal_removed_group().connect(
+ sigc::mem_fun(this, &GraphCanvas::remove_group));
_graph->signal_new_port().connect(
sigc::mem_fun(this, &GraphCanvas::add_port));
_graph->signal_removed_port().connect(
@@ -357,7 +379,21 @@ GraphCanvas::remove_block(SPtr<const BlockModel> bm)
void
GraphCanvas::add_port(SPtr<const PortModel> pm)
{
- GraphPortModule* view = GraphPortModule::create(*this, pm);
+ const Atom& group = pm->get_property(_app.uris().pg_group);
+ if (group.is_valid()) {
+ // Port is on a group, add to group module
+ const Raul::URI group_uri(_app.forge().str(group, false));
+ auto gm = _app.store()->object(uri_to_path(group_uri));
+ auto g = _views.find(gm);
+ if (g != _views.end()) {
+ Port::create(_app, *g->second, pm, true);
+ return;
+ }
+ }
+
+ // Port has no group, add it to anonymous module
+ GraphPortModule* view = GraphPortModule::create(
+ *this, SPtr<const GroupModel>(), pm);
_views.insert(std::make_pair(pm, view));
view->show();
}
@@ -380,6 +416,20 @@ GraphCanvas::remove_port(SPtr<const PortModel> pm)
assert(_views.find(pm) == _views.end());
}
+void
+GraphCanvas::add_group(SPtr<const GroupModel> gm)
+{
+ GraphPortModule* view = GraphPortModule::create(
+ *this, gm, SPtr<const PortModel>());
+ _views.insert(std::make_pair(gm, view));
+ view->show();
+}
+
+void
+GraphCanvas::remove_group(SPtr<const GroupModel> gm)
+{
+}
+
Ganv::Port*
GraphCanvas::get_port_view(SPtr<PortModel> port)
{
@@ -569,12 +619,13 @@ destroy_node(GanvNode* node, void* data)
if (node_module) {
app->interface()->del(node_module->block()->uri());
} else {
- GraphPortModule* port_module = dynamic_cast<GraphPortModule*>(module);
- if (port_module &&
- strcmp(port_module->port()->path().symbol(), "control") &&
- strcmp(port_module->port()->path().symbol(), "notify")) {
- app->interface()->del(port_module->port()->uri());
- }
+ fprintf(stderr, "FIXME\n");
+ // GraphPortModule* port_module = dynamic_cast<GraphPortModule*>(module);
+ // if (port_module &&
+ // strcmp(port_module->port()->path().symbol(), "control") &&
+ // strcmp(port_module->port()->path().symbol(), "notify")) {
+ // app->interface()->del(port_module->port()->uri());
+ // }
}
}
@@ -614,7 +665,13 @@ serialise_node(GanvNode* node, void* data)
} else {
GraphPortModule* port_module = dynamic_cast<GraphPortModule*>(module);
if (port_module) {
- serialiser->serialise(port_module->port());
+ for (const auto& p : *port_module) {
+ if (Port* port = dynamic_cast<Port*>(p)) {
+ serialiser->serialise(port->model());
+ } else {
+ fprintf(stderr, "ERR\n");
+ }
+ }
}
}
}
@@ -786,33 +843,113 @@ GraphCanvas::generate_port_name(
}
void
-GraphCanvas::menu_add_port(const string& sym_base, const string& name_base,
- const Raul::URI& type, bool is_output)
+GraphCanvas::menu_add_port(const string& sym_base,
+ const string& name_base,
+ const Raul::URI& type,
+ bool is_output)
{
string sym, name;
generate_port_name(sym_base, sym, name_base, name);
- const Raul::Path& path = _graph->path().child(Raul::Symbol(sym));
- const URIs& uris = _app.uris();
+ const Raul::Path& path = _graph->path().child(Raul::Symbol(sym));
+ const URIs& uris = _app.uris();
+ Forge& forge = _app.forge();
+
+ // Basic port description
+ Properties props = {
+ {uris.rdf_type, forge.make_urid(type)},
+ {uris.rdf_type, Property(is_output
+ ? uris.lv2_OutputPort
+ : uris.lv2_InputPort)},
+ {uris.lv2_index, forge.make(int32_t(_graph->num_ports()))},
+ {uris.lv2_name, forge.alloc(name.c_str())}};
- Properties props = get_initial_data(Resource::Graph::INTERNAL);
- props.insert(make_pair(uris.rdf_type, _app.forge().make_urid(type)));
+ // Add atom:bufferType for sequence ports
if (type == uris.atom_AtomPort) {
- props.insert(make_pair(uris.atom_bufferType,
- Property(uris.atom_Sequence)));
+ props.emplace(uris.atom_bufferType,
+ Property(uris.atom_Sequence));
}
- props.insert(make_pair(uris.rdf_type,
- Property(is_output
- ? uris.lv2_OutputPort
- : uris.lv2_InputPort)));
- props.insert(make_pair(uris.lv2_index,
- _app.forge().make(int32_t(_graph->num_ports()))));
- props.insert(make_pair(uris.lv2_name,
- _app.forge().alloc(name.c_str())));
+
+ // Add initial data for canvas position etc
+ Properties init = get_initial_data(Resource::Graph::INTERNAL);
+ props.insert(std::make_move_iterator(init.begin()),
+ std::make_move_iterator(init.end()));
+
_app.interface()->put(path_to_uri(path), props);
}
void
+GraphCanvas::menu_add_group(const string& sym_base,
+ const string& name_base,
+ const Raul::URI& type,
+ bool is_output)
+{
+ string sym, name;
+ generate_port_name(sym_base, sym, name_base, name);
+
+ const Raul::Path& path = _graph->path().child(Raul::Symbol(sym));
+ const URIs& uris = _app.uris();
+ Forge& forge = _app.forge();
+ World* world = _app.world();
+ LilvWorld* lworld = world->lilv_world();
+
+ // Group description
+ Properties group_props = {
+ {uris.rdf_type, forge.make_urid(type)},
+ {uris.rdf_type, Property(is_output
+ ? uris.pg_OutputGroup
+ : uris.pg_InputGroup)},
+ {uris.lv2_name, forge.alloc(name.c_str())}};
+
+ // Add initial data for canvas position etc
+ Properties init = get_initial_data(Resource::Graph::INTERNAL);
+ group_props.insert(std::make_move_iterator(init.begin()),
+ std::make_move_iterator(init.end()));
+
+ // Create a port for each element of this group type
+ std::map<uint32_t, std::pair<Raul::Path, Properties>> ports;
+ LilvNode* group = lilv_new_uri(lworld, type.c_str());
+ LilvNodes* elements = lilv_world_find_nodes(
+ lworld, group, uris.pg_element, NULL);
+ LILV_FOREACH(nodes, e, elements) {
+ const LilvNode* element = lilv_nodes_get(elements, e);
+ const LilvNode* index_node = lilv_world_get(lworld, element, uris.lv2_index, NULL);
+ const LilvNode* designation = lilv_world_get(lworld, element, uris.lv2_designation, NULL);
+ const LilvNode* label = lilv_world_get(lworld, designation, uris.rdfs_label, NULL);
+ const char* label_str = lilv_node_as_string(label);
+ const uint32_t index = _graph->num_ports() + lilv_node_as_int(index_node);
+
+ Properties port_props = {
+ {uris.rdf_type, forge.make_urid(uris.lv2_AudioPort)},
+ {uris.rdf_type, Property(is_output
+ ? uris.lv2_OutputPort
+ : uris.lv2_InputPort)},
+ {uris.pg_group, forge.make_urid(path_to_uri(path))},
+ {uris.lv2_index, forge.make(int32_t(index))},
+ {uris.lv2_name, forge.alloc(label_str)}};
+
+ const Raul::Symbol sym = Raul::Symbol::symbolify(label_str);
+ ports.insert(
+ std::make_pair(
+ index,
+ std::make_pair(_graph->path().child(sym), port_props)));
+
+ // fprintf(stderr, "Element: %s %s %s %s\n",
+ // lilv_node_as_string(index),
+ // lilv_node_as_string(designation),
+ // lilv_node_as_string(label),
+ // Raul::Symbol::symbolify(lilv_node_as_string(label)).c_str());
+ }
+
+ _app.interface()->bundle_begin();
+ _app.interface()->put(path_to_uri(path), group_props);
+ for (const auto& p : ports) {
+ _app.interface()->put(path_to_uri(p.second.first), p.second.second);
+ }
+ _app.interface()->bundle_end();
+}
+
+void
GraphCanvas::load_plugin(WPtr<PluginModel> weak_plugin)
{
SPtr<PluginModel> plugin = weak_plugin.lock();
diff --git a/src/gui/GraphCanvas.hpp b/src/gui/GraphCanvas.hpp
index c2212426..1865b846 100644
--- a/src/gui/GraphCanvas.hpp
+++ b/src/gui/GraphCanvas.hpp
@@ -34,7 +34,7 @@
namespace Ingen {
-namespace Client { class GraphModel; }
+namespace Client { class GraphModel; class GroupModel; }
namespace GUI {
@@ -68,6 +68,8 @@ public:
void remove_block(SPtr<const Client::BlockModel> bm);
void add_port(SPtr<const Client::PortModel> pm);
void remove_port(SPtr<const Client::PortModel> pm);
+ void add_group(SPtr<const Client::GroupModel> gm);
+ void remove_group(SPtr<const Client::GroupModel> gm);
void connection(SPtr<const Client::ArcModel> am);
void disconnection(SPtr<const Client::ArcModel> am);
@@ -93,6 +95,11 @@ private:
const Raul::URI& type,
bool is_output);
+ void menu_add_group(const std::string& sym_base,
+ const std::string& name_base,
+ const Raul::URI& type,
+ bool is_output);
+
void menu_load_plugin();
void menu_new_graph();
void menu_load_graph();
@@ -134,6 +141,7 @@ private:
Gtk::Menu* _menu;
Gtk::Menu* _internal_menu;
PluginMenu* _plugin_menu;
+ Gtk::Menu* _add_input_group_menu;
Gtk::MenuItem* _menu_add_audio_input;
Gtk::MenuItem* _menu_add_audio_output;
Gtk::MenuItem* _menu_add_control_input;
diff --git a/src/gui/GraphPortModule.cpp b/src/gui/GraphPortModule.cpp
index 7a0ce102..40851b6d 100644
--- a/src/gui/GraphPortModule.cpp
+++ b/src/gui/GraphPortModule.cpp
@@ -22,6 +22,7 @@
#include "ingen/Interface.hpp"
#include "ingen/client/BlockModel.hpp"
#include "ingen/client/GraphModel.hpp"
+#include "ingen/client/GroupModel.hpp"
#include "App.hpp"
#include "Style.hpp"
@@ -42,46 +43,74 @@ using namespace Client;
namespace GUI {
-GraphPortModule::GraphPortModule(GraphCanvas& canvas,
- SPtr<const Client::PortModel> model)
+GraphPortModule::GraphPortModule(GraphCanvas& canvas,
+ SPtr<const Client::GroupModel> gm,
+ SPtr<const Client::PortModel> pm)
: Ganv::Module(canvas, "", 0, 0, false) // FIXME: coords?
- , _model(model)
- , _port(NULL)
+ , _group(gm)
{
- assert(model);
-
- assert(dynamic_ptr_cast<const GraphModel>(model->parent()));
-
- set_stacked(model->polyphonic());
- if (model->is_input() && !model->is_numeric()) {
- set_is_source(true);
+ if (pm) {
+ set_stacked(pm->polyphonic());
+ if (pm->is_input() && !pm->is_numeric()) {
+ set_is_source(true);
+ }
}
- model->signal_property().connect(
- sigc::mem_fun(this, &GraphPortModule::property_changed));
+ if (gm) {
+ gm->signal_property().connect(
+ sigc::mem_fun(this, &GraphPortModule::property_changed));
+ } else if (pm) {
+ pm->signal_property().connect(
+ sigc::mem_fun(this, &GraphPortModule::property_changed));
+ }
signal_moved().connect(
sigc::mem_fun(this, &GraphPortModule::store_location));
}
GraphPortModule*
-GraphPortModule::create(GraphCanvas& canvas,
- SPtr<const PortModel> model)
+GraphPortModule::create(GraphCanvas& canvas,
+ SPtr<const GroupModel> gm,
+ SPtr<const PortModel> pm)
{
- GraphPortModule* ret = new GraphPortModule(canvas, model);
- Port* port = Port::create(canvas.app(), *ret, model, true);
+ GraphPortModule* ret = new GraphPortModule(canvas, gm, pm);
+
+ if (pm) {
+ Port* port = Port::create(canvas.app(), *ret, pm, true);
- ret->set_port(port);
- if (model->is_numeric()) {
- port->show_control();
+ if (pm->is_numeric()) {
+ port->show_control();
+ }
}
- for (const auto& p : model->properties())
+ for (const auto& p : ret->model()->properties()) {
ret->property_changed(p.first, p.second);
+ }
return ret;
}
+SPtr<const ObjectModel>
+GraphPortModule::model()
+{
+ if (_group) {
+ return _group;
+ } else if (Port* const port = get_port()) {
+ return port->model();
+ }
+ return SPtr<const ObjectModel>();
+}
+
+Port*
+GraphPortModule::get_port()
+{
+ if (begin() == end()) {
+ return nullptr;
+ }
+
+ return dynamic_cast<Port*>(*begin());
+}
+
App&
GraphPortModule::app() const
{
@@ -91,12 +120,18 @@ GraphPortModule::app() const
bool
GraphPortModule::show_menu(GdkEventButton* ev)
{
- return _port->show_menu(ev);
+ if (Port* const port = get_port()) {
+ return port->show_menu(ev);
+ }
+ fprintf(stderr, "ERR\n");
+ return false;
}
void
GraphPortModule::store_location(double ax, double ay)
{
+ fprintf(stderr, "FIXME\n");
+#if 0
const URIs& uris = app().uris();
const Atom x(app().forge().make(static_cast<float>(ax)));
@@ -110,24 +145,29 @@ GraphPortModule::store_location(double ax, double ay)
{{uris.ingen_canvasX, Property(x, Property::Graph::INTERNAL)},
{uris.ingen_canvasY, Property(y, Property::Graph::INTERNAL)}});
}
+#endif
}
void
GraphPortModule::show_human_names(bool b)
{
const URIs& uris = app().uris();
- const Atom& name = _model->get_property(uris.lv2_name);
+ const Atom& name = model()->get_property(uris.lv2_name);
if (b && name.type() == uris.forge.String) {
set_name(name.ptr<char>());
} else {
- set_name(_model->symbol().c_str());
+ set_name(model()->symbol().c_str());
}
}
void
GraphPortModule::set_name(const std::string& n)
{
- _port->set_label(n.c_str());
+ if (_group) {
+ set_label(n.c_str());
+ } else if (Port* port = get_port()) {
+ port->set_label(n.c_str());
+ }
}
void
diff --git a/src/gui/GraphPortModule.hpp b/src/gui/GraphPortModule.hpp
index 9722e330..5a98df12 100644
--- a/src/gui/GraphPortModule.hpp
+++ b/src/gui/GraphPortModule.hpp
@@ -26,9 +26,13 @@
namespace Raul { class Atom; }
-namespace Ingen { namespace Client {
+namespace Ingen {
+namespace Client {
+class GroupModel;
+class ObjectModel;
class PortModel;
-} }
+}
+}
namespace Ingen {
namespace GUI {
@@ -47,8 +51,9 @@ class GraphPortModule : public Ganv::Module
{
public:
static GraphPortModule* create(
- GraphCanvas& canvas,
- SPtr<const Client::PortModel> model);
+ GraphCanvas& canvas,
+ SPtr<const Client::GroupModel> gm,
+ SPtr<const Client::PortModel> pm);
App& app() const;
@@ -57,21 +62,20 @@ public:
void set_name(const std::string& n);
- SPtr<const Client::PortModel> port() const { return _model; }
-
protected:
- GraphPortModule(GraphCanvas& canvas,
- SPtr<const Client::PortModel> model);
+ GraphPortModule(GraphCanvas& canvas,
+ SPtr<const Client::GroupModel> gm,
+ SPtr<const Client::PortModel> pm);
bool show_menu(GdkEventButton* ev);
void set_selected(gboolean b);
- void set_port(Port* port) { _port = port; }
+ SPtr<const Client::ObjectModel> model();
+ Port* get_port();
void property_changed(const Raul::URI& predicate, const Atom& value);
- SPtr<const Client::PortModel> _model;
- Port* _port;
+ SPtr<const Client::GroupModel> _group;
};
} // namespace GUI
diff --git a/src/gui/ingen_gui.ui b/src/gui/ingen_gui.ui
index 9e751064..1339fdec 100644
--- a/src/gui/ingen_gui.ui
+++ b/src/gui/ingen_gui.ui
@@ -64,6 +64,20 @@ See COPYING file included with this distribution, or http://www.gnu.org/licenses
<object class="GtkMenu" id="input1_menu">
<property name="can_focus">False</property>
<child>
+ <object class="GtkImageMenuItem" id="canvas_menu_add_group_input_menuitem">
+ <property name="label">_Multi-Channel</property>
+ <property name="visible">True</property>
+ <property name="can_focus">False</property>
+ <property name="use_underline">True</property>
+ <property name="use_stock">False</property>
+ <child type="submenu">
+ <object class="GtkMenu" id="canvas_menu_add_group_input">
+ <property name="can_focus">False</property>
+ </object>
+ </child>
+ </object>
+ </child>
+ <child>
<object class="GtkMenuItem" id="canvas_menu_add_audio_input">
<property name="visible">True</property>
<property name="can_focus">False</property>