/* 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 "App.hpp" #include "GraphTreeWindow.hpp" #include "Window.hpp" #include "WindowFactory.hpp" #include "ingen/Atom.hpp" #include "ingen/Forge.hpp" #include "ingen/Interface.hpp" #include "ingen/Log.hpp" #include "ingen/URIs.hpp" #include "ingen/client/ClientStore.hpp" #include "ingen/client/GraphModel.hpp" #include "raul/Path.hpp" #include "raul/Symbol.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace ingen { using client::GraphModel; using client::ObjectModel; namespace gui { GraphTreeWindow::GraphTreeWindow(BaseObjectType* cobject, const Glib::RefPtr& xml) : Window(cobject) , _graphs_treeview(nullptr) , _enable_signal(true) { xml->get_widget_derived("graphs_treeview", _graphs_treeview); _graph_treestore = Gtk::TreeStore::create(_graph_tree_columns); _graphs_treeview->set_window(this); _graphs_treeview->set_model(_graph_treestore); Gtk::TreeViewColumn* name_col = Gtk::manage( new Gtk::TreeViewColumn("Graph", _graph_tree_columns.name_col)); Gtk::TreeViewColumn* enabled_col = Gtk::manage( new Gtk::TreeViewColumn("Run", _graph_tree_columns.enabled_col)); name_col->set_resizable(true); name_col->set_expand(true); _graphs_treeview->append_column(*name_col); _graphs_treeview->append_column(*enabled_col); auto* enabled_renderer = dynamic_cast( _graphs_treeview->get_column_cell_renderer(1)); enabled_renderer->property_activatable() = true; _graph_tree_selection = _graphs_treeview->get_selection(); _graphs_treeview->signal_row_activated().connect( sigc::mem_fun(this, &GraphTreeWindow::event_graph_activated)); enabled_renderer->signal_toggled().connect( sigc::mem_fun(this, &GraphTreeWindow::event_graph_enabled_toggled)); _graphs_treeview->columns_autosize(); } void GraphTreeWindow::init(App& app, client::ClientStore& store) { init_window(app); store.signal_new_object().connect( sigc::mem_fun(this, &GraphTreeWindow::new_object)); } void GraphTreeWindow::new_object(const std::shared_ptr& object) { auto graph = std::dynamic_pointer_cast(object); if (graph) { add_graph(graph); } } void GraphTreeWindow::add_graph(const std::shared_ptr& pm) { if (!pm->parent()) { const auto iter = _graph_treestore->append(); auto row = *iter; if (pm->path().is_root()) { row[_graph_tree_columns.name_col] = _app->interface()->uri().string(); } else { row[_graph_tree_columns.name_col] = pm->symbol().c_str(); } row[_graph_tree_columns.enabled_col] = pm->enabled(); row[_graph_tree_columns.graph_model_col] = pm; _graphs_treeview->expand_row(_graph_treestore->get_path(iter), true); } else { const auto& children = _graph_treestore->children(); auto c = find_graph(children, pm->parent()); if (c != children.end()) { const auto iter = _graph_treestore->append(c->children()); auto row = *iter; row[_graph_tree_columns.name_col] = pm->symbol().c_str(); row[_graph_tree_columns.enabled_col] = pm->enabled(); row[_graph_tree_columns.graph_model_col] = pm; _graphs_treeview->expand_row(_graph_treestore->get_path(iter), true); } } pm->signal_property().connect( sigc::bind(sigc::mem_fun(this, &GraphTreeWindow::graph_property_changed), pm)); pm->signal_moved().connect( sigc::bind(sigc::mem_fun(this, &GraphTreeWindow::graph_moved), pm)); pm->signal_destroyed().connect( sigc::bind(sigc::mem_fun(this, &GraphTreeWindow::remove_graph), pm)); } void GraphTreeWindow::remove_graph(const std::shared_ptr& pm) { const auto i = find_graph(_graph_treestore->children(), pm); if (i != _graph_treestore->children().end()) { _graph_treestore->erase(i); } } Gtk::TreeModel::iterator GraphTreeWindow::find_graph(Gtk::TreeModel::Children root, const std::shared_ptr& graph) { for (auto c = root.begin(); c != root.end(); ++c) { std::shared_ptr pm = (*c)[_graph_tree_columns.graph_model_col]; if (graph == pm) { return c; } else if (!(*c)->children().empty()) { auto ret = find_graph(c->children(), graph); if (ret != c->children().end()) { return ret; } } } return root.end(); } /** Show the context menu for the selected graph in the graphs treeview. */ void GraphTreeWindow::show_graph_menu(GdkEventButton* ev) { const auto active = _graph_tree_selection->get_selected(); if (active) { auto row = *active; auto col = _graph_tree_columns.graph_model_col; const std::shared_ptr& pm = row[col]; if (pm) { _app->log().warn("TODO: graph menu from tree window"); } } } void GraphTreeWindow::event_graph_activated(const Gtk::TreeModel::Path& path, Gtk::TreeView::Column* col) { const auto active = _graph_treestore->get_iter(path); auto row = *active; std::shared_ptr pm = row[_graph_tree_columns.graph_model_col]; _app->window_factory()->present_graph(pm); } void GraphTreeWindow::event_graph_enabled_toggled(const Glib::ustring& path_str) { Gtk::TreeModel::Path path(path_str); auto active = _graph_treestore->get_iter(path); auto row = *active; std::shared_ptr pm = row[_graph_tree_columns.graph_model_col]; assert(pm); if (_enable_signal) { _app->set_property(pm->uri(), _app->uris().ingen_enabled, _app->forge().make(static_cast(!pm->enabled()))); } } void GraphTreeWindow::graph_property_changed( const URI& key, const Atom& value, const std::shared_ptr& graph) { const URIs& uris = _app->uris(); _enable_signal = false; if (key == uris.ingen_enabled && value.type() == uris.forge.Bool) { const auto i = find_graph(_graph_treestore->children(), graph); if (i != _graph_treestore->children().end()) { auto row = *i; row[_graph_tree_columns.enabled_col] = value.get(); } else { _app->log().error("Unable to find graph %1%\n", graph->path()); } } _enable_signal = true; } void GraphTreeWindow::graph_moved(const std::shared_ptr& graph) { _enable_signal = false; auto i = find_graph(_graph_treestore->children(), graph); if (i != _graph_treestore->children().end()) { auto row = *i; row[_graph_tree_columns.name_col] = graph->symbol().c_str(); } else { _app->log().error("Unable to find graph %1%\n", graph->path()); } _enable_signal = true; } } // namespace gui } // namespace ingen