All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.sf.saxon.tree.tiny.TinyAttributeImpl Maven / Gradle / Ivy

There is a newer version: 12.5
Show newest version
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2018-2023 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.event.Receiver;
import net.sf.saxon.lib.FeatureKeys;
import net.sf.saxon.om.*;
import net.sf.saxon.s9api.Location;
import net.sf.saxon.str.StringView;
import net.sf.saxon.str.UnicodeString;
import net.sf.saxon.trans.XPathException;

import net.sf.saxon.type.BuiltInAtomicType;
import net.sf.saxon.type.SchemaType;
import net.sf.saxon.type.Type;


/**
 * A node in the XML parse tree representing an attribute. Note that this is
 * generated only "on demand", when the attribute is selected by a select pattern.
 *
 */

final public class TinyAttributeImpl extends TinyNodeImpl {

    public TinyAttributeImpl(TinyTree tree, int nodeNr) {
        this.tree = tree;
        this.nodeNr = nodeNr;
    }


    @Override
    public void setSystemId(String uri) {
        // no action: an attribute has the same base URI as its parent
    }

    @Override
    public String getSystemId() {
        NodeInfo parent = getParent();
        return parent == null ? null : getParent().getSystemId();
    }

    /**
     * Get the parent node
     */

    @Override
    public TinyNodeImpl getParent() {
        return tree.getNode(tree.attParent[nodeNr]);
    }

    /**
     * Get the root node of the tree (not necessarily a document node)
     *
     * @return the NodeInfo representing the root of this tree
     */

    @Override
    public NodeInfo getRoot() {
        NodeInfo parent = getParent();
        if (parent == null) {
            return this;    // doesn't happen - parentless attributes are represented by the Orphan class
        } else {
            return parent.getRoot();
        }
    }

    /**
     * Get the node sequence number (in document order). Sequence numbers are monotonic but not
     * consecutive. In this implementation, elements have a zero
     * least-significant word, while attributes and namespaces use the same value in the top word as
     * the containing element, and use the bottom word to hold
     * a sequence number, which numbers namespaces first and then attributes.
     */

    @Override
    protected long getSequenceNumber() {
        return getParent().getSequenceNumber()
                        + 0x8000 +
                        (nodeNr - tree.alpha[tree.attParent[nodeNr]]);
        // note the 0x8000 is to leave room for namespace nodes
    }

    /**
     * Return the type of node.
     *
     * @return Node.ATTRIBUTE
     */

    @Override
    public int getNodeKind() {
        return Type.ATTRIBUTE;
    }

    /**
     * Return the string value of the node.
     *
     * @return the attribute value
     */

    @Override
    public UnicodeString getUnicodeStringValue() {
        return StringView.of(tree.attValue[nodeNr]).tidy();
    }

    /**
     * Get the fingerprint of the node, used for matching names
     */

    @Override
    public int getFingerprint() {
        return tree.attCode[nodeNr] & 0xfffff;
    }

    /**
     * Get the name code of the node, used for finding names in the name pool
     *
     * @return the name code
     */

    public int getNameCode() {
        return tree.attCode[nodeNr];
    }

    /**
     * Get the prefix part of the name of this node. This is the name before the ":" if any.
     *
     * @return the prefix part of the name. For an unnamed node, return null.
     */

    @Override
    public String getPrefix() {
        int code = tree.attCode[nodeNr];
        if (!NamePool.isPrefixed(code)) {
            return "";
        }
        return tree.prefixPool.getPrefix(code >> 20);
    }

    /**
     * Get the display name of this node. For elements and attributes this is {@code [prefix:]localname}.
     * For unnamed nodes, it is an empty string.
     *
     * @return The display name of this node.
     *         For a node with no name, return an empty string.
     */

    @Override
    public String getDisplayName() {
        int code = tree.attCode[nodeNr];
        if (code < 0) {
            return "";
        }
        if (NamePool.isPrefixed(code)) {
            return getPrefix() + ":" + getLocalPart();
        } else {
            return getLocalPart();
        }
    }


    /**
     * Get the local name of this node.
     *
     * @return The local name of this node.
     *         For a node with no name, return an empty string.
     */

