/*
 * Copyright (c) 2005, 2012, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code 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 General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package com.sun.xml.internal.txw2.output;

import com.sun.xml.internal.txw2.TxwException;
import org.xml.sax.ContentHandler;
import org.xml.sax.SAXException;
import org.xml.sax.ext.LexicalHandler;
import org.xml.sax.helpers.AttributesImpl;

import javax.xml.transform.sax.SAXResult;
import java.util.Stack;

Author:Ryan.Shoemaker@Sun.COM
/** * {@link XmlSerializer} for {@link SAXResult} and {@link ContentHandler}. * * @author Ryan.Shoemaker@Sun.COM */
public class SaxSerializer implements XmlSerializer { private final ContentHandler writer; private final LexicalHandler lexical; public SaxSerializer(ContentHandler handler) { this(handler,null,true); }
Creates an XmlSerializer that writes SAX events.

Sepcifying a non-null LexicalHandler allows applications to write comments and CDATA sections.

/** * Creates an {@link XmlSerializer} that writes SAX events. * * <p> * Sepcifying a non-null {@link LexicalHandler} allows applications * to write comments and CDATA sections. */
public SaxSerializer(ContentHandler handler,LexicalHandler lex) { this(handler, lex, true); } public SaxSerializer(ContentHandler handler,LexicalHandler lex, boolean indenting) { if(!indenting) { writer = handler; lexical = lex; } else { IndentingXMLFilter indenter = new IndentingXMLFilter(handler, lex); writer = indenter; lexical = indenter; } } public SaxSerializer(SAXResult result) { this(result.getHandler(),result.getLexicalHandler()); } // XmlSerializer implementation public void startDocument() { try { writer.startDocument(); } catch (SAXException e) { throw new TxwException(e); } } // namespace prefix bindings // add in #writeXmlns and fired in #endStartTag private final Stack<String> prefixBindings = new Stack<String>(); public void writeXmlns(String prefix, String uri) { // defend against parsers that pass null in for "xmlns" prefix if (prefix == null) { prefix = ""; } if (prefix.equals("xml")) { return; } prefixBindings.add(uri); prefixBindings.add(prefix); } // element stack private final Stack<String> elementBindings = new Stack<String>(); public void beginStartTag(String uri, String localName, String prefix) { // save element bindings for #endTag elementBindings.add(getQName(prefix, localName)); elementBindings.add(localName); elementBindings.add(uri); } // attribute storage // attrs are buffered in #writeAttribute and sent to the content // handler in #endStartTag private final AttributesImpl attrs = new AttributesImpl(); public void writeAttribute(String uri, String localName, String prefix, StringBuilder value) { attrs.addAttribute(uri, localName, getQName(prefix, localName), "CDATA", value.toString()); } public void endStartTag(String uri, String localName, String prefix) { try { while (prefixBindings.size() != 0) { writer.startPrefixMapping(prefixBindings.pop(), // prefix prefixBindings.pop() // uri ); } writer.startElement(uri, localName, getQName(prefix, localName), attrs); attrs.clear(); } catch (SAXException e) { throw new TxwException(e); } } public void endTag() { try { writer.endElement(elementBindings.pop(), // uri elementBindings.pop(), // localName elementBindings.pop() // qname ); } catch (SAXException e) { throw new TxwException(e); } } public void text(StringBuilder text) { try { writer.characters(text.toString().toCharArray(), 0, text.length()); } catch (SAXException e) { throw new TxwException(e); } } public void cdata(StringBuilder text) { if(lexical==null) throw new UnsupportedOperationException("LexicalHandler is needed to write PCDATA"); try { lexical.startCDATA(); text(text); lexical.endCDATA(); } catch (SAXException e) { throw new TxwException(e); } } public void comment(StringBuilder comment) { try { if(lexical==null) throw new UnsupportedOperationException("LexicalHandler is needed to write comments"); else lexical.comment(comment.toString().toCharArray(), 0, comment.length() ); } catch (SAXException e) { throw new TxwException(e); } } public void endDocument() { try { writer.endDocument(); } catch (SAXException e) { throw new TxwException(e); } } public void flush() { // noop } // other methods private static String getQName(String prefix, String localName) { final String qName; if (prefix == null || prefix.length() == 0) qName = localName; else qName = prefix + ':' + localName; return qName; } }