org.apache.shindig.gadgets.parse.HtmlSerialization Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of shindig-gadgets Show documentation
Show all versions of shindig-gadgets Show documentation
Renders gadgets, provides the gadget metadata service, and serves
all javascript required by the OpenSocial specification.
The newest version!
/*
* 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.
*/
package org.apache.shindig.gadgets.parse;
import com.google.caja.lexer.escaping.Escaping;
import com.google.common.collect.ImmutableSet;
import org.apache.xerces.xni.QName;
import org.w3c.dom.Attr;
import org.w3c.dom.Document;
import org.w3c.dom.DocumentType;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import java.io.IOException;
import java.io.StringWriter;
import java.util.Set;
/**
* Static class with helpers to manage serialization of a Document.
* Binds an HtmlSerializer to a Document as user data, and pulls it out
* to achieve actual serialization.
*/
public class HtmlSerialization {
/**
* Used to key an instance of HtmlSerializer in
* document.getUserData
*/
public static final String KEY = "serializer";
/**
* Used by a parser to record the original length of the content it parsed
* Can be used to optimize output buffers
*/
private static final String ORIGINAL_LENGTH = "original-length";
public static final Set URL_ATTRIBUTES = ImmutableSet.of("href", "src");
/**
* Attach a serializer instance to the document
* @param doc
* @param serializer
* @param originalContent may be null
*/
public static void attach(Document doc, HtmlSerializer serializer, String originalContent) {
doc.setUserData(KEY, serializer, null);
if (originalContent != null) {
doc.setUserData(ORIGINAL_LENGTH, originalContent.length(), null);
}
}
/**
* Copy serializer from one document to another. Note this requires that
* serializers are thread safe
*/
static void copySerializer(Document from, Document to) {
Integer length = (Integer)from.getUserData(ORIGINAL_LENGTH);
if (length != null) to.setUserData(ORIGINAL_LENGTH, length, null);
to.setUserData(KEY, from.getUserData(KEY), null);
}
/**
* Get the length of the original version of the document
* @param doc
* @return
*/
private static int getOriginalLength(Document doc) {
Integer length = (Integer)doc.getUserData(ORIGINAL_LENGTH);
if (length == null) return -1;
return length;
}
/**
* Create a writer sized to the original length of the document
* @param doc
* @return
*/
public static StringWriter createWriter(Document doc) {
int originalLength = getOriginalLength(doc);
if (originalLength == -1) {
return new StringWriter(8192);
} else {
// Typically rewriting makes a document larger
return new StringWriter((originalLength * 11) / 10);
}
}
/**
* Call the attached serializer and output the document
* @param doc
* @return
*/
public static String serialize(Document doc) {
return ((HtmlSerializer) doc.getUserData(KEY)).serialize(doc);
}
public static void printEscapedText(CharSequence text, Appendable output) throws IOException {
Escaping.escapeXml(text, true, output);
}
/**
* Print the start of an HTML element. If withXmlClose==true, this is an
* empty element that should have its content
*/
public static void printStartElement(Element elem, Appendable output, boolean withXmlClose)
throws IOException {
printStartElement(elem.getTagName(), elem.getAttributes(), output, withXmlClose);
}
public static void printStartElement(String tagName, NamedNodeMap attributes, Appendable output,
boolean withXmlClose) throws IOException {
output.append("<").append(tagName);
for (int i = 0; i < attributes.getLength(); i++) {
Attr attr = (Attr)attributes.item(i);
String attrName = attr.getNodeName();
output.append(' ').append(attrName);
if (attr.getNodeValue() != null) {
output.append("=\"");
if (attr.getNodeValue().length() != 0) {
printEscapedText(attr.getNodeValue(), output);
}
output.append('"');
}
}
output.append(withXmlClose ? "/>" : ">");
}
public static void outputDocType(DocumentType docType, Appendable output) throws IOException {
output.append(" 0) {
output.append(" ");
output.append("PUBLIC ").append('"').append(docType.getPublicId()).append('"');
}
if (docType.getSystemId() != null && docType.getSystemId().length() > 0) {
output.append(" ");
output.append('"').append(docType.getSystemId()).append('"');
}
output.append(">\n");
}
/**
* Returns true if the listed attribute is an URL attribute.
*/
public static boolean isUrlAttribute(QName name, String attributeName) {
return name.uri == null && URL_ATTRIBUTES.contains(attributeName);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy