/* This file is part of Ingen.
 * Copyright (C) 2007 Dave Robillard <http://drobilla.net>
 * 
 * Ingen 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.
 * 
 * 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 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 <cassert>
#include <fstream>
#include "interface/EngineInterface.hpp"
#include "client/PatchModel.hpp"
#include "App.hpp"
#include "PatchView.hpp"
#include "PatchCanvas.hpp"
#include "LoadPluginWindow.hpp"
#include "NewSubpatchWindow.hpp"
#include "LoadSubpatchWindow.hpp"
#include "NodeControlWindow.hpp"
#include "PatchPropertiesWindow.hpp"
#include "PatchTreeWindow.hpp"
#include "GladeFactory.hpp"

namespace Ingen {
namespace GUI {


PatchView::PatchView(BaseObjectType* cobject, const Glib::RefPtr<Gnome::Glade::Xml>& xml)
	: Gtk::Box(cobject)
	, _breadcrumb_container(NULL)
	, _enable_signal(true)
{
	property_visible() = false;

	xml->get_widget("patch_view_breadcrumb_container", _breadcrumb_container);
	xml->get_widget("patch_view_toolbar", _toolbar);
	xml->get_widget("patch_view_process_but", _process_but);
	xml->get_widget("patch_view_poly_spin", _poly_spin);
	xml->get_widget("patch_view_clear_but", _clear_but);
	xml->get_widget("patch_view_refresh_but", _refresh_but);
	xml->get_widget("patch_view_save_but", _save_but);
	xml->get_widget("patch_view_zoom_full_but", _zoom_full_but);
	xml->get_widget("patch_view_zoom_normal_but", _zoom_normal_but);
	xml->get_widget("patch_view_edit_mode_but", _edit_mode_but);
	xml->get_widget("patch_view_scrolledwindow", _canvas_scrolledwindow);

	_toolbar->set_toolbar_style(Gtk::TOOLBAR_ICONS);
	_canvas_scrolledwindow->property_hadjustment().get_value()->set_step_increment(10);
	_canvas_scrolledwindow->property_vadjustment().get_value()->set_step_increment(10);

}


void
PatchView::set_patch(SharedPtr<PatchModel> patch)
{
	assert(!_canvas); // FIXME: remove

	//cerr << "Creating view for " << patch->path() << endl;

	assert(_breadcrumb_container); // ensure created

	_patch = patch;
	_canvas = SharedPtr<PatchCanvas>(new PatchCanvas(patch, 1600*2, 1200*2));
	_canvas->build();

	_canvas_scrolledwindow->add(*_canvas);

	_poly_spin->set_value(patch->poly());

	for (GraphObject::Properties::const_iterator i = patch->properties().begin();
			i != patch->properties().end(); ++i)
		property_changed(i->first, i->second);

	// Connect model signals to track state
	patch->signal_property.connect(sigc::mem_fun(this, &PatchView::property_changed));

	// Connect widget signals to do things
	_process_but->signal_toggled().connect(sigc::mem_fun(this, &PatchView::process_toggled));
	_clear_but->signal_clicked().connect(sigc::mem_fun(this, &PatchView::clear_clicked));
	_refresh_but->signal_clicked().connect(sigc::mem_fun(this, &PatchView::refresh_clicked));

	_zoom_normal_but->signal_clicked().connect(sigc::bind(sigc::mem_fun(
		_canvas.get(), &PatchCanvas::set_zoom), 1.0));

	_zoom_full_but->signal_clicked().connect(
		sigc::mem_fun(_canvas.get(), &PatchCanvas::zoom_full));

	patch->signal_editable.connect(sigc::mem_fun(
			*this, &PatchView::on_editable_sig));

	_edit_mode_but->signal_toggled().connect(sigc::mem_fun(
			*this, &PatchView::editable_toggled));
	
	_poly_spin->signal_value_changed().connect(
			sigc::mem_fun(*this, &PatchView::poly_changed));

	_canvas->signal_item_entered.connect(
			sigc::mem_fun(*this, &PatchView::canvas_item_entered));
	
	_canvas->signal_item_left.connect(
			sigc::mem_fun(*this, &PatchView::canvas_item_left));

	_canvas->grab_focus();
}


PatchView::~PatchView()
{
	//cerr << "Destroying view for " << _patch->path() << endl;
}


SharedPtr<PatchView>
PatchView::create(SharedPtr<PatchModel> patch)

{
	
const Glib::RefPtr<Gnome::Glade::Xml>& xml = GladeFactory::new_glade_reference("patch_view_box");
	PatchView* result = NULL;
	xml->get_widget_derived("patch_view_box", result);
	assert(result);
	result->set_patch(patch);
	return SharedPtr<PatchView>(result);
}


void
PatchView::on_editable_sig(bool editable)
{
	_edit_mode_but->set_active(editable);
	_canvas->lock(!editable);
}


void
PatchView::editable_toggled()
{
	const bool editable = _edit_mode_but->get_active();
	set_editable(editable);
}


void
PatchView::set_editable(bool editable)
{
	_patch->set_editable(editable);
	_canvas->lock(!editable);
}


void
PatchView::canvas_item_entered(Gnome::Canvas::Item* item)
{
	NodeModule* m = dynamic_cast<NodeModule*>(item);
	if (m) {
		signal_object_entered.emit(m->node().get());
		return;
	}
	
	Port* p = dynamic_cast<Port*>(item);
	if (p)
		signal_object_entered.emit(p->model().get());
}


void
PatchView::canvas_item_left(Gnome::Canvas::Item* item)
{
	NodeModule* m = dynamic_cast<NodeModule*>(item);
	if (m) {
		signal_object_left.emit(m->node().get());
		return;
	}
	
	Port* p = dynamic_cast<Port*>(item);
	if (p)
		signal_object_left.emit(p->model().get());
}


void
PatchView::process_toggled()
{
	if (!_enable_signal)
		return;

	App::instance().engine()->set_property(_patch->path(), "ingen:enabled",
			(bool)_process_but->get_active());
}


void
PatchView::poly_changed()
{
	App::instance().engine()->set_property(_patch->path(), "ingen:polyphony",
			_poly_spin->get_value_as_int());
}


void
PatchView::clear_clicked()
{
	App::instance().engine()->clear_patch(_patch->path());
}


void
PatchView::refresh_clicked()
{
	App::instance().engine()->request_object(_patch->path());
}


void
PatchView::property_changed(const std::string& predicate, const Raul::Atom& value)
{
	_enable_signal = false;
	if (predicate == "ingen:enabled") {
	   if (value.type() == Atom::BOOL)
		   _process_but->set_active(value.get_bool());
	   else
		   cerr << "WARNING: Bad type for ingen:enabled property: " << value.type() << endl;
	}
	_enable_signal = true;
}


} // namespace GUI
} // namespace Ingen