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

net.sf.saxon.style.XSLPerformSort 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.sort.SortExpression;
import net.sf.saxon.expr.sort.SortKeyDefinitionList;
import net.sf.saxon.om.AttributeCollection;
import net.sf.saxon.om.AxisInfo;
import net.sf.saxon.om.NodeInfo;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.tree.iter.AxisIterator;
import net.sf.saxon.type.Type;
import net.sf.saxon.value.Whitespace;


/**
 * Handler for xsl:perform-sort elements in stylesheet (XSLT 2.0). 
*/ public class XSLPerformSort extends StyleElement { /*@Nullable*/ Expression select = null; /** * 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; } /** * Specify that xsl:sort is a permitted child */ protected boolean isPermittedChild(StyleElement child) { return (child instanceof XSLSort); } public void prepareAttributes() throws XPathException { AttributeCollection atts = getAttributeList(); String selectAtt = null; for (int a = 0; a < atts.getLength(); a++) { String f = atts.getQName(a); if (f.equals("select")) { selectAtt = atts.getValue(a); select = makeExpression(selectAtt, a); } else { checkUnknownAttribute(atts.getNodeName(a)); } } } public void validate(ComponentDeclaration decl) throws XPathException { checkSortComesFirst(true); if (select != null) { // if there is a select attribute, check that there are no children other than xsl:sort and xsl:fallback AxisIterator kids = iterateAxis(AxisInfo.CHILD); while (true) { NodeInfo child = (NodeInfo) kids.next(); if (child == null) { break; } if (child instanceof XSLSort || child instanceof XSLFallback) { // no action } else if (child.getNodeKind() == Type.TEXT && !Whitespace.isWhite(child.getStringValueCS())) { // with xml:space=preserve, white space nodes may still be there compileError("Within xsl:perform-sort, significant text must not appear if there is a select attribute", "XTSE1040"); } else { ((StyleElement) child).compileError( "Within xsl:perform-sort, child instructions are not allowed if there is a select attribute", "XTSE1040"); } } } select = typeCheck("select", select); } public Expression compile(Compilation compilation, ComponentDeclaration decl) throws XPathException { SortKeyDefinitionList sortKeys = makeSortKeys(compilation, decl); if (select != null) { return new SortExpression(select, sortKeys); } else { Expression body = compileSequenceConstructor(compilation, decl, true); if (body == null) { body = Literal.makeEmptySequence(); } try { return new SortExpression(body.simplify(), sortKeys); } catch (XPathException e) { compileError(e); return null; } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy