
ezvcard.io.html.HCardPage Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ez-vcard Show documentation
Show all versions of ez-vcard Show documentation
A library that reads and writes vCards, supporting all versions of the vCard standard (2.1, 3.0, and 4.0) as well as xCard (XML-encoded vCards), hCard (HTML-encoded vCards), and jCard (JSON-encoded vCards).
package ezvcard.io.html;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.StringWriter;
import java.io.UncheckedIOException;
import java.io.Writer;
import java.nio.file.Files;
import java.nio.file.Path;
import java.time.Instant;
import java.time.ZoneId;
import java.time.ZoneOffset;
import java.time.format.DateTimeFormatter;
import java.time.temporal.Temporal;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Pattern;
import ezvcard.Ezvcard;
import ezvcard.VCard;
import ezvcard.io.scribe.ScribeIndex;
import ezvcard.parameter.ImageType;
import ezvcard.property.Photo;
import ezvcard.util.DataUri;
import ezvcard.util.VCardDateFormat;
import freemarker.template.Configuration;
import freemarker.template.Template;
import freemarker.template.TemplateException;
/*
Copyright (c) 2012-2023, Michael Angstadt
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
The views and conclusions contained in the software and documentation are those
of the authors and should not be interpreted as representing official policies,
either expressed or implied, of the FreeBSD Project.
*/
/**
*
* Writes {@link VCard} objects to a templated HTML page (hCard format).
*
*
* Example:
*
*
*
* VCard vcard1 = ...
* VCard vcard2 = ...
*
* HCardPage page = new HCardPage();
* page.add(vcard1);
* page.add(vcard2);
*
* File file = new File("hcard.html");
* page.write(file);
*
* @author Michael Angstadt
* @see http://microformats.org/wiki/hcard
*/
public class HCardPage {
private final Template template;
private final List vcards = new ArrayList<>();
/**
* Creates a new hCard page that uses the default template.
*/
public HCardPage() {
Configuration cfg = new Configuration(Configuration.VERSION_2_3_23);
cfg.setClassForTemplateLoading(HCardPage.class, "");
cfg.setWhitespaceStripping(true);
try {
template = cfg.getTemplate("hcard-template.html");
} catch (IOException e) {
//should never be thrown because it's always on the classpath
throw new UncheckedIOException(e);
}
}
/**
* Creates a new hCard page using a custom template. See the default template for an example.
* @param template the template to use
*/
public HCardPage(Template template) {
this.template = template;
}
/**
* Adds a vCard to the HTML page.
* @param vcard the vCard to add
*/
public void add(VCard vcard) {
vcards.add(vcard);
}
/**
* Writes the HTML document to a string.
* @return the HTML document
*/
public String write() {
StringWriter sw = new StringWriter();
try {
write(sw);
} catch (IOException e) {
//should never thrown because we're writing to a string
throw new UncheckedIOException(e);
}
return sw.toString();
}
/**
* Writes the HTML document to an output stream.
* @param out the output stream
* @throws IOException if there's a problem writing to the output stream
*/
public void write(OutputStream out) throws IOException {
write(new OutputStreamWriter(out));
}
/**
* Writes the HTML document to a file.
* @param file the file
* @throws IOException if there's a problem writing to the file
*/
public void write(Path file) throws IOException {
try (Writer writer = Files.newBufferedWriter(file)) {
write(writer);
}
}
/**
* Writes the HTML document to a writer.
* @param writer the writer
* @throws IOException if there's a problem writing to the writer
*/
public void write(Writer writer) throws IOException {
Map map = new HashMap<>();
map.put("vcards", vcards);
map.put("utils", new TemplateUtils());
map.put("translucentBg", readImage("translucent-bg.png", ImageType.PNG));
map.put("noProfile", readImage("no-profile.png", ImageType.PNG));
map.put("ezVCardVersion", Ezvcard.VERSION);
map.put("ezVCardUrl", Ezvcard.URL);
map.put("scribeIndex", new ScribeIndex());
try {
template.process(map, writer);
} catch (TemplateException e) {
throw new RuntimeException(e);
}
writer.flush();
}
/**
* Reads an image from the classpath.
* @param name the file name, relative to this class
* @param mediaType the media type of the image
* @return the image
* @throws IOException if there's a problem reading the image
*/
private Photo readImage(String name, ImageType mediaType) throws IOException {
return new Photo(getClass().getResourceAsStream(name), mediaType);
}
/**
* Utility functions for the freemarker template.
*/
public static class TemplateUtils {
private final Pattern newlineRegex = Pattern.compile("\\r\\n|\\r|\\n");
public String base64(String contentType, byte[] data) {
return new DataUri(contentType, data).toString();
}
public String lineBreaks(String value) {
return newlineRegex.matcher(value).replaceAll("
");
}
public String format(Temporal temporal, String format) {
return format(temporal, format, ZoneOffset.UTC);
}
public String formatLocal(Temporal temporal, String format) {
return format(temporal, format, ZoneId.systemDefault());
}
private String format(Temporal temporal, String format, ZoneId offset) {
if (temporal instanceof Instant) {
temporal = ((Instant) temporal).atZone(offset);
}
return DateTimeFormatter.ofPattern(format).format(temporal);
}
public String format(ZoneOffset offset) {
return VCardDateFormat.BASIC.format(offset);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy