/*
 * The Apache Software License, Version 1.1
 *
 *
 * Copyright (c) 2000 The Apache Software Foundation.  All rights 
 * reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer. 
 *
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 *
 * 3. The end-user documentation included with the redistribution,
 *    if any, must include the following acknowledgment:  
 *       "This product includes software developed by the
 *        Apache Software Foundation (http://www.apache.org/)."
 *    Alternately, this acknowledgment may appear in the software itself,
 *    if and wherever such third-party acknowledgments normally appear.
 *
 * 4. The names "Xalan" and "Apache Software Foundation" must
 *    not be used to endorse or promote products derived from this
 *    software without prior written permission. For written 
 *    permission, please contact apache@apache.org.
 *
 * 5. Products derived from this software may not be called "Apache",
 *    nor may "Apache" appear in their name, without prior written
 *    permission of the Apache Software Foundation.
 *
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * ====================================================================
 *
 * This software consists of voluntary contributions made by many
 * individuals on behalf of the Apache Software Foundation and was
 * originally based on software copyright (c) 1999, Lotus
 * Development Corporation., http://www.lotus.com.  For more
 * information on the Apache Software Foundation, please see
 * <http://www.apache.org/>.
 */

package org.apache.xalan.lib;

import org.w3c.dom.Node;
import org.w3c.dom.Document;
import org.w3c.dom.DocumentFragment;
import org.w3c.dom.NodeList;
import org.w3c.dom.Text;
import org.w3c.dom.traversal.NodeIterator;
import org.apache.xpath.NodeSet;
import org.apache.xpath.objects.XObject;
import org.apache.xpath.objects.XBoolean;
import org.apache.xpath.objects.XNumber;
import org.apache.xpath.XPath;
import org.apache.xpath.XPathContext;
import org.apache.xpath.DOMHelper;
import org.xml.sax.SAXNotSupportedException;
import java.util.Hashtable;
import java.util.StringTokenizer;

import org.apache.xalan.extensions.ExpressionContext;

/**
 * <meta name="usage" content="general"/>
 * This class contains many of the Xalan-supplied extensions.
 * It is accessed by specifying a namespace URI as follows:
 * <pre>
 *    xmlns:xalan="http://xml.apache.org/xalan"
 * </pre>
 */

public class Extensions {

  private Extensions() {}	// Make sure class cannot be instantiated

  /**
   * This method is an extension that implements as a Xalan extension
   * the node-set function also found in xt and saxon.
   * If the argument is a Result Tree Fragment, then <code>nodeset</code>
   * returns a node-set consisting of a single root node as described in
   * section 11.1 of the XSLT 1.0 Recommendation.  If the argument is a 
   * node-set, <code>nodeset</code> returns a node-set.  If the argument
   * is a string, number, or boolean, then <code>nodeset</code> returns
   * a node-set consisting of a single root node with a single text node
   * child that is the result of calling the XPath string() function on the 
   * passed parameter.  If the argument is anything else, then a node-set
   * is returned consisting of a single root node with a single text node
   * child that is the result of calling the java <code>toString()</code>
   * method on the passed argument.
   * Most of the
   * actual work here is done in <code>MethodResolver</code> and
   * <code>XRTreeFrag</code>.
   * @param myProcessor Context passed by the extension processor
   * @param rtf Argument in the stylesheet to the nodeset extension function
   */

  public static NodeSet nodeset(ExpressionContext myProcessor, Object rtf) {

    String textNodeValue;

    if (rtf instanceof NodeIterator)
    {
      return new NodeSet((NodeIterator) rtf);
    }
    else
    {
      if (rtf instanceof String)
      {
        textNodeValue = (String) rtf;
      }
      else if (rtf instanceof Boolean)
      {
        textNodeValue = new XBoolean(((Boolean) rtf).booleanValue()).str();
      }
      else if (rtf instanceof Double)
      {
        textNodeValue = new XNumber(((Double) rtf).doubleValue()).str();
      }
      else
      {
        textNodeValue = rtf.toString();
      }
      Document myDoc = myProcessor.getContextNode().getOwnerDocument();
      Text textNode = myDoc.createTextNode(textNodeValue);
      DocumentFragment docFrag = myDoc.createDocumentFragment();
      docFrag.appendChild(textNode);
      return new NodeSet(docFrag);
    }
  }
  
	/**
	 * Returns the intersection of two node-sets.
	 * @param n1 NodeIterator for first node-set
	 * @param ni2 NodeIterator for second node-set
	 * @return a NodeSet containing the nodes in ni1 that are also
	 * in ni2
	 */	
	public static NodeSet intersection(NodeIterator ni1, NodeIterator ni2)
			throws javax.xml.transform.TransformerException
	{
		NodeSet ns1 = new NodeSet(ni1);
		NodeSet ns2 = new NodeSet(ni2);
		NodeSet inter= new NodeSet();
		inter.setShouldCacheNodes(true);
		for (int i=0; i < ns1.getLength(); i++)
		{
			Node n = ns1.elementAt(i);
			if(ns2.contains(n))
				inter.addElement(n);
		}
		return inter;
	}
	
