/* libobby - Network text editing library
 * Copyright (C) 2005, 2006 0x539 dev group
 *
 * This program 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.
 *
 * This program 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 more 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., 675 Mass Ave, Cambridge, MA 02139, USA.
 */

#ifndef _OBBY_NO_OPERATION_HPP_
#define _OBBY_NO_OPERATION_HPP_

#include "operation.hpp"

namespace obby
{

/** no_operation is an operation that does nothing. This is a helper class
 * that may be created if a delete_operation gets transformed against a
 * delete_operation which fully overlaps the range of the first operation. In
 * this case, the effect of the first operation is fully handled by the second
 * one and nothing has to be done anymore.
 */
template<typename Document>
class no_operation: public operation<Document>
{
public:
	typedef operation<Document> operation_type;
	typedef typename operation_type::document_type document_type;

	/** Standard constructor.
	 */
	no_operation();

	/** Reads a no_operation from the given network packet.
	 */
	no_operation(const net6::packet& pack, unsigned int& index);

	/** Creates a copy of this operation.
	 */
	virtual operation_type* clone() const;

	/** Creates the reverse operation of this one.
	 * @param doc Document to receive additional information from.
	 */
	virtual operation_type* reverse(const document_type& doc) const;

	/** Applies this operation to a document. Since this is a no_operation,
	 * nothing will be done.
	 */
	virtual void apply(document_type& doc, const user* author) const;

	/** Transforms <em>base_op</em> against this operation. Since this is
	 * a no_operation, nothing will be transformed.
	 */
	virtual operation_type*	transform(const operation_type& base_op) const;

	/** Includes the effect of the given insertion into this operation.
	 * Since this is a no_operation, nothing will be done.
	 */
	virtual operation_type* transform_insert(position pos,
	                                         const std::string& text) const;

	/** Includes the effect of the given deletion into this operation.
	 * Since this is a no_operation, nothing will be done.
	 */
	virtual operation_type* transform_delete(position pos,
	                                         position len) const;

	/** Appends the operation to the given packet.
	 */
	virtual void append_packet(net6::packet& pack) const;
};

template<typename Document>
no_operation<Document>::no_operation():
	operation<Document>()
{
}

template<typename Document>
no_operation<Document>::no_operation(const net6::packet& pack,
                                     unsigned int& index):
	operation<Document>()
{
}

template<typename Document>
typename no_operation<Document>::operation_type*
no_operation<Document>::clone() const
{
	return new no_operation<Document>();
}

template<typename Document>
typename no_operation<Document>::operation_type*
no_operation<Document>::reverse(const document_type& doc) const
{
	return new no_operation<Document>();
}

template<typename Document>
void no_operation<Document>::apply(document_type& doc,
                                   const user* author) const
{
}

template<typename Document>
typename no_operation<Document>::operation_type*
no_operation<Document>::transform(const operation_type& base_op) const
{
	return base_op.clone();
}

template<typename Document>
typename no_operation<Document>::operation_type*
no_operation<Document>::transform_insert(position pos,
                                         const std::string& text) const
{
	return clone();
}

template<typename Document>
typename no_operation<Document>::operation_type*
no_operation<Document>::transform_delete(position pos,
                                         position len) const
{
	return clone();
}

template<typename Document>
void no_operation<Document>::append_packet(net6::packet& pack) const
{
	pack << "noop";
}

} // namespace obby

#endif // _OBBY_NO_OPERATION_HPP_


syntax highlighted by Code2HTML, v. 0.9.1