blob: 761485e184f7e83b9ff2ff0ab5e1ee0584270224 [file] [log] [blame]
/*
* The Apache Software License, Version 1.1
*
*
* Copyright (c) 1999 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.xml.dtm.ref;
import org.apache.xml.dtm.*;
import org.w3c.dom.Node;
import org.w3c.dom.DOMException;
/**
* <code>DTMNodeList</code> gives us an implementation of the DOM's
* NodeList interface wrapped around a DTM Iterator. The author
* considers this something of an abominations, since NodeList was not
* intended to be a general purpose "list of nodes" API and is
* generally considered by the DOM WG to have be a mistake... but I'm
* told that some of the XPath/XSLT folks say they must have this
* solution.
*
* Please note that this is not necessarily equivlaent to a DOM
* NodeList operating over the same document. In particular:
* <ul>
*
* <li>If there are several Text nodes in logical succession (ie,
* across CDATASection and EntityReference boundaries), we will return
* only the first; the caller is responsible for stepping through
* them.
* (%REVIEW% Provide a convenience routine here to assist, pending
* proposed DOM Level 3 getAdjacentText() operation?) </li>
*
* <li>Since the whole XPath/XSLT architecture assumes that the source
* document is not altered while we're working with it, we do not
* promise to implement the DOM NodeList's "live view" response to
* document mutation. </li>
*
* </ul>
*
* <p>State: In progress!!</p>
* */
public class DTMNodeList implements org.w3c.dom.NodeList
{
private DTMIterator dtm_iter;
private boolean valid=true;
private int m_firstChild;
private DTM m_parentDTM;
//================================================================
// Methods unique to this class
/** Public constructor: Wrap a DTMNodeList around an existing
* and preconfigured DTMIterator
*
* WARNING: THIS HAS THE SIDE EFFECT OF ISSUING setShouldCacheNodes(true)
* AGAINST THE DTMIterator.
* */
public DTMNodeList(DTMIterator dtmIterator)
{
int pos = dtmIterator.getCurrentPos();
try
{
dtm_iter=(DTMIterator)dtmIterator.cloneWithReset();
}
catch(CloneNotSupportedException cnse) {}
dtm_iter.setShouldCacheNodes(true);
dtm_iter.runTo(-1);
dtm_iter.setCurrentPos(pos);
}
/** Public constructor: Create a NodeList to support
* DTMNodeProxy.getChildren().
*
* Unfortunately AxisIterators and DTMIterators don't share an API,
* so I can't use the existing Axis.CHILD iterator. Rather than
* create Yet Another Class, let's set up a special case of this
* one.
*
* @param parentDTM The DTM containing this node
* @param parentHandle DTM node-handle integer
* */
public DTMNodeList(DTM parentDTM,int parentHandle)
{
dtm_iter=null;
m_parentDTM=parentDTM;
m_firstChild=parentDTM.getFirstChild(parentHandle);
}
/** Access the wrapped DTMIterator. I'm not sure whether anyone will
* need this or not, but let's write it and think about it.
* */
DTMIterator getDTMIterator()
{
return dtm_iter;
}
//================================================================
// org.w3c.dom.NodeList API follows
/**
* Returns the <code>index</code>th item in the collection. If
* <code>index</code> is greater than or equal to the number of nodes in
* the list, this returns <code>null</code>.
* @param indexIndex into the collection.
* @return The node at the <code>index</code>th position in the
* <code>NodeList</code>, or <code>null</code> if that is not a valid
* index.
*/
public Node item(int index)
{
if(dtm_iter!=null)
{
int handle=dtm_iter.item(index);
return dtm_iter.getDTM(handle).getNode(handle);
}
else
{
int handle=m_firstChild;
while(--index>=0 && handle!=DTM.NULL)
handle=m_parentDTM.getNextSibling(handle);
return m_parentDTM.getNode(handle);
}
}
/**
* The number of nodes in the list. The range of valid child node indices
* is 0 to <code>length-1</code> inclusive.
*/
public int getLength()
{
if(dtm_iter!=null)
{
return dtm_iter.getLength();
}
else
{
int count=0;
for(int handle=m_firstChild;
handle!=DTM.NULL;
handle=m_parentDTM.getNextSibling(handle))
++count;
return count;
}
}
}