blob: 0f55e4f00b843a9c4a7b4b735773c8b3ce529331 [file] [log] [blame]
<?xml version="1.0"?>
<!--
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You 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.
-->
<!--
- XSP Core logicsheet for the Java language
-
- @author <a href="mailto:ricardo@apache.org>Ricardo Rocha</a>
- @author <a href="sylvain.wallez@anyware-tech.com">Sylvain Wallez</a>
- @version CVS $Id$
-->
<xsl:stylesheet version="1.0"
xmlns:xsp="http://apache.org/xsp"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:XSLTExtension="org.apache.cocoon.components.language.markup.xsp.XSLTExtension">
<xsl:output method="text"/>
<!--
Set to "strip" to strip empty space like XSLT.
Else space is preserved to remain compatible.
-->
<xsl:variable name="space" select="/xsp:page/@space"/>
<xsl:variable name="xsp-uri" select="'http://apache.org/xsp'"/>
<!--
this variable holds the instance of extension class to properly
escape text into Java strings
ovidiu: use the class name as the namespace to identify the
class. This is supposedly portable across XSLT implementations.
-->
<xsl:variable
name="extension"
select="XSLTExtension:new()"
xmlns:XSLTExtension="org.apache.cocoon.components.language.markup.xsp.XSLTExtension"/>
<xsl:template match="/">
<code xml:space="preserve">
<xsl:apply-templates select="xsp:page"/>
</code>
</xsl:template>
<xsl:template match="xsp:page">
/*
* Generated by XSP. Edit at your own risk, :-)
*/
<xsl:if test="$space = 'strip'">
/*
* Whitespace introduced in the XSP will not propagate into
* the resulting XML document!
* To retain all whitespace, remove the "space" attribute from
* the root &lt;xsp:page&gt; element.
*/
</xsl:if>
package <xsl:value-of select="translate(@file-path, '/', '.')"/>;
import java.io.File;
import java.io.IOException;
import java.io.StringReader;
import java.util.Date;
import java.util.List;
import java.util.Stack;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.AttributesImpl;
import org.apache.avalon.framework.component.Component;
import org.apache.avalon.framework.component.ComponentException;
import org.apache.avalon.framework.component.ComponentManager;
import org.apache.avalon.framework.component.ComponentSelector;
import org.apache.avalon.framework.context.Context;
import org.apache.cocoon.Constants;
import org.apache.cocoon.ProcessingException;
import org.apache.cocoon.generation.Generator;
import org.apache.cocoon.components.language.markup.xsp.XSPGenerator;
import org.apache.cocoon.components.language.markup.xsp.XSPObjectHelper;
import org.apache.cocoon.components.language.markup.xsp.XSPRequestHelper;
import org.apache.cocoon.components.language.markup.xsp.XSPResponseHelper;
import org.apache.cocoon.components.language.markup.xsp.XSPSessionHelper;
/* User Imports */
<xsl:for-each select="xsp:structure/xsp:include">
import <xsl:value-of select="."/>;
</xsl:for-each>
public class <xsl:value-of select="@file-name"/> extends XSPGenerator {
// Files this XSP depends on
private static File[] _dependentFiles = new File[] {
<xsl:for-each select="//xsp:dependency">
new File("<xsl:value-of select="translate(., '\','/')"/>"),
</xsl:for-each>
};
// Initialize attributes used by modifiedSince() (see AbstractServerPage)
{
this.dateCreated = <xsl:value-of select="@creation-date"/>L;
this.dependencies = _dependentFiles;
}
// Internally used list of attributes for SAX events. Being on
// class scope allows xsp:logic to define markup generating methods.
private final AttributesImpl _xspAttr = new AttributesImpl();
private void _startElem(String namespace, String localname, String name) throws SAXException {
this.contentHandler.startElement(namespace, localname, name, _xspAttr);
_xspAttr.clear();
}
private void _endElem(String namespace, String localname, String name) throws SAXException {
this.contentHandler.endElement(namespace, localname, name);
}
/* Built-in parameters available for use */
// context - org.apache.cocoon.environment.Context
// request - org.apache.cocoon.environment.Request
// response - org.apache.cocoon.environment.Response
// parameters - parameters defined in the sitemap
// objectModel- java.util.Map
// resolver - org.apache.cocoon.environment.SourceResolver
/* User Class Declarations */
<xsl:apply-templates select="xsp:logic"/>
/**
* Generate XML data.
*/
public void generate() throws SAXException, IOException, ProcessingException {
<!-- Locally scoped xspAttr for backwards compatibility to pre-2.1.9 logicsheets. -->
final AttributesImpl xspAttr = _xspAttr;
<!-- Do any user-defined necessary initializations -->
<xsl:for-each select="xsp:init-page">
<xsl:value-of select="XSLTExtension:escape($extension,.)"/>
</xsl:for-each>
this.contentHandler.startDocument();
xspAttr.clear();
<!-- Generate top-level processing instructions -->
<xsl:apply-templates select="/processing-instruction()"/>
<!-- Process only 1st non-XSP element as generated root -->
<xsl:call-template name="process-first-element">
<xsl:with-param name="content" select="*[not(namespace-uri(.) = $xsp-uri)][1]"/>
</xsl:call-template>
this.contentHandler.endDocument();
<!-- Do any user-defined necessary clean-ups -->
<xsl:for-each select="xsp:exit-page">
<xsl:value-of select="XSLTExtension:escape($extension,.)"/>
</xsl:for-each>
}
}
</xsl:template>
<xsl:template name="process-first-element">
<xsl:param name="content"/>
<!-- Generate top-level namespaces declarations -->
<xsl:variable name="parent-element" select="$content/.."/>
<xsl:for-each select="$content/namespace::*">
<xsl:variable name="ns-prefix" select="local-name(.)"/>
<xsl:variable name="ns-uri" select="string(.)"/>
<!-- Declare namespaces that also exist on the parent (i.e. not locally declared),
and filter out "xmlns:xmlns" namespace produced by Xerces+Saxon
and suppress xmlns:xsp which no longer occurs in output. -->
<xsl:if test="($ns-prefix != 'xmlns') and ($ns-uri != $xsp-uri) and $parent-element/namespace::*[local-name(.) = $ns-prefix and string(.) = $ns-uri]">
this.contentHandler.startPrefixMapping(
"<xsl:value-of select="$ns-prefix"/>",
"<xsl:value-of select="$ns-uri"/>"
);
</xsl:if>
</xsl:for-each>
<!-- Generate content -->
<xsl:apply-templates select="$content"/>
<!-- Close top-level namespaces declarations-->
<xsl:for-each select="$content/namespace::*">
<xsl:variable name="ns-prefix" select="local-name(.)"/>
<xsl:variable name="ns-uri" select="string(.)"/>
<xsl:if test="($ns-prefix != 'xmlns') and ($ns-uri != $xsp-uri) and $parent-element/namespace::*[local-name(.) = $ns-prefix and string(.) = $ns-uri]">
this.contentHandler.endPrefixMapping(
"<xsl:value-of select="local-name(.)"/>"
);
</xsl:if>
</xsl:for-each>
</xsl:template>
<xsl:template match="xsp:element">
<xsl:variable name="uri">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">uri</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:variable name="prefix">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">prefix</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:if test="@name and contains(@name, ':')">
<xsl:call-template name="error">
<xsl:with-param name="message">[&lt;xsp:element name="<xsl:value-of select="@name"/>"&gt;]
Name can not contain ':'. If you want to create namespaced element, specify 'uri' and 'prefix'.
</xsl:with-param>
</xsl:call-template>
</xsl:if>
<xsl:variable name="name">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">name</xsl:with-param>
<xsl:with-param name="required">true</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:variable name="raw-name">
<xsl:if test="
($uri = '&quot;&quot;' and $prefix != '&quot;&quot;') or
($uri != '&quot;&quot;' and $prefix = '&quot;&quot;')
">
<xsl:call-template name="error">
<xsl:with-param name="message">[&lt;xsp:element&gt;]
Either both 'uri' and 'prefix' or none of them must be specified
</xsl:with-param>
</xsl:call-template>
</xsl:if>
<xsl:choose>
<xsl:when test="$prefix = '&quot;&quot;'">
<xsl:copy-of select="$name"/>
</xsl:when>
<xsl:otherwise>
(<xsl:copy-of select="$prefix"/>)+":"+(<xsl:copy-of select="$name"/>)
</xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:for-each select="namespace::*">
<xsl:variable name="ns-prefix" select="local-name(.)"/>
<xsl:variable name="ns-uri" select="string(.)"/>
<xsl:if test="not(../namespace::*[local-name(.) = $ns-prefix and string(.) = $ns-uri])">
this.contentHandler.startPrefixMapping(
"<xsl:value-of select="local-name(.)"/>",
"<xsl:value-of select="."/>");
</xsl:if>
</xsl:for-each>
<!-- Declare namespace defined by @uri and @prefix attribute -->
<xsl:if test="$uri != '&quot;&quot;'">
<xsl:if test="not(../namespace::*[local-name(.) = $prefix and string(.) = $uri])">
this.contentHandler.startPrefixMapping(
<xsl:value-of select="$prefix"/>,
<xsl:value-of select="$uri"/>);
</xsl:if>
</xsl:if>
<xsl:apply-templates select="xsp:attribute | xsp:logic[xsp:attribute]" mode="check"/>
this.contentHandler.startElement(
<xsl:copy-of select="$uri"/>,
<xsl:copy-of select="$name"/>,
<xsl:copy-of select="$raw-name"/>,
_xspAttr
);
_xspAttr.clear();
<xsl:apply-templates select="node()[not(
(namespace-uri(.) = $xsp-uri and local-name(.) = 'attribute') or
(namespace-uri(.) = $xsp-uri and local-name(.) = 'logic' and ./xsp:attribute)
)]"/>
this.contentHandler.endElement(
<xsl:copy-of select="$uri"/>,
<xsl:copy-of select="$name"/>,
<xsl:copy-of select="$raw-name"/>);
<!-- Declare namespace defined by @uri and @prefix attribute -->
<xsl:if test="$uri != '&quot;&quot;'">
<xsl:if test="not(../namespace::*[local-name(.) = $prefix and string(.) = $uri])">
this.contentHandler.endPrefixMapping(<xsl:value-of select="$prefix"/>);
</xsl:if>
</xsl:if>
<xsl:for-each select="namespace::*">
<xsl:variable name="ns-prefix" select="local-name(.)"/>
<xsl:variable name="ns-uri" select="string(.)"/>
<xsl:if test="not(../namespace::*[local-name(.) = $ns-prefix and string(.) = $ns-uri])">
this.contentHandler.endPrefixMapping("<xsl:value-of select="local-name(.)"/>");
</xsl:if>
</xsl:for-each>
</xsl:template>
<xsl:template match="xsp:attribute | xsp:logic[xsp:attribute]" mode="check">
<xsl:if test="preceding-sibling::text()[normalize-space()]
| preceding-sibling::processing-instruction()
| preceding-sibling::*[not(self::xsp:attribute
| self::xsp:logic[xsp:attribute]
| self::xsp:param
| self::xsp:text[not(normalize-space())])]">
<xsl:message terminate="yes">
You can not add attributes to elements after other node types already have been added.
</xsl:message>
</xsl:if>
<xsl:apply-templates select="."/>
</xsl:template>
<xsl:template match="xsp:attribute">
<xsl:variable name="uri">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">uri</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:variable name="prefix">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">prefix</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:variable name="name">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">name</xsl:with-param>
<xsl:with-param name="required">true</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:variable name="raw-name">
<xsl:if test="
($uri = '&quot;&quot;' and $prefix != '&quot;&quot;') or
($uri != '&quot;&quot;' and $prefix = '&quot;&quot;')
">
<xsl:call-template name="error">
<xsl:with-param name="message">[&lt;xsp:attribute&gt;]
Either both 'uri' and 'prefix' or none of them must be specified
</xsl:with-param>
</xsl:call-template>
</xsl:if>
<xsl:choose>
<xsl:when test="$prefix = '&quot;&quot;'">
<xsl:copy-of select="$name"/>
</xsl:when>
<xsl:otherwise> (<xsl:copy-of select="$prefix"/> + ":" + <xsl:copy-of select="$name"/>) </xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:variable name="content">
<xsl:for-each select="text()|xsp:expr|xsp:text">
<xsl:if test="position() &gt; 1"> + </xsl:if>
<xsl:choose>
<xsl:when test="namespace-uri(.) = $xsp-uri and local-name(.) = 'expr'">
String.valueOf(<xsl:value-of select="XSLTExtension:escape($extension,.)"/>)
</xsl:when>
<xsl:otherwise> "<xsl:value-of select="XSLTExtension:escapeJava($extension,.)"/>" </xsl:otherwise>
</xsl:choose>
</xsl:for-each>
<xsl:if test="not(text()|xsp:expr|xsp:text)"> "" </xsl:if>
</xsl:variable>
_xspAttr.addAttribute(
<xsl:copy-of select="$uri"/>,
<xsl:copy-of select="$name"/>,
<xsl:copy-of select="$raw-name"/>,
"CDATA",
<xsl:value-of select="normalize-space($content)"/>
);
</xsl:template>
<xsl:template match="xsp:expr">
<xsl:choose>
<xsl:when test="namespace-uri(..) = $xsp-uri and local-name(..) != 'content' and local-name(..) != 'element'">
<!--
Expression is nested inside another XSP tag:
preserve its Java type
-->
(<xsl:value-of select="XSLTExtension:escape($extension,.)"/>)
</xsl:when>
<xsl:when test="string-length(.) = 0">
<!-- Do nothing -->
</xsl:when>
<xsl:otherwise>
<!-- Output the value as elements or character data depending on its type -->
XSPObjectHelper.xspExpr(contentHandler, <xsl:value-of select="XSLTExtension:escape($extension,.)"/>);
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<!-- FIXME: Is this _really_ necessary? -->
<xsl:template match="xsp:content">
<xsl:apply-templates/>
</xsl:template>
<xsl:template match="xsp:logic">
<xsl:apply-templates/>
</xsl:template>
<xsl:template match="xsp:pi">
<xsl:variable name="target">
<xsl:call-template name="get-parameter">
<xsl:with-param name="name">target</xsl:with-param>
</xsl:call-template>
</xsl:variable>
<xsl:variable name="content">
<xsl:for-each select="text()|xsp:expr">
<xsl:choose>
<xsl:when test="namespace-uri(.) = $xsp-uri and local-name(.) = 'expr'">
String.valueOf(<xsl:value-of select="."/>)
</xsl:when>
<xsl:otherwise> "<xsl:value-of select="."/>" </xsl:otherwise>
</xsl:choose>
<xsl:text> + </xsl:text>
</xsl:for-each>
<xsl:text> "" </xsl:text>
</xsl:variable>
this.contentHandler.processingInstruction(
<xsl:copy-of select="$target"/>,
<xsl:value-of select="normalize-space($content)"/>
);
</xsl:template>
<!-- FIXME: How to create comments in SAX? -->
<xsl:template match="xsp:comment">
this.comment("<xsl:value-of select="."/>");
</xsl:template>
<!-- $xsp-uri = 'http://apache.org/xsp' -->
<xsl:template match="*[not(namespace-uri(.) = 'http://apache.org/xsp')]">
<xsl:variable name="parent-element" select=".."/>
<xsl:for-each select="namespace::*">
<xsl:variable name="ns-prefix" select="local-name(.)"/>
<xsl:variable name="ns-uri" select="string(.)"/>
<xsl:if test="not($parent-element/namespace::*[local-name(.) = $ns-prefix and string(.) = $ns-uri])">
this.contentHandler.startPrefixMapping(
"<xsl:value-of select="local-name(.)"/>",
"<xsl:value-of select="."/>"
);
</xsl:if>
</xsl:for-each>
<xsl:apply-templates select="@*"/>
<xsl:apply-templates select="xsp:attribute | xsp:logic[xsp:attribute]"/>
_startElem(
"<xsl:value-of select="namespace-uri(.)"/>",
"<xsl:value-of select="local-name(.)"/>",
"<xsl:value-of select="name(.)"/>"
);
<xsl:apply-templates select="node()[not(
(namespace-uri(.) = $xsp-uri and local-name(.) = 'attribute') or
(namespace-uri(.) = $xsp-uri and local-name(.) = 'logic' and ./xsp:attribute)
)]"/>
_endElem(
"<xsl:value-of select="namespace-uri(.)"/>",
"<xsl:value-of select="local-name(.)"/>",
"<xsl:value-of select="name(.)"/>"
);
<xsl:for-each select="namespace::*">
<xsl:variable name="ns-prefix" select="local-name(.)"/>
<xsl:variable name="ns-uri" select="string(.)"/>
<xsl:if test="not($parent-element/namespace::*[local-name(.) = $ns-prefix and string(.) = $ns-uri])">
this.contentHandler.endPrefixMapping(
"<xsl:value-of select="local-name(.)"/>"
);
</xsl:if>
</xsl:for-each>
</xsl:template>
<xsl:template match="@*">
<!-- Filter out namespace declaration attributes -->
<xsl:if test="not(starts-with(name(.), 'xmlns:'))">
_xspAttr.addAttribute(
"<xsl:value-of select="namespace-uri(.)"/>",
"<xsl:value-of select="local-name(.)"/>",
"<xsl:value-of select="name(.)"/>",
"CDATA",
"<xsl:value-of select="XSLTExtension:escapeJava($extension,.)"/>"
);
</xsl:if>
</xsl:template>
<xsl:template match="text()">
<xsl:choose>
<xsl:when test="namespace-uri(..) = $xsp-uri and (local-name(..) = 'logic' or local-name(..) = 'expr')">
<xsl:value-of select="XSLTExtension:escape($extension, .)"/>
</xsl:when>
<xsl:when test="$space = 'strip'">
<xsl:variable name="txt" select="normalize-space(.)"/>
<xsl:if test="$txt != '' and $txt != ' '">
this.characters("<xsl:value-of select="XSLTExtension:escapeJava($extension, $txt)"/>");
</xsl:if>
</xsl:when>
<xsl:otherwise>
this.characters("<xsl:value-of select="XSLTExtension:escapeJava($extension, .)"/>");
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template match="processing-instruction()">
this.contentHandler.processingInstruction(
"<xsl:value-of select="name()"/>",
"<xsl:value-of select="."/>"
);
</xsl:template>
<!-- Utility templates -->
<xsl:template name="get-parameter">
<xsl:param name="name"/>
<xsl:param name="default"/>
<xsl:param name="required">false</xsl:param>
<xsl:choose>
<xsl:when test="@*[name(.) = $name]">"<xsl:value-of select="@*[name(.) = $name]"/>"</xsl:when>
<xsl:when test="xsp:param[@name = $name]">
<xsl:call-template name="get-parameter-content">
<xsl:with-param name="content"
select="(*[namespace-uri(.)=$xsp-uri and local-name(.) = 'param'])[@name = $name]"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:choose>
<xsl:when test="string-length($default) = 0">
<xsl:choose>
<xsl:when test="$required = 'true'">
<xsl:call-template name="error">
<xsl:with-param name="message">[Logicsheet processor]
Parameter '<xsl:value-of select="$name"/>' missing in dynamic tag &lt;<xsl:value-of select="name(.)"/>&gt;
</xsl:with-param>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>""</xsl:otherwise>
</xsl:choose>
</xsl:when>
<xsl:otherwise><xsl:copy-of select="$default"/></xsl:otherwise>
</xsl:choose>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template name="get-parameter-content">
<xsl:param name="content"/>
<xsl:choose>
<xsl:when test="$content/*[namespace-uri(.)=$xsp-uri and local-name(.) != 'text']">
<xsl:apply-templates select="$content/*[namespace-uri(.)=$xsp-uri and local-name(.) != 'text']"/>
</xsl:when>
<xsl:otherwise>"<xsl:value-of select="$content"/>"</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template name="get-nested-content">
<xsl:param name="content"/>
<xsl:choose>
<xsl:when test="$content/*">
<xsl:apply-templates select="$content/*"/>
</xsl:when>
<xsl:otherwise>"<xsl:value-of select="$content"/>"</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template name="error">
<xsl:param name="message"/>
<xsl:message terminate="yes"><xsl:value-of select="$message"/></xsl:message>
</xsl:template>
<!-- Ignored elements -->
<xsl:template match="xsp:logicsheet|xsp:dependency|xsp:param"/>
</xsl:stylesheet>