/*
  This file is part of Ingen.
  Copyright 2007-2012 David Robillard <http://drobilla.net/>

  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 <http://www.gnu.org/licenses/>.
*/

#include "ingen/shared/LV2Features.hpp"
#include "ingen/shared/URIs.hpp"
#include "ingen/shared/World.hpp"
#include "raul/log.hpp"

#include "AudioBuffer.hpp"
#include "Broadcaster.hpp"
#include "ControlBindings.hpp"
#include "Driver.hpp"
#include "Engine.hpp"
#include "EngineStore.hpp"
#include "MessageContext.hpp"
#include "NodeImpl.hpp"
#include "PortImpl.hpp"
#include "ProcessContext.hpp"
#include "SetPortValue.hpp"

namespace Ingen {
namespace Server {
namespace Events {

/** Internal */
SetPortValue::SetPortValue(Engine&              engine,
                           SharedPtr<Interface> client,
                           int32_t              id,
                           SampleCount          timestamp,
                           PortImpl*            port,
                           const Raul::Atom&    value)
	: Event(engine, client, id, timestamp)
	, _port_path(port->path())
	, _value(value)
	, _port(port)
{
}

SetPortValue::~SetPortValue()
{
}

bool
SetPortValue::pre_process()
{
	if (_port->is_output()) {
		return Event::pre_process_done(DIRECTION_MISMATCH);
	}

	// Port is on a message context node, set value and run
	if (_port->parent_node()->context() == Context::MESSAGE) {
		apply(_engine.message_context());
		_engine.message_context().run(
			_engine.message_context(),
			_port->parent_node(),
			_engine.driver()->frame_time() + _engine.driver()->block_length());
	}

	// Set value metadata (does not affect buffers)
	_port->set_value(_value);
	_port->set_property(_engine.world()->uris().ingen_value, _value);

	_binding = _engine.control_bindings()->port_binding(_port);

	return Event::pre_process_done(SUCCESS);
}

void
SetPortValue::execute(ProcessContext& context)
{
	assert(_time >= context.start() && _time <= context.end());

	if (_port->parent_node()->context() == Context::MESSAGE)
		return;

	apply(context);
	_engine.control_bindings()->port_value_changed(context, _port, _binding, _value);
}

void
SetPortValue::apply(Context& context)
{
	if (_status) {
		return;
	}

	Ingen::Shared::URIs& uris = _engine.world()->uris();
	Buffer* const        buf  = _port->buffer(0).get();

	AudioBuffer* const abuf = dynamic_cast<AudioBuffer*>(buf);
	if (abuf) {
		if (_value.type() != uris.forge.Float) {
			_status = TYPE_MISMATCH;
			return;
		}

		for (uint32_t v = 0; v < _port->poly(); ++v) {
			((AudioBuffer*)_port->buffer(v).get())->set_value(
				_value.get_float(), context.start(), _time);
		}
	} else if (buf->type() == uris.atom_Sequence) {
		buf->prepare_write(context);  // FIXME: incorrect
		if (buf->append_event(_time - context.start(),
		                      _value.size(),
		                      _value.type(),
		                      (const uint8_t*)_value.get_body())) {
			_port->raise_set_by_user_flag();
		} else {
			Raul::warn(Raul::fmt("Error writing to port %1%\n") % _port_path);
		}
	} else {
		Raul::warn(Raul::fmt("Unknown value type %1%\n") % _value.type());
	}
}

void
SetPortValue::post_process()
{
	respond(_status);
	if (!_status) {
		_engine.broadcaster()->set_property(
			_port_path,
			_engine.world()->uris().ingen_value,
			_value);
	}
}

} // namespace Events
} // namespace Server
} // namespace Ingen