	/**
	 * Returns the difference between two node-sets.
	 * @param n1 NodeIterator for first node-set
	 * @param ni2 NodeIterator for second node-set
	 * @return a NodeSet containing the nodes in ni1 that are not
	 * in ni2
	 */	
	public static NodeSet difference(NodeIterator ni1, NodeIterator ni2)
			throws javax.xml.transform.TransformerException	
	{
		NodeSet ns1 = new NodeSet(ni1);
		NodeSet ns2 = new NodeSet(ni2);
		// NodeSet inter= new NodeSet();
		NodeSet diff = new NodeSet();
		diff.setShouldCacheNodes(true);		
		for (int i = 0; i < ns1.getLength(); i++)
		{
			Node n = ns1.elementAt(i);
			if(!ns2.contains(n))
				diff.addElement(n);
		}
		return diff;
	}

	/**
	 * Returns node-set containing distinct string values.
	 * @param ni NodeIterator for node-set
	 * @return a NodeSet with nodes from ni containing distinct string values. 
	 * In other words, if more than one node in ni contains the same string value, 
	 * only include the first such node found.
	 */	
	public static NodeSet distinct(NodeIterator ni)
			throws javax.xml.transform.TransformerException	
	{
		NodeSet ns = new NodeSet(ni);
		NodeSet dist = new NodeSet();
		dist.setShouldCacheNodes(true);		
		Hashtable stringTable = new Hashtable();
		for (int i = 0; i < ns.getLength(); i++)
		{
			Node n = ns.elementAt(i);
      String key = DOMHelper.getNodeData(n);
      if (!stringTable.containsKey(key))
			{
        stringTable.put(key, n);
        dist.addElement(n);
			}
		}
		return dist;
	}

	/**
	 * Returns true of both node-sets contain the same set of nodes.
	 * @param n1 NodeIterator for first node-set
	 * @param ni2 NodeIterator for second node-set
	 * @return true if ni1 and ni2 contain exactly the same set of nodes.
	 */	
	public static boolean hasSameNodes(NodeIterator ni1, NodeIterator ni2)
	{
		NodeSet ns1 = new NodeSet(ni1);
		NodeSet ns2 = new NodeSet(ni2);		
		if (ns1.getLength() != ns2.getLength())
			return false;
		for (int i = 0; i < ns1.getLength(); i++)
		{
			Node n = ns1.elementAt(i);
			if(!ns2.contains(n))
				return false;
		}
		return true;
	}  

	/**
	 * Returns the result of evaluating the argument as a string containing
   * an XPath expression.  Used where the XPath expression is not known until
   * run-time.  The expression is evaluated as if the run-time value of the
   * argument appeared in place of the evaluate function call at compile time.
	 * @param myContext an <code>ExpressionContext</code> passed in by the
   *                  extension mechanism.  This must be an XPathContext.
	 * @param xpathExtr The XPath expression to be evaluated.
	 * @return the XObject resulting from evaluating the XPath
	 */	
	public static XObject evaluate(ExpressionContext myContext, String xpathExpr)
                  throws SAXNotSupportedException, Exception
	{
    if (myContext instanceof XPathContext)
    {
      try
      {
        XPathContext xctxt = (XPathContext) myContext;
        XPath dynamicXPath = new XPath(xpathExpr,
                                  xctxt.getSAXLocator(),
                                  xctxt.getNamespaceContext(),
                                  XPath.SELECT);
        return dynamicXPath.execute(xctxt,
                                    myContext.getContextNode(),
                                    xctxt.getNamespaceContext());
      }
      catch (Exception e)
      {
        throw e;
      }
    }
    else
      throw new SAXNotSupportedException("Invalid context passed to evaluate " + myContext);
	}  

	/**
	 * Returns a NodeSet containing one text node for each token in the first argument.
   * Delimiters are specified in the second argument.
   * Tokens are determined by a call to <code>StringTokenizer</code>.
   * If the first argument is an empty string or contains only delimiters, the result
   * will be an empty NodeSet.
   * Contributed to XalanJ1 by <a href="mailto:benoit.cerrina@writeme.com">Benoit Cerrina</a>.  
	 * @param myContext an <code>ExpressionContext</code> passed in by the
   *                  extension mechanism.  This must be an XPathContext.
	 * @param toTokenize The string to be split into text tokens.
   * @param delims The delimiters to use.
	 * @return a NodeSet as described above.
   *
	 */	
	public static NodeSet tokenize(ExpressionContext myContext, String toTokenize, String delims)
	{
    Document lDoc = myContext.getContextNode().getOwnerDocument();
    StringTokenizer lTokenizer = new StringTokenizer(toTokenize, delims);
    NodeSet resultSet = new NodeSet();
    while (lTokenizer.hasMoreTokens())
    {
      resultSet.addNode(lDoc.createTextNode(lTokenizer.nextToken()));
    }
    return resultSet;
}  

	/**
	 * Returns a NodeSet containing one text node for each token in the first argument.
   * Delimiters are whitespace.  That is, the delimiters that are used are tab (&#x09),
   * linefeed (&#x0A), return (&#x0D), and space (&#x20).
   * Tokens are determined by a call to <code>StringTokenizer</code>.
   * If the first argument is an empty string or contains only delimiters, the result
   * will be an empty NodeSet.
   * Contributed to XalanJ1 by <a href="mailto:benoit.cerrina@writeme.com">Benoit Cerrina</a>.  
	 * @param myContext an <code>ExpressionContext</code> passed in by the
   *                  extension mechanism.  This must be an XPathContext.
	 * @param toTokenize The string to be split into text tokens.
	 * @return a NodeSet as described above.
   *
	 */	
	public static NodeSet tokenize(ExpressionContext myContext, String toTokenize)
	{
    return tokenize(myContext, toTokenize, " \t\n\r");
  }  

}
