/*
 *   surf - visualizing algebraic curves and algebraic surfaces
 *   Copyright (C) 1996-1997 Friedrich-Alexander-Universitaet
 *                           Erlangen-Nuernberg
 *                 1997-2000 Johannes Gutenberg-Universitaet Mainz
 *   Authors: Stephan Endrass, Hans Huelf, Ruediger Oertel,
 *            Kai Schneider, Ralf Schmitt, Johannes Beigel
 *
 *   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 CMonom_H
#define CMonom_H

#include <stdlib.h>
#include <string.h>
#include <iostream.h>

#include "doubleMath.h"


template<class Coeff, int num> 
class CMonom
{
public:
	CMonom() {memset(exponent, 0, sizeof(int[num]));};
	CMonom(const CMonom &m) : coeff(m.coeff) {memcpy(exponent, m.exponent, sizeof(int[num])); coeff=m.coeff;};
	~CMonom(){};
	
	CMonom &operator= (const CMonom &m) 
		{
			memcpy(exponent, m.exponent, sizeof(int[num]));
			coeff=m.coeff; 
			return *this;
		};
	
	CMonom &operator+=(const CMonom &m) {coeff+=m.coeff; return *this;};
	CMonom &operator-=(const CMonom &m) {coeff-=m.coeff; return *this;};
	CMonom &operator*=(const CMonom &m) {int i;for(i=0; i<num; i++) exponent[i]+=m.exponent[i];coeff*=m.coeff; return *this;};
//	CMonom &operator/=(const CMonom &m) {int i;for(i=0; i<num; i++) exponent[i]-=m.exponent[i];coeff/=m.coeff; return *this;};
	
	int lexcmp (const CMonom &m) const 
		{
			int i; 
			for(i=0; i<num && exponent[i]==m.exponent[i]; i++)
				;
			return i==num ? 0 : exponent[i]-m.exponent[i];
		};

	void print(ostream &os) const 
		{
			static const char vars[]="xyzabcdefg";
			os << "(" << coeff << ")";
			int i;
			for (i=0; i<num; i++) {
				os << "*" << vars[i] << "^" << exponent[i];
			}
		}
	
	Coeff &getCoeff() {return coeff;};
	const Coeff&getCoeff() const {return coeff;};
	
	int &getExponent(int var) {assert(var>=0 && var<num); return exponent[var];};

	static int size() {return num;};

// 	friend ostream & operator << (Monom &m, ostream &os)
// 		{
// 			m.print(os);
// 			return os;
// 		}

// 	friend bool isNull (Monom &m)
// 		{
// 			return isNull (m.getCoeff());
// 		}
protected:
	int exponent[num];
	Coeff coeff;
};

template<class Coeff,  int num> 
ostream & operator<<(ostream &os, const CMonom<Coeff,num> &m)
{
	m.print(os);
 	return os;
}

// template<class Type>
// ostream & operator<<(ostream &os, const Type & m)
// {
// 	m.print(os);
// 	return os;
// }

template<class Coeff,  int num> 
inline int isNull(const CMonom<Coeff,num> &m)
{
	return isNull(m.getCoeff());
}

// template<class Type>
// inline void negate (Type &m)
// {
// 	negate(m.getCoeff());
// }

template<class Coeff,  int num> 
inline void negate(CMonom<Coeff,num> &m)
{
	negate(m.getCoeff());
}


template<class Coeff> 
CMonom<Coeff, 2> monom(const Coeff &coeff, int e0, int e1)
{
	CMonom<Coeff, 2> mon;
	mon.getCoeff()=coeff;
	mon.getExponent(0)=e0;
	mon.getExponent(1)=e1;
	return mon;
}

template<class Coeff> 
CMonom<Coeff, 3> monom(const Coeff &coeff, int e0, int e1, int e2)
{
	CMonom<Coeff, 3> mon;
	mon.getCoeff()=coeff;

	mon.getExponent(0)=e0;
	mon.getExponent(1)=e1;
	mon.getExponent(2)=e2;
	// mon.getExponent(3)=0;
	return mon;
}

#endif


syntax highlighted by Code2HTML, v. 0.9.1