/*
* eXist Open Source Native XML Database
* Copyright (C) 2008-2010 The eXist Project
* http://exist-db.org
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
* $Id$
*/
package org.exist.xslt.expression;
import javax.xml.stream.XMLStreamException;
import org.exist.interpreter.ContextAtExist;
import org.exist.memtree.DocumentBuilderReceiver;
import org.exist.memtree.MemTreeBuilder;
import org.exist.xquery.AnalyzeContextInfo;
import org.exist.xquery.Dependency;
import org.exist.xquery.Profiler;
import org.exist.xquery.XPathException;
import org.exist.xquery.util.ExpressionDumper;
import org.exist.xquery.value.Item;
import org.exist.xquery.value.Sequence;
import org.exist.xquery.value.SequenceIterator;
import org.exist.xquery.value.Type;
import org.exist.xslt.XSLContext;
import org.exist.xslt.pattern.Pattern;
import org.w3c.dom.Attr;
import org.xml.sax.SAXException;
/**
* <!-- Category: instruction -->
* <xsl:value-of
* select? = expression
* separator? = { string }
* [disable-output-escaping]? = "yes" | "no">
* <!-- Content: sequence-constructor -->
* </xsl:value-of>
*
* @author <a href="mailto:shabanovd@gmail.com">Dmitriy Shabanov</a>
*
*/
public class ValueOf extends SimpleConstructor {
private String attr_select = null;
private XSLPathExpr select = null;
private String separator = null;
private Boolean disable_output_escaping = null;
protected boolean sequenceItSelf = false;
public ValueOf(XSLContext context) {
super(context);
}
public void setToDefaults() {
attr_select = null;
select = null;
//the default separator is a single space (#x20) when the content is specified using the select attribute,
//or a zero-length string when the content is specified using a sequence constructor.
separator = null;
disable_output_escaping = null;
}
public void prepareAttribute(ContextAtExist context, Attr attr) throws XPathException {
String attr_name = attr.getNodeName();
if (attr_name.equals(SELECT)) {
attr_select = attr.getValue();
} else if (attr_name.equals(SEPARATOR)) {
separator = attr.getValue();
} else if (attr_name.equals(DISABLE_OUTPUT_ESCAPING)) {
disable_output_escaping = getBoolean(attr.getValue());
}
}
public void analyze(AnalyzeContextInfo contextInfo) throws XPathException {
boolean atRootCall = false;//XXX: rewrite
if (attr_select != null) {
select = new XSLPathExpr(getXSLContext());
Pattern.parse(contextInfo.getContext(), attr_select, select);
//UNDERSTAND: <node>text<node> step = "." -> SELF:node(), but need CHILD:node()
// if ((contextInfo.getFlags() & DOT_TEST) != 0) {
// atRootCall = true;
// _check_(select);
// contextInfo.removeFlag(DOT_TEST);
// }
//
// _check_childNodes_(select);
}
super.analyze(contextInfo);
if (atRootCall)
contextInfo.addFlag(DOT_TEST);
}
// protected void _check_(Expression path) {
// for (int pos = 0; pos < path.getLength(); pos++) {
// Expression expr = path.getExpression(pos);
// if (expr instanceof RootNode) {
// expr = new LocationStep(getContext(), Constants.CHILD_AXIS, new AnyNodeTest());
// path.replaceExpression(pos, expr);
// continue;
// }
// if ((pos == 0) && (expr instanceof LocationStep)) {
// LocationStep location = (LocationStep) expr;
// if (location.getAxis() == Constants.SELF_AXIS) {
// location.setAxis(Constants.CHILD_AXIS);
// }
// } else {
// _check_(expr);
// }
// }
// if (path.getLength() != 0) {
// Expression expr = path.getExpression(path.getLength()-1);
// if (expr instanceof LocationStep) {
// LocationStep location = (LocationStep) expr;
// //TODO: rewrite
// if (!"node()".equals(location.getTest().toString())) {
// ((PathExpr)path).add(new LocationStep(getContext(), Constants.CHILD_AXIS, new AnyNodeTest()));
// }
// }
// }
// }
public Sequence eval(Sequence contextSequence, Item contextItem) throws XPathException {
if (context.getProfiler().isEnabled()) {
context.getProfiler().start(this);
context.getProfiler().message(this, Profiler.DEPENDENCIES, "DEPENDENCIES", Dependency.getDependenciesName(this.getDependencies()));
if (contextSequence != null)
context.getProfiler().message(this, Profiler.START_SEQUENCES, "CONTEXT SEQUENCE", contextSequence);
if (contextItem != null)
context.getProfiler().message(this, Profiler.START_SEQUENCES, "CONTEXT ITEM", contextItem.toSequence());
}
// if (contextItem != null) {
// contextSequence = contextItem.toSequence();
// }
// evaluate the expression
context.pushDocumentContext();
Sequence result;
try {
// System.out.println("=================================================================");
// System.out.println("select = "+select);
// System.out.println("contextSequence = "+contextSequence);
// System.out.println("contextItem = "+contextItem);
result = select.eval(contextSequence, contextItem);
if (sequenceItSelf)
return result;
} finally {
context.popDocumentContext();
}
// create the output
MemTreeBuilder builder = context.getDocumentBuilder();
DocumentBuilderReceiver receiver = new DocumentBuilderReceiver(builder);
try {
SequenceIterator i = result.iterate();
Item next = i.nextItem();
StringBuilder buf = null;
boolean allowAttribs = true;
while (next != null) {
context.proceed(this, builder);
// if item is an atomic value, collect the string values of all
// following atomic values and seperate them by a space.
if (Type.subTypeOf(next.getType(), Type.ATOMIC)) {
if(buf == null)
buf = new StringBuilder();
else if (buf.length() > 0)
buf.append(' ');
buf.append(next.getStringValue());
allowAttribs = false;
next = i.nextItem(); // if item is a node, flush any collected character data and
// copy the node to the target doc.
} else if (Type.subTypeOf(next.getType(), Type.NODE)) {
if (buf != null && buf.length() > 0) {
receiver.characters(buf);
buf.setLength(0);
}
if (next.getType() == Type.ATTRIBUTE && !allowAttribs)
throw new XPathException(this, "XQTY0024: An attribute may not appear after " +
"another child node.");
next.copyTo(context.getBroker(), receiver);
allowAttribs = next.getType() == Type.ATTRIBUTE;
next = i.nextItem();
}
}
// flush remaining character data
if (buf != null && buf.length() > 0)
receiver.characters(buf);
} catch (SAXException e) {
LOG.warn("SAXException during serialization: " + e.getMessage(), e);
throw new XPathException(this, e.getMessage());
//throw new XPathException(getASTNode(),
// "Encountered SAX exception while serializing enclosed expression: "
// + ExpressionDumper.dump(this));
}
if (context.getProfiler().isEnabled())
context.getProfiler().end(this, "", result);
return result;
}
/* (non-Javadoc)
* @see org.exist.xquery.Expression#dump(org.exist.xquery.util.ExpressionDumper)
*/
public void dump(ExpressionDumper dumper) {
dumper.display("<xsl:value-of");
if (select != null) {
dumper.display(" select = ");
select.dump(dumper);
}
if (separator != null)
dumper.display(" separator = "+separator);
if (disable_output_escaping != null)
dumper.display(" disable-output-escaping = "+disable_output_escaping);
dumper.display("> ");
super.dump(dumper);
dumper.display("</xsl:value-of>");
}
public String toString() {
StringBuilder result = new StringBuilder();
result.append("<xsl:value-of");
if (select != null)
result.append(" select = "+select.toString());
if (separator != null)
result.append(" separator = "+separator.toString());
if (disable_output_escaping != null)
result.append(" disable-output-escaping = "+disable_output_escaping);
result.append("> ");
result.append(super.toString());
result.append("</xsl:value-of>");
return result.toString();
}
/**
* @deprecated Use {@link #process(XSLContext,SequenceIterator)} instead
*/
public void process(SequenceIterator sequenceIterator, XSLContext context) {
process(context, sequenceIterator);
}
public void process(XSLContext context, SequenceIterator sequenceIterator) {
try {
Sequence result = select.eval(null, null);
context.getResultWriter().writeCharacters(result.getStringValue());
} catch (XPathException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (XMLStreamException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}