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

net.sf.saxon.style.XSLElement Maven / Gradle / Ivy

There is a newer version: 12.5
Show newest version
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2015 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.style;

import net.sf.saxon.expr.Expression;
import net.sf.saxon.expr.Literal;
import net.sf.saxon.expr.StringLiteral;
import net.sf.saxon.expr.instruct.*;
import net.sf.saxon.lib.StandardURIChecker;
import net.sf.saxon.lib.Validation;
import net.sf.saxon.om.*;
import net.sf.saxon.trans.Err;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.type.SchemaType;
import net.sf.saxon.value.Whitespace;


/**
 * An xsl:element element in the stylesheet. 
*/ public class XSLElement extends StyleElement { /*@Nullable*/ private Expression elementName; private Expression namespace = null; private String use; private StructuredQName[] attributeSets = null; private int validation; private SchemaType schemaType = null; private boolean inheritNamespaces = true; /** * Determine whether this node is an instruction. * * @return true - it is an instruction */ public boolean isInstruction() { return true; } /** * Determine whether this type of element is allowed to contain a template-body * * @return true: yes, it may contain a template-body */ public boolean mayContainSequenceConstructor() { return true; } public void prepareAttributes() throws XPathException { AttributeCollection atts = getAttributeList(); String nameAtt = null; String namespaceAtt = null; String validationAtt = null; String typeAtt = null; String inheritAtt = null; for (int a = 0; a < atts.getLength(); a++) { String f = atts.getQName(a); if (f.equals("name")) { nameAtt = Whitespace.trim(atts.getValue(a)); elementName = makeAttributeValueTemplate(nameAtt, a); } else if (f.equals("namespace")) { namespaceAtt = atts.getValue(a); namespace = makeAttributeValueTemplate(namespaceAtt, a); } else if (f.equals("validation")) { validationAtt = Whitespace.trim(atts.getValue(a)); } else if (f.equals("type")) { typeAtt = Whitespace.trim(atts.getValue(a)); } else if (f.equals("inherit-namespaces")) { inheritAtt = Whitespace.trim(atts.getValue(a)); } else if (f.equals("use-attribute-sets")) { use = atts.getValue(a); } else { checkUnknownAttribute(atts.getNodeName(a)); } } if (nameAtt == null) { reportAbsence("name"); } else { if (elementName instanceof StringLiteral) { if (!NameChecker.isQName(((StringLiteral) elementName).getStringValue())) { compileError("Element name " + Err.wrap(((StringLiteral) elementName).getStringValue(), Err.ELEMENT) + " is not a valid QName", "XTDE0820"); // to prevent duplicate error messages: elementName = new StringLiteral("saxon-error-element"); } } } if (namespaceAtt != null) { if (namespace instanceof StringLiteral) { if (!StandardURIChecker.getInstance().isValidURI(((StringLiteral) namespace).getStringValue())) { compileError("The value of the namespace attribute must be a valid URI", "XTDE0835"); } } } if (validationAtt != null) { validation = validateValidationAttribute(validationAtt); } else { validation = getDefaultValidation(); } if (typeAtt != null) { if (!isSchemaAware()) { compileError("The @type attribute is available only with a schema-aware XSLT processor", "XTSE1660"); } schemaType = getSchemaType(typeAtt); validation = Validation.BY_TYPE; } if (typeAtt != null && validationAtt != null) { compileError("The @validation and @type attributes are mutually exclusive", "XTSE1505"); } if (inheritAtt != null) { inheritNamespaces = processBooleanAttribute("inherit-namespaces", inheritAtt); } } public void validate(ComponentDeclaration decl) throws XPathException { if (use != null) { // get the names of referenced attribute sets attributeSets = getUsedAttributeSets(use); } elementName = typeCheck("name", elementName); namespace = typeCheck("namespace", namespace); } public Expression compile(Compilation exec, ComponentDeclaration decl) throws XPathException { // deal specially with the case where the element name is known statically if (elementName instanceof StringLiteral) { CharSequence qName = ((StringLiteral) elementName).getStringValue(); String[] parts; try { parts = NameChecker.getQNameParts(qName); } catch (QNameException e) { compileErrorInAttribute("Invalid element name: " + qName, "XTDE0820", "name"); return null; } String nsuri = null; if (namespace instanceof StringLiteral) { nsuri = ((StringLiteral) namespace).getStringValue(); if (nsuri.isEmpty()) { parts[0] = ""; } } else if (namespace == null) { nsuri = getURIForPrefix(parts[0], true); if (nsuri == null) { undeclaredNamespaceError(parts[0], "XTDE0830", "name"); } } if (nsuri != null) { // Local name and namespace are both known statically: generate a FixedElement instruction FingerprintedQName qn = new FingerprintedQName(parts[0], nsuri, parts[1]); qn.allocateNameCode(getNamePool()); FixedElement inst = new FixedElement(qn, NamespaceBinding.EMPTY_ARRAY, inheritNamespaces, true, schemaType, validation); return compileContentExpression(exec, decl, inst); } } ComputedElement inst = new ComputedElement(elementName, namespace, schemaType, validation, inheritNamespaces, false); return compileContentExpression(exec, decl, inst); } private Expression compileContentExpression(Compilation exec, ComponentDeclaration decl, ElementCreator inst) throws XPathException { Expression content = compileSequenceConstructor(exec, decl, true); if (attributeSets != null) { Expression use = UseAttributeSet.makeUseAttributeSets(attributeSets, this); if (content == null) { content = use; } else { content = Block.makeBlock(use, content); content.setLocation(allocateLocation()); } } if (content == null) { content = Literal.makeEmptySequence(); } inst.setContentExpression(content); return inst; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy