/* This file is part of Ingen.
 * Copyright (C) 2007-2009 David 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 <boost/format.hpp>
#include "raul/Array.hpp"
#include "raul/List.hpp"
#include "raul/Maid.hpp"
#include "raul/Path.hpp"
#include "ClientBroadcaster.hpp"
#include "ConnectionImpl.hpp"
#include "events/DisconnectAll.hpp"
#include "events/Disconnect.hpp"
#include "Engine.hpp"
#include "InputPort.hpp"
#include "NodeImpl.hpp"
#include "EngineStore.hpp"
#include "OutputPort.hpp"
#include "PatchImpl.hpp"
#include "PortImpl.hpp"
#include "Request.hpp"
#include "util.hpp"

using namespace std;
using namespace Raul;

namespace Ingen {
namespace Events {


DisconnectAll::DisconnectAll(Engine& engine, SharedPtr<Request> request, SampleCount timestamp, const Path& parent_path, const Path& node_path)
	: QueuedEvent(engine, request, timestamp)
	, _parent_path(parent_path)
	, _path(node_path)
	, _parent(NULL)
	, _node(NULL)
	, _port(NULL)
	, _compiled_patch(NULL)
	, _deleting(false)
{
}


/** Internal version for use by other events.
 */
DisconnectAll::DisconnectAll(Engine& engine, PatchImpl* parent, GraphObjectImpl* object)
	: QueuedEvent(engine)
	, _parent_path(parent->path())
	, _path(object->path())
	, _parent(parent)
	, _node(dynamic_cast<NodeImpl*>(object))
	, _port(dynamic_cast<PortImpl*>(object))
	, _compiled_patch(NULL)
	, _deleting(true)
{
}


DisconnectAll::~DisconnectAll()
{
	for (Raul::List<Disconnect*>::iterator i = _disconnect_events.begin(); i != _disconnect_events.end(); ++i)
		delete (*i);
}


void
DisconnectAll::pre_process()
{
	if (!_deleting) {
		_parent = _engine.engine_store()->find_patch(_parent_path);

		if (_parent == NULL) {
			_error = PARENT_NOT_FOUND;
			QueuedEvent::pre_process();
			return;
		}

		GraphObjectImpl* object = _engine.engine_store()->find_object(_path);

		if (object == NULL) {
			_error = OBJECT_NOT_FOUND;
			QueuedEvent::pre_process();
			return;
		}

		if (object->parent_patch() != _parent && object->parent()->parent_patch() != _parent) {
			_error = INVALID_PARENT_PATH;
			QueuedEvent::pre_process();
			return;
		}

		// Only one of these will succeed
		_node = dynamic_cast<NodeImpl*>(object);
		_port = dynamic_cast<PortImpl*>(object);

		assert((_node || _port) && !(_node && _port));
	}

	if (_node) {
		for (PatchImpl::Connections::const_iterator i = _parent->connections().begin();
				i != _parent->connections().end(); ++i) {
			ConnectionImpl* c = (ConnectionImpl*)i->second.get();
			const bool reconnect_input = !_deleting || (c->dst_port()->parent_node() != _node);
			if ((c->src_port()->parent_node() == _node || c->dst_port()->parent_node() == _node)
					&& !c->pending_disconnection()) {
				Disconnect* ev = new Disconnect(_engine,
						SharedPtr<Request>(), _time, c->src_port(), c->dst_port(), reconnect_input);
				ev->pre_process();
				_disconnect_events.push_back(new Raul::List<Disconnect*>::Node(ev));
				c->pending_disconnection(true);
			}
		}
	} else { // _port
		for (PatchImpl::Connections::const_iterator i = _parent->connections().begin();
				i != _parent->connections().end(); ++i) {
			ConnectionImpl* c = (ConnectionImpl*)i->second.get();
			const bool reconnect_input = !_deleting || (c->dst_port()->parent_node() != _node);
			if ((c->src_port() == _port || c->dst_port() == _port) && !c->pending_disconnection()) {
				Disconnect* ev = new Disconnect(_engine,
						SharedPtr<Request>(), _time, c->src_port(), c->dst_port(), reconnect_input);
				ev->pre_process();
				_disconnect_events.push_back(new Raul::List<Disconnect*>::Node(ev));
				c->pending_disconnection(true);
			}
		}
	}

	if (!_deleting && _parent->enabled())
		_compiled_patch = _parent->compile();

	QueuedEvent::pre_process();
}


void
DisconnectAll::execute(ProcessContext& context)
{
	QueuedEvent::execute(context);

	if (_error == NO_ERROR) {
		for (Raul::List<Disconnect*>::iterator i = _disconnect_events.begin();
				i != _disconnect_events.end(); ++i)
			(*i)->execute(context);
	}

	_engine.maid()->push(_parent->compiled_patch());
	_parent->compiled_patch(_compiled_patch);
}


void
DisconnectAll::post_process()
{
	if (_error == NO_ERROR) {
		if (_request)
			_request->respond_ok();
		for (Raul::List<Disconnect*>::iterator i = _disconnect_events.begin();
				i != _disconnect_events.end(); ++i)
			(*i)->post_process();
	} else {
		if (_request) {
			boost::format fmt("Unable to disconnect %1% (%2%)");
			fmt % _path;
			switch (_error) {
				case INVALID_PARENT_PATH:
					fmt % string("Invalid parent path: ").append(_parent_path.str());
					break;
				case PARENT_NOT_FOUND:
					fmt % string("Unable to find parent: ").append(_parent_path.str());
					break;
				case OBJECT_NOT_FOUND:
					fmt % string("Unable to find object");
				default:
					break;
			}
			_request->respond_error(fmt.str());
		}
	}
}


} // namespace Ingen
} // namespace Events