/* * Copyright 1999-2004 The Apache Software Foundation. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* * $Id: KeyRefIterator.java,v 1.19 2005/01/23 00:37:17 mcnamara Exp $ */ package org.apache.xalan.transformer; import java.util.Vector; import org.apache.xalan.res.XSLMessages; import org.apache.xalan.res.XSLTErrorResources; import org.apache.xalan.templates.KeyDeclaration; import org.apache.xml.dtm.DTM; import org.apache.xml.dtm.DTMIterator; import org.apache.xml.utils.QName; import org.apache.xml.utils.XMLString; import org.apache.xpath.objects.XNodeSet; import org.apache.xpath.objects.XObject; /** * This class filters nodes from a key iterator, according to * whether or not the use value matches the ref value. * @xsl.usage internal */ public class KeyRefIterator extends org.apache.xpath.axes.ChildTestIterator { static final long serialVersionUID = 3837456451659435102L; /** * Constructor KeyRefIterator * * * @param ref Key value to match * @param ki The main key iterator used to walk the source tree */ public KeyRefIterator(QName name, XMLString ref, Vector keyDecls, DTMIterator ki) { super(null); m_name = name; m_ref = ref; m_keyDeclarations = keyDecls; m_keysNodes = ki; setWhatToShow(org.apache.xml.dtm.DTMFilter.SHOW_ALL); } DTMIterator m_keysNodes; /** * Get the next node via getNextXXX. Bottlenecked for derived class override. * @return The next node on the axis, or DTM.NULL. */ protected int getNextNode() { int next; while(DTM.NULL != (next = m_keysNodes.nextNode())) { if(DTMIterator.FILTER_ACCEPT == filterNode(next)) break; } m_lastFetched = next; return next; } /** * Test whether a specified node is visible in the logical view of a * TreeWalker or NodeIterator. This function will be called by the * implementation of TreeWalker and NodeIterator; it is not intended to * be called directly from user code. * * @param testNode The node to check to see if it passes the filter or not. * * @return a constant to determine whether the node is accepted, * rejected, or skipped, as defined above . */ public short filterNode(int testNode) { boolean foundKey = false; Vector keys = m_keyDeclarations; QName name = m_name; KeyIterator ki = (KeyIterator)(((XNodeSet)m_keysNodes).getContainedIter()); org.apache.xpath.XPathContext xctxt = ki.getXPathContext(); if(null == xctxt) assertion(false, "xctxt can not be null here!"); try { XMLString lookupKey = m_ref; // System.out.println("lookupKey: "+lookupKey); int nDeclarations = keys.size(); // Walk through each of the declarations made with xsl:key for (int i = 0; i < nDeclarations; i++) { KeyDeclaration kd = (KeyDeclaration) keys.elementAt(i); // Only continue if the name on this key declaration // matches the name on the iterator for this walker. if (!kd.getName().equals(name)) continue; foundKey = true; // xctxt.setNamespaceContext(ki.getPrefixResolver()); // Query from the node, according the the select pattern in the // use attribute in xsl:key. XObject xuse = kd.getUse().execute(xctxt, testNode, ki.getPrefixResolver()); if (xuse.getType() != xuse.CLASS_NODESET) { XMLString exprResult = xuse.xstr(); if (lookupKey.equals(exprResult)) return DTMIterator.FILTER_ACCEPT; } else { DTMIterator nl = ((XNodeSet)xuse).iterRaw(); int useNode; while (DTM.NULL != (useNode = nl.nextNode())) { DTM dtm = getDTM(useNode); XMLString exprResult = dtm.getStringValue(useNode); if ((null != exprResult) && lookupKey.equals(exprResult)) return DTMIterator.FILTER_ACCEPT; } } } // end for(int i = 0; i < nDeclarations; i++) } catch (javax.xml.transform.TransformerException te) { throw new org.apache.xml.utils.WrappedRuntimeException(te); } if (!foundKey) throw new RuntimeException( XSLMessages.createMessage( XSLTErrorResources.ER_NO_XSLKEY_DECLARATION, new Object[] { name.getLocalName()})); return DTMIterator.FILTER_REJECT; } protected XMLString m_ref; protected QName m_name; /** Vector of Key declarations in the stylesheet. * @serial */ protected Vector m_keyDeclarations; }