/*
This file is part of Ingen.
Copyright 2007-2012 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
#include
#include "ingen/client/SigClientInterface.hpp"
#include "App.hpp"
#include "BreadCrumbs.hpp"
namespace Ingen {
namespace GUI {
using namespace std;
using namespace Raul;
BreadCrumbs::BreadCrumbs(App& app)
: Gtk::HBox()
, _active_path("/")
, _full_path("/")
, _enable_signal(true)
{
app.client()->signal_object_deleted().connect(
sigc::mem_fun(this, &BreadCrumbs::object_destroyed));
set_can_focus(false);
}
SharedPtr
BreadCrumbs::view(const Path& path)
{
for (std::list::const_iterator i = _breadcrumbs.begin(); i != _breadcrumbs.end(); ++i)
if ((*i)->path() == path)
return (*i)->view();
return SharedPtr();
}
/** Sets up the crumbs to display @a path.
*
* If @a path is already part of the shown path, it will be selected and the
* children preserved.
*/
void
BreadCrumbs::build(Path path, SharedPtr view)
{
bool old_enable_signal = _enable_signal;
_enable_signal = false;
// Moving to a path we already contain, just switch the active button
if (!_breadcrumbs.empty() && (path.is_parent_of(_full_path) || path == _full_path)) {
for (std::list::iterator i = _breadcrumbs.begin(); i != _breadcrumbs.end(); ++i) {
if ((*i)->path() == path) {
(*i)->set_active(true);
if (!(*i)->view())
(*i)->set_view(view);
// views are expensive, having two around for the same patch is a bug
assert((*i)->view() == view);
} else {
(*i)->set_active(false);
}
}
_active_path = path;
_enable_signal = old_enable_signal;
// Moving to a child of the full path, just append crumbs (preserve view cache)
} else if (!_breadcrumbs.empty() && path.is_child_of(_full_path)) {
string suffix = path.substr(_full_path.length());
while (suffix.length() > 0) {
if (suffix[0] == '/')
suffix = suffix.substr(1);
const string name = suffix.substr(0, suffix.find("/"));
_full_path = _full_path.child(Raul::Symbol(name));
BreadCrumb* but = create_crumb(_full_path, view);
pack_start(*but, false, false, 1);
_breadcrumbs.push_back(but);
but->show();
if (suffix.find("/") == string::npos)
break;
else
suffix = suffix.substr(suffix.find("/")+1);
}
for (std::list::iterator i = _breadcrumbs.begin(); i != _breadcrumbs.end(); ++i)
(*i)->set_active(false);
_breadcrumbs.back()->set_active(true);
// Rebuild from scratch
// Getting here is bad unless absolutely necessary, since the PatchView cache is lost
} else {
_full_path = path;
_active_path = path;
// Empty existing breadcrumbs
for (std::list::iterator i = _breadcrumbs.begin(); i != _breadcrumbs.end(); ++i)
remove(**i);
_breadcrumbs.clear();
// Add root
BreadCrumb* root_but = create_crumb(Raul::Path("/"), view);
pack_start(*root_but, false, false, 1);
_breadcrumbs.push_front(root_but);
root_but->set_active(root_but->path() == _active_path);
Path working_path("/");
string suffix = path.substr(1);
while (suffix.length() > 0) {
if (suffix[0] == '/')
suffix = suffix.substr(1);
const string name = suffix.substr(0, suffix.find("/"));
working_path = working_path.child(Raul::Symbol(name));
BreadCrumb* but = create_crumb(working_path, view);
pack_start(*but, false, false, 1);
_breadcrumbs.push_back(but);
but->set_active(working_path == _active_path);
but->show();
if (suffix.find("/") == string::npos)
break;
else
suffix = suffix.substr(suffix.find("/")+1);
}
}
_enable_signal = old_enable_signal;
}
/** Create a new crumb, assigning it a reference to @a view if their paths
* match, otherwise ignoring @a view.
*/
BreadCrumbs::BreadCrumb*
BreadCrumbs::create_crumb(const Path& path,
SharedPtr view)
{
BreadCrumb* but = manage(new BreadCrumb(path,
(view && path == view->patch()->path()) ? view : SharedPtr()));
but->signal_toggled().connect(sigc::bind(sigc::mem_fun(
this, &BreadCrumbs::breadcrumb_clicked), but));
return but;
}
void
BreadCrumbs::breadcrumb_clicked(BreadCrumb* crumb)
{
if (_enable_signal) {
_enable_signal = false;
if (!crumb->get_active()) {
// Tried to turn off the current active button, bad user, no cookie
crumb->set_active(true);
} else {
signal_patch_selected.emit(crumb->path(), crumb->view());
if (crumb->path() != _active_path)
crumb->set_active(false);
}
_enable_signal = true;
}
}
void
BreadCrumbs::object_destroyed(const URI& uri)
{
for (std::list::iterator i = _breadcrumbs.begin(); i != _breadcrumbs.end(); ++i) {
if ((*i)->path() == uri.c_str()) {
// Remove all crumbs after the removed one (inclusive)
for (std::list::iterator j = i; j != _breadcrumbs.end(); ) {
BreadCrumb* bc = *j;
j = _breadcrumbs.erase(j);
remove(*bc);
}
break;
}
}
}
void
BreadCrumbs::object_moved(const Path& old_path, const Path& new_path)
{
for (std::list::iterator i = _breadcrumbs.begin(); i != _breadcrumbs.end(); ++i) {
if ((*i)->path() == old_path)
(*i)->set_path(new_path);
}
}
} // namespace GUI
} // namespace Ingen