blob: 8db1fbb7120a38f04bb7ebbf9624293aaed01371 [file] [log] [blame]
// ***************************************************************************************************************************
// * 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.juneau.html;
import java.util.*;
import java.util.concurrent.*;
import org.apache.juneau.*;
import org.apache.juneau.annotation.*;
import org.apache.juneau.html.annotation.*;
import org.apache.juneau.serializer.*;
import org.apache.juneau.xml.*;
/**
* Serializes POJO models to HTML.
*
* <h5 class='topic'>Media types</h5>
*
* Handles <c>Accept</c> types: <bc>text/html</bc>
* <p>
* Produces <c>Content-Type</c> types: <bc>text/html</bc>
*
* <h5 class='topic'>Description</h5>
*
* The conversion is as follows...
* <ul class='spaced-list'>
* <li>
* {@link Map Maps} (e.g. {@link HashMap}, {@link TreeMap}) and beans are converted to HTML tables with
* 'key' and 'value' columns.
* <li>
* {@link Collection Collections} (e.g. {@link HashSet}, {@link LinkedList}) and Java arrays are converted
* to HTML ordered lists.
* <li>
* {@code Collections} of {@code Maps} and beans are converted to HTML tables with keys as headers.
* <li>
* Everything else is converted to text.
* </ul>
*
* <p>
* This serializer provides several serialization options. Typically, one of the predefined <jsf>DEFAULT</jsf>
* serializers will be sufficient.
* However, custom serializers can be constructed to fine-tune behavior.
*
* <p>
* The {@link HtmlLink} annotation can be used on beans to add hyperlinks to the output.
*
* <h5 class='topic'>Behavior-specific subclasses</h5>
*
* The following direct subclasses are provided for convenience:
* <ul class='spaced-list'>
* <li>
* {@link Sq} - Default serializer, single quotes.
* <li>
* {@link SqReadable} - Default serializer, single quotes, whitespace added.
* </ul>
*
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Use one of the default serializers to serialize a POJO</jc>
* String html = HtmlSerializer.<jsf>DEFAULT</jsf>.serialize(someObject);
*
* <jc>// Create a custom serializer that doesn't use whitespace and newlines</jc>
* HtmlSerializer serializer = <jk>new</jk> HtmlSerializerBuider().ws().build();
*
* <jc>// Same as above, except uses cloning</jc>
* HtmlSerializer serializer = HtmlSerializer.<jsf>DEFAULT</jsf>.builder().ws().build();
*
* <jc>// Serialize POJOs to HTML</jc>
*
* <jc>// Produces: </jc>
* <jc>// &lt;ul&gt;&lt;li&gt;1&lt;li&gt;2&lt;li&gt;3&lt;/ul&gt;</jc>
* List l = new ObjectList(1, 2, 3);
* String html = HtmlSerializer.<jsf>DEFAULT</jsf>.serialize(l);
*
* <jc>// Produces: </jc>
* <jc>// &lt;table&gt; </jc>
* <jc>// &lt;tr&gt;&lt;th&gt;firstName&lt;/th&gt;&lt;th&gt;lastName&lt;/th&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;Bob&lt;/td&gt;&lt;td&gt;Costas&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;Billy&lt;/td&gt;&lt;td&gt;TheKid&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;Barney&lt;/td&gt;&lt;td&gt;Miller&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;/table&gt; </jc>
* l = <jk>new</jk> ObjectList();
* l.add(<jk>new</jk> ObjectMap(<js>"{firstName:'Bob',lastName:'Costas'}"</js>));
* l.add(<jk>new</jk> ObjectMap(<js>"{firstName:'Billy',lastName:'TheKid'}"</js>));
* l.add(<jk>new</jk> ObjectMap(<js>"{firstName:'Barney',lastName:'Miller'}"</js>));
* String html = HtmlSerializer.<jsf>DEFAULT</jsf>.serialize(l);
*
* <jc>// Produces: </jc>
* <jc>// &lt;table&gt; </jc>
* <jc>// &lt;tr&gt;&lt;th&gt;key&lt;/th&gt;&lt;th&gt;value&lt;/th&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;foo&lt;/td&gt;&lt;td&gt;bar&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;baz&lt;/td&gt;&lt;td&gt;123&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;/table&gt; </jc>
* Map m = <jk>new</jk> ObjectMap(<js>"{foo:'bar',baz:123}"</js>);
* String html = HtmlSerializer.<jsf>DEFAULT</jsf>.serialize(m);
*
* <jc>// HTML elements can be nested arbitrarily deep</jc>
* <jc>// Produces: </jc>
* <jc>// &lt;table&gt; </jc>
* <jc>// &lt;tr&gt;&lt;th&gt;key&lt;/th&gt;&lt;th&gt;value&lt;/th&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;foo&lt;/td&gt;&lt;td&gt;bar&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;baz&lt;/td&gt;&lt;td&gt;123&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;someNumbers&lt;/td&gt;&lt;td&gt;&lt;ul&gt;&lt;li&gt;1&lt;li&gt;2&lt;li&gt;3&lt;/ul&gt;&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;someSubMap&lt;/td&gt;&lt;td&gt; </jc>
* <jc>// &lt;table&gt; </jc>
* <jc>// &lt;tr&gt;&lt;th&gt;key&lt;/th&gt;&lt;th&gt;value&lt;/th&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;tr&gt;&lt;td&gt;a&lt;/td&gt;&lt;td&gt;b&lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;/table&gt; </jc>
* <jc>// &lt;/td&gt;&lt;/tr&gt; </jc>
* <jc>// &lt;/table&gt; </jc>
* Map m = <jk>new</jk> ObjectMap(<js>"{foo:'bar',baz:123}"</js>);
* m.put(<js>"someNumbers"</js>, <jk>new</jk> ObjectList(1, 2, 3));
* m.put(<js>"someSubMap"</js>, <jk>new</jk> ObjectMap(<js>"{a:'b'}"</js>));
* String html = HtmlSerializer.<jsf>DEFAULT</jsf>.serialize(m);
* </p>
*/
@ConfigurableContext
public class HtmlSerializer extends XmlSerializer implements HtmlMetaProvider, HtmlCommon {
//-------------------------------------------------------------------------------------------------------------------
// Configurable properties
//-------------------------------------------------------------------------------------------------------------------
static final String PREFIX = "HtmlSerializer";
/**
* Configuration property: Add <js>"_type"</js> properties when needed.
*
* <h5 class='section'>Property:</h5>
* <ul>
* <li><b>ID:</b> {@link org.apache.juneau.html.HtmlSerializer#HTML_addBeanTypes HTML_addBeanTypes}
* <li><b>Name:</b> <js>"HtmlSerializer.addBeanTypes.b"</js>
* <li><b>Data type:</b> <jk>boolean</jk>
* <li><b>System property:</b> <c>HtmlSerializer.addBeanTypes</c>
* <li><b>Environment variable:</b> <c>HTMLSERIALIZER_ADDBEANTYPES</c>
* <li><b>Default:</b> <jk>false</jk>
* <li><b>Session property:</b> <jk>false</jk>
* <li><b>Annotations:</b>
* <ul>
* <li class='ja'>{@link org.apache.juneau.html.annotation.HtmlConfig#addBeanTypes()}
* </ul>
* <li><b>Methods:</b>
* <ul>
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#addBeanTypes(boolean)}
* </ul>
* </ul>
*
* <h5 class='section'>Description:</h5>
* <p>
* If <jk>true</jk>, then <js>"_type"</js> properties will be added to beans if their type cannot be inferred
* through reflection.
*
* <p>
* When present, this value overrides the {@link #SERIALIZER_addBeanTypes} setting and is
* provided to customize the behavior of specific serializers in a {@link SerializerGroup}.
*/
public static final String HTML_addBeanTypes = PREFIX + ".addBeanTypes.b";
/**
* Configuration property: Add key/value headers on bean/map tables.
*
* <h5 class='section'>Property:</h5>
* <ul>
* <li><b>ID:</b> {@link org.apache.juneau.html.HtmlSerializer#HTML_addKeyValueTableHeaders HTML_addKeyValueTableHeaders}
* <li><b>Name:</b> <js>"HtmlSerializer.addKeyValueTableHeaders.b"</js>
* <li><b>Data type:</b> <jk>boolean</jk>
* <li><b>System property:</b> <c>HtmlSerializer.addKeyValueTableHeaders</c>
* <li><b>Environment variable:</b> <c>HTMLSERIALIZER_ADDKEYVALUETABLEHEADERS</c>
* <li><b>Default:</b> <jk>false</jk>
* <li><b>Session property:</b> <jk>false</jk>
* <li><b>Annotations:</b>
* <ul>
* <li class='ja'>{@link org.apache.juneau.html.annotation.Html#noTableHeaders()}
* </ul>
* <li><b>Methods:</b>
* <ul>
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#addKeyValueTableHeaders(boolean)}
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#addKeyValueTableHeaders()}
* </ul>
* </ul>
*
* <p>
* When enabled, <bc>key</bc> and <bc>value</bc> column headers are added to tables.
*
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> String <jf>f1</jf> = <js>"foo"</js>;
* <jk>public</jk> String <jf>f2</jf> = <js>"bar"</js>;
* }
*
* <jc>// Serializer without headers.</jc>
* WriterSerializer s1 = HtmlSerializer.<jsf>DEFAULT</jsf>;
*
* <jc>// Serializer with headers.</jc>
* WriterSerializer s2 = HtmlSerializer
* .<jsm>create</jsm>()
* .addKeyValueTableHeaders()
* .build();
*
* String withoutHeaders = s1.serialize(<jk>new</jk> MyBean());
* String withHeaders = s2.serialize(<jk>new</jk> MyBean());
* </p>
*
* <p>
* The following shows the difference between the two generated outputs:
*
* <table class='styled'>
* <tr>
* <th><c>withoutHeaders</c></th>
* <th><c>withHeaders</c></th>
* </tr>
* <tr>
* <td>
* <table class='unstyled'>
* <tr><td>f1</td><td>foo</td></tr>
* <tr><td>f2</td><td>bar</td></tr>
* </table>
* </td>
* <td>
* <table class='unstyled'>
* <tr><th>key</th><th>value</th></tr>
* <tr><td>f1</td><td>foo</td></tr>
* <tr><td>f2</td><td>bar</td></tr>
* </table>
* </td>
* </tr>
* </table>
*/
public static final String HTML_addKeyValueTableHeaders = PREFIX + ".addKeyValueTableHeaders.b";
/**
* Configuration property: Look for URLs in {@link String Strings}.
*
* <h5 class='section'>Property:</h5>
* <ul>
* <li><b>ID:</b> {@link org.apache.juneau.html.HtmlSerializer#HTML_detectLinksInStrings HTML_detectLinksInStrings}
* <li><b>Name:</b> <js>"HtmlSerializer.detectLinksInStrings.b"</js>
* <li><b>Data type:</b> <jk>boolean</jk>
* <li><b>System property:</b> <c>HtmlSerializer.detectLinksInStrings</c>
* <li><b>Environment variable:</b> <c>HTMLSERIALIZER_DETECTLINKSINSTRINGS</c>
* <li><b>Default:</b> <jk>true</jk>
* <li><b>Session property:</b> <jk>false</jk>
* <li><b>Annotations:</b>
* <ul>
* <li class='ja'>{@link org.apache.juneau.html.annotation.HtmlConfig#detectLinksInStrings()}
* </ul>
* <li><b>Methods:</b>
* <ul>
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#detectLinksInStrings(boolean)}
* </ul>
* </ul>
*
* <h5 class='section'>Description:</h5>
* <p>
* If a string looks like a URL (i.e. starts with <js>"http://"</js> or <js>"https://"</js>, then treat it like a URL
* and make it into a hyperlink based on the rules specified by {@link #HTML_uriAnchorText}.
*
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a property containing what looks like a URL.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> String <jf>f1</jf> = <js>"http://www.apache.org"</js>;
* }
*
* <jc>// Serializer with link detection.</jc>
* WriterSerializer s1 = HtmlSerializer
* .<jsm>create</jsm>()
* .addKeyValueTableHeaders()
* .build();
*
* <jc>// Serializer without link detection.</jc>
* WriterSerializer s2 = HtmlSerializer
* .<jsm>create</jsm>()
* .addKeyValueTableHeaders()
* .detectLinksInStrings(<jk>false</jk>)
* .build();
*
* String withLinks = s1.serialize(<jk>new</jk> MyBean());
* String withoutLinks = s2.serialize(<jk>new</jk> MyBean());
* </p>
*
* <p>
* The following shows the difference between the two generated outputs:
*
* <table class='styled'>
* <tr>
* <th><c>withLinks</c></th>
* <th><c>withoutLinks</c></th>
* </tr>
* <tr>
* <td>
* <table class='unstyled'>
* <tr><th>key</th><th>value</th></tr>
* <tr><td>f1</td><td><a href='http://www.apache.org'>http://www.apache.org</a></td></tr>
* </table>
* </td>
* <td>
* <table class='unstyled'>
* <tr><th>key</th><th>value</th></tr>
* <tr><td>f1</td><td>http://www.apache.org</td></tr>
* </table>
* </td>
* </tr>
* </table>
*/
public static final String HTML_detectLinksInStrings = PREFIX + ".detectLinksInStrings.b";
/**
* Configuration property: Link label parameter name.
*
* <h5 class='section'>Property:</h5>
* <ul>
* <li><b>ID:</b> {@link org.apache.juneau.html.HtmlSerializer#HTML_labelParameter HTML_labelParameter}
* <li><b>Name:</b> <js>"HtmlSerializer.labelParameter.s"</js>
* <li><b>Data type:</b> <c>String</c>
* <li><b>System property:</b> <c>HtmlSerializer.labelParameter</c>
* <li><b>Environment variable:</b> <c>HTMLSERIALIZER_LABELPARAMETER</c>
* <li><b>Default:</b> <js>"label"</js>
* <li><b>Session property:</b> <jk>false</jk>
* <li><b>Annotations:</b>
* <ul>
* <li class='ja'>{@link org.apache.juneau.html.annotation.HtmlConfig#labelParameter()}
* </ul>
* <li><b>Methods:</b>
* <ul>
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#labelParameter(String)}
* </ul>
* </ul>
*
* <p>
* The parameter name to look for when resolving link labels via {@link #HTML_detectLabelParameters}.
*
* <ul class='seealso'>
* <li class='jf'>{@link #HTML_detectLabelParameters}
* </ul>
*/
public static final String HTML_labelParameter = PREFIX + ".labelParameter.s";
/**
* Configuration property: Look for link labels in URIs.
*
* <h5 class='section'>Property:</h5>
* <ul>
* <li><b>ID:</b> {@link org.apache.juneau.html.HtmlSerializer#HTML_detectLabelParameters HTML_detectLabelParameters}
* <li><b>Name:</b> <js>"HtmlSerializer.detectLabelParameters.b"</js>
* <li><b>Data type:</b> <jk>boolean</jk>
* <li><b>System property:</b> <c>HtmlSerializer.detectLabelParameters</c>
* <li><b>Environment variable:</b> <c>HTMLSERIALIZER_DETECTLABELPARAMETERS</c>
* <li><b>Default:</b> <jk>true</jk>
* <li><b>Session property:</b> <jk>false</jk>
* <li><b>Annotations:</b>
* <ul>
* <li class='ja'>{@link org.apache.juneau.html.annotation.HtmlConfig#detectLabelParameters()}
* </ul>
* <li><b>Methods:</b>
* <ul>
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#detectLabelParameters(boolean)}
* </ul>
* </ul>
*
* <h5 class='section'>Description:</h5>
* <p>
* If the URL has a label parameter (e.g. <js>"?label=foobar"</js>), then use that as the anchor text of the link.
*
* <p>
* The parameter name can be changed via the {@link #HTML_labelParameter} property.
*
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a property containing what looks like a URL.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"http://www.apache.org?label=Apache%20Foundation"</js>);
* }
*
* <jc>// Serializer with label detection.</jc>
* WriterSerializer s1 = HtmlSerializer
* .<jsm>create</jsm>()
* .addKeyValueTableHeaders()
* .build();
*
* <jc>// Serializer without label detection.</jc>
* WriterSerializer s2 = HtmlSerializer
* .<jsm>create</jsm>()
* .addKeyValueTableHeaders()
* .lookForLabelParameters(<jk>false</jk>)
* .build();
*
* String withLabels = s1.serialize(<jk>new</jk> MyBean());
* String withoutLabels = s2.serialize(<jk>new</jk> MyBean());
* </p>
*
* <p>
* The following shows the difference between the two generated outputs.
* <br>Note that they're both hyperlinks, but the anchor text differs:
*
* <table class='styled'>
* <tr>
* <th><c>withLabels</c></th>
* <th><c>withoutLabels</c></th>
* </tr>
* <tr>
* <td>
* <table class='unstyled'>
* <tr><th>key</th><th>value</th></tr>
* <tr><td>f1</td><td><a href='http://www.apache.org?label=Apache%20Foundation'>Apache Foundation</a></td></tr>
* </table>
* </td>
* <td>
* <table class='unstyled'>
* <tr><th>key</th><th>value</th></tr>
* <tr><td>f1</td><td><a href='http://www.apache.org?label=Apache%20Foundation'>http://www.apache.org?label=Apache%20Foundation</a></td></tr>
* </table>
* </td>
* </tr>
* </table>
*/
public static final String HTML_detectLabelParameters = PREFIX + ".detectLabelParameters.b";
/**
* Configuration property: Anchor text source.
*
* <h5 class='section'>Property:</h5>
* <ul>
* <li><b>ID:</b> {@link org.apache.juneau.html.HtmlSerializer#HTML_uriAnchorText HTML_uriAnchorText}
* <li><b>Name:</b> <js>"HtmlSerializer.uriAnchorText.s"</js>
* <li><b>Data type:</b> {@link org.apache.juneau.html.AnchorText}
* <li><b>System property:</b> <c>HtmlSerializer.uriAnchorText</c>
* <li><b>Environment variable:</b> <c>HTMLSERIALIZER_URIANCHORTEXT</c>
* <li><b>Default:</b> <js>"TO_STRING"</js>
* <li><b>Session property:</b> <jk>false</jk>
* <li><b>Annotations:</b>
* <ul>
* <li class='ja'>{@link org.apache.juneau.html.annotation.Html#anchorText()}
* <li class='ja'>{@link org.apache.juneau.html.annotation.HtmlConfig#uriAnchorText()}
* </ul>
* <li><b>Methods:</b>
* <ul>
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#uriAnchorText(AnchorText)}
* <li class='jm'>{@link org.apache.juneau.html.HtmlSerializerBuilder#uriAnchorText(String)}
* </ul>
* </ul>
*
* <h5 class='section'>Description:</h5>
* <p>
* When creating anchor tags (e.g. <code><xt>&lt;a</xt> <xa>href</xa>=<xs>'...'</xs>
* <xt>&gt;</xt>text<xt>&lt;/a&gt;</xt></code>) in HTML, this setting defines what to set the inner text to.
*
* <p>
* The possible values are:
* <ul>
* <li class='jc'>{@link AnchorText}
* <ul>
* <li class='jf'>{@link AnchorText#TO_STRING TO_STRING} (default) - Set to whatever is returned by {@link #toString()} on the object.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"http://www.apache.org?foo=bar#myAnchor"</js>);
* }
*
* <jc>// Serializer with TO_STRING anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer.<jsm>create</jsm>().anchorText(<jsf>TO_STRING</jsf>).build();
*
* <jc>// Produces: &lt;a href='http://www.apache.org?foo=bar#myAnchor'&gt;http://www.apache.org?foo=bar#myAnchor&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#PROPERTY_NAME PROPERTY_NAME} - Set to the bean property name.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"http://www.apache.org?foo=bar#myAnchor"</js>);
* }
*
* <jc>// Serializer with PROPERTY_NAME anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer.<jsm>create</jsm>().anchorText(<jsf>PROPERTY_NAME</jsf>).build();
*
* <jc>// Produces: &lt;a href='http://www.apache.org?foo=bar#myAnchor'&gt;f1&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#URI URI} - Set to the URI value.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"http://www.apache.org?foo=bar#myAnchor"</js>);
* }
*
* <jc>// Serializer with URI anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer.<jsm>create</jsm>().anchorText(<jsf>URI</jsf>).build();
*
* <jc>// Produces: &lt;a href='http://www.apache.org?foo=bar#myAnchor'&gt;http://www.apache.org?foo=bar&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#LAST_TOKEN LAST_TOKEN} - Set to the last token of the URI value.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"http://www.apache.org/foo/bar?baz=qux#myAnchor"</js>);
* }
*
* <jc>// Serializer with LAST_TOKEN anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer.<jsm>create</jsm>().anchorText(<jsf>LAST_TOKEN</jsf>).build();
*
* <jc>// Produces: &lt;a href='http://www.apache.org/foo/bar?baz=qux#myAnchor'&gt;bar&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#URI_ANCHOR URI_ANCHOR} - Set to the anchor of the URL.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"http://www.apache.org/foo/bar?baz=qux#myAnchor"</js>);
* }
*
* <jc>// Serializer with URI_ANCHOR anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer.<jsm>create</jsm>().anchorText(<jsf>URI_ANCHOR</jsf>).build();
*
* <jc>// Produces: &lt;a href='http://www.apache.org/foo/bar?baz=qux#myAnchor'&gt;myAnchor&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#CONTEXT_RELATIVE CONTEXT_RELATIVE} - Same as {@link AnchorText#TO_STRING TO_STRING} but assumes it's a context-relative path.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"bar/baz"</js>);
* }
*
* <jc>// Serializer with CONTEXT_RELATIVE anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer
* .<jsm>create</jsm>()
* .anchorText(<jsf>CONTEXT_RELATIVE</jsf>)
* .uriResolution(<jsf>ROOT_RELATIVE</jsf>)
* .uriRelativity(<jsf>RESOURCE</jsf>)
* .uriContext(<js>"{authority:'http://localhost:10000',contextRoot:'/myContext',servletPath:'/myServlet',pathInfo:'/foo'}"</js>)
* .build();
*
* <jc>// Produces: &lt;a href&#61;'/myContext/myServlet/bar/baz'&gt;myServlet/bar/baz&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#SERVLET_RELATIVE SERVLET_RELATIVE} - Same as {@link AnchorText#TO_STRING TO_STRING} but assumes it's a servlet-relative path.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"bar/baz"</js>);
* }
*
* <jc>// Serializer with SERVLET_RELATIVE anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer
* .<jsm>create</jsm>()
* .anchorText(<jsf>SERVLET_RELATIVE</jsf>)
* .uriResolution(<jsf>ROOT_RELATIVE</jsf>)
* .uriRelativity(<jsf>RESOURCE</jsf>)
* .uriContext(<js>"{authority:'http://localhost:10000',contextRoot:'/myContext',servletPath:'/myServlet',pathInfo:'/foo'}"</js>)
* .build();
*
* <jc>// Produces: &lt;a href&#61;'/myContext/myServlet/bar/baz'&gt;bar/baz&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* <li class='jf'>{@link AnchorText#PATH_RELATIVE PATH_RELATIVE} - Same as {@link AnchorText#TO_STRING TO_STRING} but assumes it's a path-relative path.
* <br>
* <h5 class='section'>Example:</h5>
* <p class='bcode w800'>
* <jc>// Our bean class with a URI property.</jc>
* <jk>public class</jk> MyBean {
* <jk>public</jk> URI <jf>f1</jf> = URI.<jsm>create</jsm>(<js>"bar/baz"</js>);
* }
*
* <jc>// Serializer with PATH_RELATIVE anchor text.</jc>
* WriterSerializer s1 = HtmlSerializer
* .<jsm>create</jsm>()
* .anchorText(<jsf>PATH_RELATIVE</jsf>)
* .uriResolution(<jsf>ROOT_RELATIVE</jsf>)
* .uriRelativity(<jsf>PATH_INFO</jsf>)
* .uriContext(<js>"{authority:'http://localhost:10000',contextRoot:'/myContext',servletPath:'/myServlet',pathInfo:'/foo'}"</js>)
* .build();
*
* <jc>// Produces: &lt;a href&#61;'/myContext/myServlet/foo/bar/baz'&gt;bar/baz&lt;/a&gt;</jc>
* String html = s1.serialize(<jk>new</jk> MyBean());
* </p>
* </ul>
* </ul>
*/
public static final String HTML_uriAnchorText = PREFIX + ".uriAnchorText.s";
//-------------------------------------------------------------------------------------------------------------------
// Predefined instances
//-------------------------------------------------------------------------------------------------------------------
/** Default serializer, all default settings. */
public static final HtmlSerializer DEFAULT = new HtmlSerializer(PropertyStore.DEFAULT);
/** Default serializer, single quotes. */
public static final HtmlSerializer DEFAULT_SQ = new HtmlSerializer.Sq(PropertyStore.DEFAULT);
/** Default serializer, single quotes, whitespace added. */
public static final HtmlSerializer DEFAULT_SQ_READABLE = new HtmlSerializer.SqReadable(PropertyStore.DEFAULT);
//-------------------------------------------------------------------------------------------------------------------
// Predefined subclasses
//-------------------------------------------------------------------------------------------------------------------
/** Default serializer, single quotes. */
public static class Sq extends HtmlSerializer {
/**
* Constructor.
*
* @param ps The property store containing all the settings for this object.
*/
public Sq(PropertyStore ps) {
super(
ps.builder()
.set(WSERIALIZER_quoteChar, '\'')
.build()
);
}
}
/** Default serializer, single quotes, whitespace added. */
public static class SqReadable extends HtmlSerializer {
/**
* Constructor.
*
* @param ps The property store containing all the settings for this object.
*/
public SqReadable(PropertyStore ps) {
super(
ps.builder()
.set(WSERIALIZER_quoteChar, '\'')
.set(WSERIALIZER_useWhitespace, true)
.build()
);
}
}
//-------------------------------------------------------------------------------------------------------------------
// Instance
//-------------------------------------------------------------------------------------------------------------------
private final AnchorText uriAnchorText;
private final boolean
detectLabelParameters,
detectLinksInStrings,
addKeyValueTableHeaders,
addBeanTypes;
private final String labelParameter;
private final Map<ClassMeta<?>,HtmlClassMeta> htmlClassMetas = new ConcurrentHashMap<>();
private final Map<BeanPropertyMeta,HtmlBeanPropertyMeta> htmlBeanPropertyMetas = new ConcurrentHashMap<>();
private volatile HtmlSchemaSerializer schemaSerializer;
/**
* Constructor.
*
* @param ps
* The property store containing all the settings for this object.
*/
public HtmlSerializer(PropertyStore ps) {
this(ps, "text/html", (String)null);
}
/**
* Constructor.
*
* @param ps
* The property store containing all the settings for this object.
* @param produces
* The media type that this serializer produces.
* @param accept
* The accept media types that the serializer can handle.
* <p>
* Can contain meta-characters per the <c>media-type</c> specification of
* {@doc RFC2616.section14.1}
* <p>
* If empty, then assumes the only media type supported is <c>produces</c>.
* <p>
* For example, if this serializer produces <js>"application/json"</js> but should handle media types of
* <js>"application/json"</js> and <js>"text/json"</js>, then the arguments should be:
* <p class='bcode w800'>
* <jk>super</jk>(ps, <js>"application/json"</js>, <js>"application/json,text/json"</js>);
* </p>
* <br>...or...
* <p class='bcode w800'>
* <jk>super</jk>(ps, <js>"application/json"</js>, <js>"*&#8203;/json"</js>);
* </p>
* <p>
* The accept value can also contain q-values.
*/
public HtmlSerializer(PropertyStore ps, String produces, String accept) {
super(ps, produces, accept);
uriAnchorText = getProperty(HTML_uriAnchorText, AnchorText.class, AnchorText.TO_STRING);
detectLabelParameters = getBooleanProperty(HTML_detectLabelParameters, true);
detectLinksInStrings = getBooleanProperty(HTML_detectLinksInStrings, true);
labelParameter = getStringProperty(HTML_labelParameter, "label");
addKeyValueTableHeaders = getBooleanProperty(HTML_addKeyValueTableHeaders, false);
addBeanTypes = getBooleanProperty(HTML_addBeanTypes, getBooleanProperty(SERIALIZER_addBeanTypes, false));
}
@Override /* Context */
public HtmlSerializerBuilder builder() {
return new HtmlSerializerBuilder(getPropertyStore());
}
/**
* Instantiates a new clean-slate {@link HtmlSerializerBuilder} object.
*
* <p>
* This is equivalent to simply calling <code><jk>new</jk> HtmlSerializerBuilder()</code>.
*
* <p>
* Note that this method creates a builder initialized to all default settings, whereas {@link #builder()} copies
* the settings of the object called on.
*
* @return A new {@link HtmlSerializerBuilder} object.
*/
public static HtmlSerializerBuilder create() {
return new HtmlSerializerBuilder();
}
@Override /* Serializer */
public HtmlSerializerSession createSession() {
return createSession(createDefaultSessionArgs());
}
@Override /* Serializer */
public HtmlSerializerSession createSession(SerializerSessionArgs args) {
return new HtmlSerializerSession(this, args);
}
@Override /* XmlSerializer */
public HtmlSerializer getSchemaSerializer() {
if (schemaSerializer == null)
schemaSerializer = builder().build(HtmlSchemaSerializer.class);
return schemaSerializer;
}
//-----------------------------------------------------------------------------------------------------------------
// Extended metadata
//-----------------------------------------------------------------------------------------------------------------
@Override /* HtmlMetaProvider */
public HtmlClassMeta getHtmlClassMeta(ClassMeta<?> cm) {
HtmlClassMeta m = htmlClassMetas.get(cm);
if (m == null) {
m = new HtmlClassMeta(cm, this);
htmlClassMetas.put(cm, m);
}
return m;
}
@Override /* HtmlMetaProvider */
public HtmlBeanPropertyMeta getHtmlBeanPropertyMeta(BeanPropertyMeta bpm) {
if (bpm == null)
return HtmlBeanPropertyMeta.DEFAULT;
HtmlBeanPropertyMeta m = htmlBeanPropertyMetas.get(bpm);
if (m == null) {
m = new HtmlBeanPropertyMeta(bpm.getDelegateFor(), this);
htmlBeanPropertyMetas.put(bpm, m);
}
return m;
}
//-----------------------------------------------------------------------------------------------------------------
// Properties
//-----------------------------------------------------------------------------------------------------------------
/**
* Configuration property: Add <js>"_type"</js> properties when needed.
*
* @see #HTML_addBeanTypes
* @return
* <jk>true</jk> if <js>"_type"</js> properties will be added to beans if their type cannot be inferred
* through reflection.
*/
@Override
protected final boolean isAddBeanTypes() {
return addBeanTypes;
}
/**
* Configuration property: Add key/value headers on bean/map tables.
*
* @see #HTML_addKeyValueTableHeaders
* @return
* <jk>true</jk> if <bc>key</bc> and <bc>value</bc> column headers are added to tables.
*/
protected final boolean isAddKeyValueTableHeaders() {
return addKeyValueTableHeaders;
}
/**
* Configuration property: Look for link labels in URIs.
*
* @see #HTML_detectLabelParameters
* @return
* <jk>true</jk> if we should look for URL label parameters (e.g. <js>"?label=foobar"</js>).
*/
protected final boolean isDetectLabelParameters() {
return detectLabelParameters;
}
/**
* Configuration property: Look for URLs in {@link String Strings}.
*
* @see #HTML_detectLinksInStrings
* @return
* <jk>true</jk> if we should automatically convert strings to URLs if they look like a URL.
*/
protected final boolean isDetectLinksInStrings() {
return detectLinksInStrings;
}
/**
* Configuration property: Link label parameter name.
*
* @see #HTML_labelParameter
* @return
* The parameter name to look for when resolving link labels via {@link #HTML_detectLabelParameters}.
*/
protected final String getLabelParameter() {
return labelParameter;
}
/**
* Configuration property: Anchor text source.
*
* @see #HTML_uriAnchorText
* @return
* When creating anchor tags (e.g. <code><xt>&lt;a</xt> <xa>href</xa>=<xs>'...'</xs>
* <xt>&gt;</xt>text<xt>&lt;/a&gt;</xt></code>) in HTML, this setting defines what to set the inner text to.
*/
protected final AnchorText getUriAnchorText() {
return uriAnchorText;
}
//-----------------------------------------------------------------------------------------------------------------
// Other methods
//-----------------------------------------------------------------------------------------------------------------
@Override /* Context */
public ObjectMap toMap() {
return super.toMap()
.append("HtmlSerializer", new DefaultFilteringObjectMap()
.append("uriAnchorText", uriAnchorText)
.append("detectLabelParameters", detectLabelParameters)
.append("detectLinksInStrings", detectLinksInStrings)
.append("labelParameter", labelParameter)
.append("addKeyValueTableHeaders", addKeyValueTableHeaders)
.append("addBeanTypes", addBeanTypes)
);
}
}