net.sf.saxon.tree.tiny.PrecedingSiblingEnumeration Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of saxon-he Show documentation
Show all versions of saxon-he Show documentation
An OSGi bundle for Saxon-HE
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2013 Saxonica Limited.
// This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
// This Source Code Form is "Incompatible With Secondary Licenses", as defined by the Mozilla Public License, v. 2.0.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
package net.sf.saxon.tree.tiny;
import net.sf.saxon.om.NodeInfo;
import net.sf.saxon.pattern.NodeTest;
import net.sf.saxon.tree.iter.AxisIterator;
import net.sf.saxon.tree.iter.AxisIteratorImpl;
/**
* This class supports the preceding-sibling axis.
* The starting node must be an element, text node, comment, or processing instruction:
* to ensure this, construct the enumeration using NodeInfo#getEnumeration()
*/
final class PrecedingSiblingEnumeration extends AxisIteratorImpl {
private TinyTree document;
private TinyNodeImpl startNode;
private int nextNodeNr;
private NodeTest test;
private TinyNodeImpl parentNode;
PrecedingSiblingEnumeration(TinyTree doc, /*@NotNull*/ TinyNodeImpl node, NodeTest nodeTest) {
document = doc;
document.ensurePriorIndex();
test = nodeTest;
startNode = node;
nextNodeNr = node.nodeNr;
parentNode = node.parent; // doesn't matter if this is null (unknown)
}
/*@Nullable*/ public NodeInfo next() {
if (nextNodeNr < 0) {
// This check is needed because an errant caller can call next() again after hitting the end of sequence
return null;
}
while (true) {
nextNodeNr = document.prior[nextNodeNr];
if (nextNodeNr < 0) {
current = null;
position = -1;
return null;
}
if (test.matches(document, nextNodeNr)) {
position++;
TinyNodeImpl next = document.getNode(nextNodeNr);
next.setParentNode(parentNode);
return (current = next);
}
}
}
/**
* Get another enumeration of the same nodes
*/
/*@NotNull*/ public AxisIterator getAnother() {
return new PrecedingSiblingEnumeration(document, startNode, test);
}
}