Package org.exist.xslt.expression

Source Code of org.exist.xslt.expression.CopyOf

/*
*  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 org.exist.interpreter.ContextAtExist;
import org.exist.memtree.DocumentBuilderReceiver;
import org.exist.memtree.MemTreeBuilder;
import org.exist.xquery.AnalyzeContextInfo;
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:copy-of
*   select = expression
*   copy-namespaces? = "yes" | "no"
*   type? = qname
*   validation? = "strict" | "lax" | "preserve" | "strip" />
*
* @author <a href="mailto:shabanovd@gmail.com">Dmitriy Shabanov</a>
*
*/
public class CopyOf extends Declaration {

    private String attr_select = null;
  private XSLPathExpr select = null;

  private Boolean copy_namespaces = null;
    private String type = null;
    private String validation = null;

  protected boolean sequenceItSelf = false;

  public CopyOf(XSLContext context) {
    super(context);
  }

  public void setToDefaults() {
    attr_select = null;
    select = null;
   
    copy_namespaces = null;
      type = null;
      validation = null;
  }

  public void prepareAttribute(ContextAtExist context, Attr attr) throws XPathException {
    String attr_name = attr.getLocalName();
     
    if (attr_name.equals(SELECT)) {
      attr_select = attr.getValue();
    } else if (attr_name.equals(COPY_NAMESPACES)) {
      copy_namespaces = getBoolean(attr.getValue());
    } else if (attr_name.equals(TYPE)) {
      type = attr.getValue();
    } else if (attr_name.equals(VALIDATION)) {
      validation = 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);
    }

    public Sequence eval(Sequence contextSequence, Item contextItem) throws XPathException {
    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:copy-of");

        if (attr_select != null) {
          dumper.display(" select = ");
          dumper.display(attr_select);
        }
        if (copy_namespaces != null) {
          dumper.display(" copy_namespaces = ");
          dumper.display(copy_namespaces);
        }
        if (type != null) {
          dumper.display(" type = ");
          dumper.display(type);
        }
        if (validation != null) {
          dumper.display(" validation = ");
          dumper.display(validation);
        }

        super.dump(dumper);

        dumper.display("</xsl:copy-of>");
    }
   
    public String toString() {
      StringBuilder result = new StringBuilder();
      result.append("<xsl:copy-of");
       
      if (attr_select != null)
          result.append(" select = "+attr_select.toString());   
      if (copy_namespaces != null)
          result.append(" copy_namespaces = "+copy_namespaces.toString());   
      if (type != null)
          result.append(" type = "+type.toString());   
      if (validation != null)
          result.append(" validation = "+validation.toString());   

        result.append("> ");
       
        result.append(super.toString());

        result.append("</xsl:copy-of> ");
        return result.toString();
    }   
}
TOP

Related Classes of org.exist.xslt.expression.CopyOf

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.