ladish/gui/project_list.cpp

368 lines
9.2 KiB
C++

// -*- Mode: C++ ; indent-tabs-mode: t -*-
/* This file is part of Patchage.
* Copyright (C) 2008 Nedko Arnaudov <nedko@arnaudov.name>
*
* Patchage is free software; you can redistribute it and/or modify it under the
* terms of the GNU General Public License as published by the Free Software
* Foundation; either version 2 of the License, or (at your option) any later
* version.
*
* Patchage 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 General Public License for details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include <iostream>
#include <gtkmm.h>
#include <libglademm/xml.h>
#include "common.hpp"
#include "project_list.hpp"
#include "Widget.hpp"
#include "Patchage.hpp"
#include "session.hpp"
#include "project.hpp"
#include "project_properties.hpp"
#include "lash_client.hpp"
#include "globals.hpp"
struct project_list_column_record : public Gtk::TreeModel::ColumnRecord
{
Gtk::TreeModelColumn<Glib::ustring> name;
Gtk::TreeModelColumn<shared_ptr<project> > project_ptr;
Gtk::TreeModelColumn<shared_ptr<lash_client> > client_ptr;
};
struct project_list_impl : public sigc::trackable
{
Patchage *_app;
Widget<Gtk::TreeView> _widget;
project_list_column_record _columns;
Glib::RefPtr<Gtk::TreeStore> _model;
Gtk::Menu _menu_popup;
project_list_impl(
Glib::RefPtr<Gnome::Glade::Xml> xml,
Patchage * app);
void project_added(shared_ptr<project> project_ptr);
void project_closed(shared_ptr<project> project_ptr);
void project_renamed(Gtk::TreeModel::iterator iter);
void client_added(shared_ptr<lash_client> client_ptr, Gtk::TreeModel::iterator iter);
void client_removed(shared_ptr<lash_client> client_ptr, Gtk::TreeModel::iterator iter);
void client_renamed(Gtk::TreeModel::iterator iter);
bool on_button_press_event(GdkEventButton * event);
void on_menu_popup_load_project();
void on_menu_popup_save_all_projects();
void on_menu_popup_save_project(shared_ptr<project> project_ptr);
void on_menu_popup_close_project(shared_ptr<project> project_ptr);
void on_menu_popup_project_properties(shared_ptr<project> project_ptr);
void on_menu_popup_close_all_projects();
};
project_list::project_list(
Patchage * app,
session * session_ptr)
{
_impl_ptr = new project_list_impl(g_xml, app);
session_ptr->_signal_project_added.connect(mem_fun(_impl_ptr, &project_list_impl::project_added));
session_ptr->_signal_project_closed.connect(mem_fun(_impl_ptr, &project_list_impl::project_closed));
}
project_list::~project_list()
{
delete _impl_ptr;
}
project_list_impl::project_list_impl(
Glib::RefPtr<Gnome::Glade::Xml> xml,
Patchage * app)
: _app(app)
{
_widget.init(xml, "projects_list");
_columns.add(_columns.name);
_columns.add(_columns.project_ptr);
_columns.add(_columns.client_ptr);
_model = Gtk::TreeStore::create(_columns);
_widget->set_model(_model);
_widget->append_column("LASH projects", _columns.name);
_menu_popup.accelerate(*_widget);
_widget->signal_button_press_event().connect(sigc::mem_fun(*this, &project_list_impl::on_button_press_event), false);
}
bool
project_list_impl::on_button_press_event(GdkEventButton * event_ptr)
{
// Then do our custom stuff:
if (event_ptr->type == GDK_BUTTON_PRESS && event_ptr->button == 3)
{
Glib::RefPtr<Gtk::TreeView::Selection> selection = _widget->get_selection();
Gtk::TreeModel::Path path;
Gtk::TreeViewColumn * column_ptr;
int cell_x;
int cell_y;
Gtk::Menu::MenuList& menulist = _menu_popup.items();
menulist.clear();
menulist.push_back(Gtk::Menu_Helpers::MenuElem("_Load project...", sigc::mem_fun(*this, &project_list_impl::on_menu_popup_load_project)));
menulist.push_back(Gtk::Menu_Helpers::MenuElem("Save _all projects", sigc::mem_fun(*this, &project_list_impl::on_menu_popup_save_all_projects)));
if (_widget->get_path_at_pos((int)event_ptr->x, (int)event_ptr->y, path, column_ptr, cell_x, cell_y))
{
//cout << path.to_string() << endl;
selection->unselect_all();
selection->select(path);
Gtk::TreeIter iter = selection->get_selected();
shared_ptr<project> project_ptr = (*iter)[_columns.project_ptr];
string name;
if (project_ptr)
{
project_ptr->get_name(name);
menulist.push_back(
Gtk::Menu_Helpers::MenuElem(
(string)"_Save project '" + name + "'",
sigc::bind(
sigc::mem_fun(
*this,
&project_list_impl::on_menu_popup_save_project),
project_ptr)));
menulist.push_back(
Gtk::Menu_Helpers::MenuElem(
(string)"_Close project '" + name + "'",
sigc::bind(
sigc::mem_fun(
*this,
&project_list_impl::on_menu_popup_close_project),
project_ptr)));
menulist.push_back(
Gtk::Menu_Helpers::MenuElem(
(string)"_Project '" + name + "' properties",
sigc::bind(
sigc::mem_fun(
*this,
&project_list_impl::on_menu_popup_project_properties),
project_ptr)));
}
}
else
{
//cout << "No row" << endl;
selection->unselect_all();
}
menulist.push_back(Gtk::Menu_Helpers::MenuElem("Cl_ose all projects", sigc::mem_fun(*this, &project_list_impl::on_menu_popup_close_all_projects)));
_menu_popup.popup(event_ptr->button, event_ptr->time);
return true;
}
return false;
}
void
project_list_impl::on_menu_popup_load_project()
{
_app->load_project_ask();
}
void
project_list_impl::on_menu_popup_save_all_projects()
{
_app->save_all_projects();
}
void
project_list_impl::on_menu_popup_save_project(
shared_ptr<project> project_ptr)
{
string name;
project_ptr->get_name(name);
_app->save_project(name);
}
void
project_list_impl::on_menu_popup_close_project(
shared_ptr<project> project_ptr)
{
string name;
project_ptr->get_name(name);
_app->close_project(name);
}
void
project_list_impl::on_menu_popup_project_properties(
shared_ptr<project> project_ptr)
{
project_properties_dialog dialog;
dialog.run(project_ptr);
}
void
project_list_impl::on_menu_popup_close_all_projects()
{
_app->close_all_projects();
}
void
project_list_impl::project_added(
shared_ptr<project> project_ptr)
{
Gtk::TreeModel::iterator iter;
Gtk::TreeModel::iterator child_iter;
Gtk::TreeModel::Row row;
string project_name;
project_ptr->get_name(project_name);
if (project_ptr->get_modified_status())
{
project_name += " *";
}
iter = _model->append();
row = *iter;
row[_columns.name] = project_name;
row[_columns.project_ptr] = project_ptr;
project_ptr->_signal_renamed.connect(bind(mem_fun(this, &project_list_impl::project_renamed), iter));
project_ptr->_signal_modified_status_changed.connect(bind(mem_fun(this, &project_list_impl::project_renamed), iter));
project_ptr->_signal_client_added.connect(bind(mem_fun(this, &project_list_impl::client_added), iter));
project_ptr->_signal_client_removed.connect(bind(mem_fun(this, &project_list_impl::client_removed), iter));
}
void
project_list_impl::project_closed(
shared_ptr<project> project_ptr)
{
shared_ptr<project> temp_project_ptr;
Gtk::TreeModel::Children children = _model->children();
Gtk::TreeModel::Children::iterator iter = children.begin();
while(iter != children.end())
{
Gtk::TreeModel::Row row = *iter;
temp_project_ptr = row[_columns.project_ptr];
if (temp_project_ptr == project_ptr)
{
_model->erase(iter);
return;
}
iter++;
}
}
void
project_list_impl::project_renamed(
Gtk::TreeModel::iterator iter)
{
shared_ptr<project> project_ptr;
string project_name;
Gtk::TreeModel::Row row = *iter;
project_ptr = row[_columns.project_ptr];
project_ptr->get_name(project_name);
if (project_ptr->get_modified_status())
{
project_name += " *";
}
row[_columns.name] = project_name;
}
void
project_list_impl::client_added(
shared_ptr<lash_client> client_ptr,
Gtk::TreeModel::iterator iter)
{
string name;
Gtk::TreeModel::Path path = _model->get_path(iter);
Gtk::TreeModel::Row row = *iter;
client_ptr->get_name(name);
iter = _model->append(row.children());
row = *iter;
row[_columns.name] = name;
row[_columns.client_ptr] = client_ptr;
client_ptr->_signal_renamed.connect(bind(mem_fun(this, &project_list_impl::client_renamed), iter));
_widget->expand_row(path, false);
}
void
project_list_impl::client_renamed(
Gtk::TreeModel::iterator iter)
{
shared_ptr<lash_client> client_ptr;
string client_name;
Gtk::TreeModel::Row row = *iter;
client_ptr = row[_columns.client_ptr];
client_ptr->get_name(client_name);
row[_columns.name] = client_name;
}
void
project_list_impl::client_removed(
shared_ptr<lash_client> client_ptr,
Gtk::TreeModel::iterator iter)
{
string name;
Gtk::TreeModel::Path path = _model->get_path(iter);
Gtk::TreeModel::Row row = *iter;
client_ptr->get_name(name);
Gtk::TreeNodeChildren childs = row.children();
for (Gtk::TreeModel::iterator child_iter = childs.begin(); iter != childs.end(); child_iter++)
{
row = *child_iter;
if (row[_columns.name] == name)
{
_widget->expand_row(path, false);
_model->erase(child_iter);
return;
}
}
}
void
project_list::set_lash_availability(
bool lash_active)
{
_impl_ptr->_widget->set_sensitive(lash_active);
}