    @Override
    public String getLocalPart() {
        return tree.getNamePool().getLocalName(tree.attCode[nodeNr]);
    }

    /**
     * Get the URI part of the name of this node.
     *
     * @return The URI of the namespace of this node. For the default namespace, return an
     *         empty string
     */

    @Override
    public NamespaceUri getNamespaceUri() {
        int code = tree.attCode[nodeNr];
        if (!NamePool.isPrefixed(code)) {
            return NamespaceUri.NULL;
        }
        return tree.getNamePool().getURI(code);
    }

    @Override
    public boolean hasURI(NamespaceUri ns) {
        int code = tree.attCode[nodeNr];
        if (!NamePool.isPrefixed(code)) {
            return ns.isEmpty();
        }
        return getNamePool().getStructuredQName(code).hasURI(ns);
    }

    /**
     * Get the type annotation of this node, if any. The type annotation is represented as
     * SchemaType object.
     * 

Types derived from a DTD are not reflected in the result of this method.

* * @return For element and attribute nodes: the type annotation derived from schema * validation (defaulting to xs:untyped and xs:untypedAtomic in the absence of schema * validation). For comments, text nodes, processing instructions, and namespaces: null. * For document nodes, either xs:untyped if the document has not been validated, or * xs:anyType if it has. * @since 9.4 */ @Override public SchemaType getSchemaType() { if (tree.attType == null) { return BuiltInAtomicType.UNTYPED_ATOMIC; } return tree.getAttributeType(nodeNr); } /** * Get the typed value. * * @return the typed value. This will either be a single AtomicValue or a Value whose items are * atomic values. * @since 8.5 */ @Override public AtomicSequence atomize() throws XPathException { return tree.getTypedValueOfAttribute(this, nodeNr); } /** * Generate id. Returns key of owning element with the attribute nameCode as a suffix * * @param buffer Buffer to contain the generated ID value */ @Override public void generateId(StringBuilder buffer) { getParent().generateId(buffer); buffer.append("a"); buffer.append(tree.attCode[nodeNr]); // we previously used the attribute name. But this breaks the requirement // that the result of generate-id consists entirely of alphanumeric ASCII // characters } /** * Copy this node to a given {@code Receiver} */ @Override public void copy(/*@NotNull*/ Receiver out, int copyOptions, Location locationId) { throw new UnsupportedOperationException("copy() applied to attribute node"); } /** * Get the line number of the node within its source document entity */ @Override public int getLineNumber() { return getParent().getLineNumber(); } /** * Get the column number of the node within its source document entity */ @Override public int getColumnNumber() { return getParent().getColumnNumber(); } /** * Determine whether this node has the is-id property * * @return true if the node is an ID */ @Override public boolean isId() { return tree.isIdAttribute(nodeNr); } /** * Determine whether this node has the is-idref property * * @return true if the node is an IDREF or IDREFS element or attribute */ @Override public boolean isIdref() { return tree.isIdrefAttribute(nodeNr); } /** * Determine whether the node has the is-nilled property * * @return true if the node has the is-nilled property */ @Override public boolean isNilled() { return false; } /** * Ask whether the attribute results from expansion of attribute defaults * * @return true if this attribute resulted from expansion of default or fixed values defined * in a schema. Note that this property will only be set if both the configuration properties * {@link FeatureKeys#EXPAND_ATTRIBUTE_DEFAULTS} and {@link FeatureKeys#MARK_DEFAULTED_ATTRIBUTES} * are set. */ public boolean isDefaultedAttribute() { return tree.isDefaultedAttribute(nodeNr); } /** * The hashCode() method obeys the contract for hashCode(): that is, if two objects are equal * (represent the same node) then they must have the same hashCode() * * @since 8.7 Previously, the effect of the equals() and hashCode() methods was not defined. Callers * should therefore be aware that third party implementations of the NodeInfo interface may * not implement the correct semantics. */ public int hashCode() { return ((int) (tree.getDocumentNumber() & 0x3ff) << 20) ^ nodeNr ^ 7 << 17; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy