javaHTML is a Java library for efficiently writing valid HTML using a fluid API. It's more readable and less error prone than string concatenation (or manually writing to stream).
Supports fully inline iteration, code execution, conditional expressions, and reusable components.
html().with(
head().with(
meta(attr("charset", "UTF-8"))
),
body().with(
h1().with("Hello world!")
)
).validate();
You'll find information about installation and integration below.
Most of the tags are not self-closing and can contain other elements. These are added by passing them to the with()
method:
ol().with(
li().with("Groucho"),
li().with("Zeppo")
);
Important: If no elements are added to the body of the tag(added using with()
method) you must call close()
at the end.
ol( cls("empty-list") ).close();
Other tags are automatically self-closing and therefore do not accept child elements via the with()
method. You can write text nodes (unwrapped) using the text()
method.
It's good policy to call validate()
at the end of the HTML generation to make sure all tags were properly closed. Failure to close elements or otherwise invalid markup will be logged out at a warn
level.
ol().with(
li().with("Groucho"),
li().with("Zeppo")
).validate();
a(Attribute... attr)
, body(Attribute... attr)
, checkbox(Attribute... attr)
, custom(String name, Attribute... attr)
, dd(Attribute... attr)
, div(Attribute... attr)
, dl(Attribute... attr)
, dt(Attribute... attr)
, em(Attribute... attr)
, fieldset(Attribute... attr)
, form(Attribute... attr)
, head(Attribute... attr)
, html(Attribute... attr)
, img(Attribute... attr)
, input(Attribute... attr)
, legend(Attribute... attr)
, li(Attribute... attr)
, meta(Attribute... attr)
, ol(Attribute... attr)
, p(Attribute... attr)
, script(Attribute... attr)
, span(Attribute... attr)
, text(Object text)
, textarea(String value, Attribute... attr)
, ul(Attribute... attr)
You can add any tag or element using the custom()
method:
custom("blink").with("Las Vegas Baby!");
Would generate:
<blink>Las Vegas Baby</blink>
You can add attributes to custom tags like any other:
custom("strong", style("background-color: red;")).with("Rocky!");
Would generate:
<strong style="background-color: red;">Rocky!</strong>
You can add attributes to the HTML tags by declaring them as formal parameters to the tag methods:
h1( id("foo"), cls("bar", "zip") )
You can add an attribute to an element if a test expression evaluates to true:
h1( id("foo"), when(hide, style("display: none;") );
cls(String... values)
, data(String name, String value)
, height(int value)
, href(String value)
, id(String id)
, src(String value)
, style(String value)
, target(String value)
, title(String value)
, width(int value)
You will find these shorthand attributes for the more common types but any attribute can be declared using the attr("name", "value")
expression:
a( href("#foo"), target("_blank"), attr("onclick", "doSomething();") );
You can conditionally output data using a when()
expression:
div().with(
when( Calendar.getInstance().get(Calendar.HOUR_OF_DAY) < 11 ).use(
h1().with("Good morning!")
).otherwise(
h2().with("Hello!")
)
);
Important: You must terminate a when()
with at least an empty otherwise()
, semantically like an otherwise do nothing.
div().with(
when( Calendar.getInstance().get(Calendar.HOUR_OF_DAY) < 11 ).use(
h1().with("Good morning!")
).otherwise()
);
You can add iteration inline using the iterate()
method and a Loop
implementation:
ol().with(
iterate(new Loop<String>("Groucho", "Zeppo") {
@Override
public void each(String brother) throws Exception {
li().with(brother);
}
})
);
The Loop
class is parameterised and instantiated with a Collection
or varargs and the each()
method is invoked for each item in the list.
An iterate()
behaves just like any other content which means you can add other elements to the same container:
dl().with(
dt().with(
text("Fibonacci")
),
iterate(new Loop<Integer>(0, 1, 1, 2, 3, 5, 8) {
@Override
public void each(Integer fib) throws Exception {
dd().with(text(fib));
}
})
);
You can execute any code inline using the exec()
method and passing it an instance of Code
where you implement the run()
method:
final Random random = new Random();
h1().with(
text("Tom "),
exec(new Code() {
@Override
public void run() throws JspException, IOException {
if (random.nextBoolean()) {
em().with("hates");
} else {
span().with("likes");
}
}
}),
text(" Jerry")
);
If you use the exec()
method in a when()
conditional block it will only execute if the condition was met.
You can encapsulate more complex patterns or markup as reusable components. The recommended pattern would be to implement a Component
class, and pass any required data in via the constructor.
public class Greeting extends Component {
private String name;
public Greeting(String name) {
this.name = name;
}
@Override
public void markup() throws IOException {
h1().with("Hello " + name + "!");
}
}
You can then use your component as if it was any other element by passing it into the component()
method:
body().with(
component( new Greeting("Groucho") ),
h2().with("Take me to your leader!")
);
Add the javaHTML jar to the class path or include using Maven:
<dependency>
<groupId>twigkit</groupId>
<artifactId>javaHTML</artifactId>
<version>2.0-SNAPSHOT</version>
</dependency>
Simple really, you build your HTML by using one of the helper classes for servlets, custom tags, extend HtmlCapabilityImpl
or use the Create
class.
Extend HtmlHttpServlet
rather than HttpServlet
and implement the doGetHtml()
method:
public class MarxServlet extends HtmlHttpServlet {
@Override
protected void doGetHtml(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ol().with(
li().with("Groucho"),
li().with("Zeppo")
).validate();
}
}
If you're using SimpleTagSupport
then extend HtmlSimpleTagSupport
and implement doTagHtml()
as you would normally have done in doTag()
:
public static class MarxTag extends HtmlSimpleTagSupport {
@Override
public void doTagHtml() throws JspException, IOException {
ol().with(
li().with("Groucho"),
li().with("Zeppo")
).validate();
}
}
Use the Create
class and either use the default toString()
method or pass in your own Writer
in the constructor.
new Create() {
@Override
public void markup() throws IOException {
div().with("Hello world!");
}
}.toString();