/* RelationalExpr.java -- 
   Copyright (C) 2004 Free Software Foundation, Inc.

This file is part of GNU Classpath.

GNU Classpath 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, or (at your option)
any later version.

GNU Classpath 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 GNU Classpath; see the file COPYING.  If not, write to the
Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
02111-1307 USA.

Linking this library statically or dynamically with other modules is
making a combined work based on this library.  Thus, the terms and
conditions of the GNU General Public License cover the whole
combination.

As a special exception, the copyright holders of this library give you
permission to link this library with independent modules to produce an
executable, regardless of the license terms of these independent
modules, and to copy and distribute the resulting executable under
terms of your choice, provided that you also meet, for each linked
independent module, the terms and conditions of the license of that
module.  An independent module is a module which is not derived from
or based on this library.  If you modify this library, you may extend
this exception to your version of the library, but you are not
obligated to do so.  If you do not wish to do so, delete this
exception statement from your version. */

package gnu.xml.xpath;

import org.w3c.dom.Node;

/**
 * Numerical comparison expression.
 *
 * @author <a href='mailto:dog@gnu.org'>Chris Burdess</a>
 */
final class RelationalExpr
  extends Expr
{

  final Expr lhs;
  final Expr rhs;
  final boolean lt;
  final boolean eq;

  RelationalExpr(Expr lhs, Expr rhs, boolean lt, boolean eq)
  {
    this.lhs = lhs;
    this.rhs = rhs;
    this.lt = lt;
    this.eq = eq;
  }

  public Object evaluate(Node context, int pos, int len)
  {
    Object left = lhs.evaluate(context, pos, len);
    Object right = rhs.evaluate(context, pos, len);
    double ln = _number(context, left);
    double rn = _number(context, right);
    if (eq && ln == rn)
      {
        return Boolean.TRUE;
      }
    if (lt)
      {
        if (ln < rn || Double.isInfinite(rn))
          {
            return Boolean.TRUE;
          }
      }
    else
      {
        if (ln > rn || Double.isInfinite(ln))
          {
            return Boolean.TRUE;
          }
      }
    return Boolean.FALSE;
  }

  public Expr clone(Object context)
  {
    return new RelationalExpr(lhs.clone(context), rhs.clone(context), lt, eq);
  }

  public String toString()
  {
    return lhs + " " + (lt ? "<" : ">") + (eq ? "=" : "") + " " + rhs;
  }
  
}
