com.x5.template.Theme Maven / Gradle / Ivy
package com.x5.template;
import java.io.PrintStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import com.x5.template.filters.ChunkFilter;
import com.x5.template.providers.TranslationsProvider;
import com.x5.util.Path;
public class Theme implements ContentSource, ChunkFactory
{
private ArrayList themeLayers = new ArrayList();
public static final String DEFAULT_THEMES_FOLDER = "themes";
private String classpathThemesFolder;
private String themesFolder;
private String themeLayerNames;
private String fileExtension;
private int cacheMins = 0;
private String localeCode = null;
private boolean hardFailMissingTemplate = false;
private boolean renderErrs = true;
private PrintStream errLog = null;
public Theme()
{
this(null, null, ThemeConfig.STANDARD_DEFAULT_EXT);
}
public Theme(ThemeConfig config)
{
this(config.getThemeFolder(), config.getLayerNames(), config.getDefaultExtension());
this.setDirtyInterval(config.getCacheMinutes());
this.localeCode = config.getLocaleCode();
if (config.hideErrors()) {
this.setErrorHandling(false, config.getErrorLog());
}
if (config.abortOnMissingTemplate()) {
this.setAbortOnMissingTemplate(true);
}
ChunkFilter[] filters = config.getFilters();
if (filters != null) {
for (ChunkFilter filter : filters) {
this.registerFilter(filter);
}
}
String encoding = config.getEncoding();
if (encoding != null) {
this.setEncoding(encoding);
}
String themeResourcePath = config.getThemeResourcePath();
if (themeResourcePath != null) {
this.setClasspathThemesFolder(themeResourcePath);
}
}
public Theme(ContentSource templates)
{
themeLayers.add(templates);
}
public Theme(String themeLayerNames)
{
this(null, themeLayerNames, ThemeConfig.STANDARD_DEFAULT_EXT);
}
public Theme(String themesFolder, String themeLayerNames)
{
this(themesFolder, themeLayerNames, ThemeConfig.STANDARD_DEFAULT_EXT);
}
public Theme(String themesFolder, String themeLayerNames, String ext)
{
this.themesFolder = themesFolder;
this.themeLayerNames = themeLayerNames;
this.fileExtension = ext;
}
public void setClasspathThemesFolder(String classpathThemesFolder)
{
if (this.themeLayers.size() > 0) {
throw new java.lang.IllegalStateException("Must specify paths before lazy init.");
} else {
this.classpathThemesFolder = classpathThemesFolder;
}
}
public void setTemplateFolder(String pathToTemplates)
{
if (this.themeLayers.size() > 0) {
throw new java.lang.IllegalStateException("Must specify paths before lazy init.");
} else {
this.themesFolder = pathToTemplates;
}
}
public void setDefaultFileExtension(String ext)
{
if (this.themeLayers.size() > 0) {
throw new java.lang.IllegalStateException("Must specify extension before lazy init.");
} else {
this.fileExtension = ext;
}
}
public void setLocale(String localeCode)
{
this.localeCode = localeCode;
}
public String getLocale()
{
return this.localeCode;
}
/**
* What encoding do this theme's template files use?
* If not UTF-8, make sure to set this explicitly.
*/
public void setEncoding(String encoding)
{
ArrayList templateSets = getTemplateSets();
if (templateSets != null) {
for (TemplateSet layer : templateSets) {
layer.setEncoding(encoding);
}
}
}
private void init()
{
if (classpathThemesFolder == null) classpathThemesFolder = "/" + DEFAULT_THEMES_FOLDER;
if (themesFolder == null) themesFolder = DEFAULT_THEMES_FOLDER;
themesFolder = Path.ensureTrailingFileSeparator(themesFolder);
String[] layerNames = parseLayerNames(themeLayerNames);
if (layerNames == null) {
TemplateSet simple = new TemplateSet(classpathThemesFolder, themesFolder, fileExtension, cacheMins);
if (!renderErrs) simple.signalFailureWithNull();
if (hardFailMissingTemplate) simple.setHardFail(true);
simple.setTranslationsProvider(translationsProvider);
themeLayers.add(simple);
} else {
for (int i=0; i getThemeLayers()
{
// funneling all access through here to enable lazy init.
if (themeLayers.size() < 1) { init(); }
return themeLayers;
}
private String[] parseLayerNames(String themeLayerNames)
{
if (themeLayerNames == null || themeLayerNames.trim().length() == 0) {
return null;
}
return themeLayerNames.split(" *, *");
}
public void setDirtyInterval(int minutes)
{
if (this.themeLayers.size() == 0) {
// lazy init has not happened yet
this.cacheMins = minutes;
} else {
// propagate setting down to each layer
ArrayList templateSets = getTemplateSets();
if (templateSets != null) {
for (TemplateSet layer : templateSets) {
layer.setDirtyInterval(minutes);
}
}
}
}
public Snippet getSnippet(String templateName, String ext)
{
ArrayList layers = getThemeLayers();
// later layers have precedence if they provide the item
for (int i=layers.size()-1; i>=0; i--) {
ContentSource x = layers.get(i);
Snippet template = x.getSnippet(";" + ext + ";" + templateName);
if (template != null) {
return template;
}
}
return prettyFail(templateName, ext);
}
public Snippet getSnippet(String itemName)
{
ArrayList layers = getThemeLayers();
// later layers have precedence if they provide the item
for (int i=layers.size()-1; i>=0; i--) {
ContentSource x = layers.get(i);
if (x.provides(itemName)) {
return x.getSnippet(itemName);
}
/*
String template = x.fetch(itemName);
if (template != null) {
return template;
}*/
}
return prettyFail(itemName, null);
}
public boolean provides(String itemName)
{
for (int i=themeLayers.size()-1; i>=0; i--) {
ContentSource x = themeLayers.get(i);
if (x.provides(itemName)) return true;
}
return false;
}
private Snippet prettyFail(String templateName, String ext)
{
if (!renderErrs && errLog == null) return null;
String prettyExt = ext;
if (prettyExt == null) {
ContentSource baseLayer = themeLayers.get(0);
if (baseLayer instanceof TemplateSet) {
prettyExt = ((TemplateSet)baseLayer).getDefaultExtension();
}
}
StringBuilder err = new StringBuilder();
err.append("[");
if (prettyExt != null) {
err.append(prettyExt);
err.append(" ");
}
err.append("template '");
err.append(templateName);
err.append("' not found]");
if (prettyExt != null) {
String places = "";
ArrayList templateSets = getTemplateSets();
if (templateSets != null) {
for (int i=templateSets.size()-1; i>=0; i--) {
TemplateSet ts = templateSets.get(i);
if (places.length() > 0) { places += ","; }
places += ts.getTemplatePath(templateName,prettyExt);
}
}
if (places.length() > 0) {
err.append("");
}
}
if (errLog != null) {
Chunk.logChunkError(errLog, err.toString());
}
if (hardFailMissingTemplate) {
throw new TemplateNotFoundException(err.toString());
}
return renderErrs ? Snippet.getSnippet(err.toString()) : null;
}
public String fetch(String itemName)
{
ArrayList layers = getThemeLayers();
// later layers have precedence if they provide the item
for (int i=layers.size()-1; i>=0; i--) {
ContentSource x = layers.get(i);
if (x.provides(itemName)) {
return x.fetch(itemName);
}
}
return null;
/*
Snippet s = getSnippet(itemName);
if (s == null) return null;
return s.toString();
*/
}
public String getProtocol()
{
return "include";
}
/**
* Creates a Chunk with no starter template and sets its tag boundary
* markers to match the other templates in this set. The Chunk will need
* to obtain template pieces via its .append() method.
* @return blank Chunk.
*/
public Chunk makeChunk()
{
Chunk c = new Chunk();
// provide theme path context and user-provided filters
c.setMacroLibrary(this, this);
// make sure chunk inherits theme settings
shareContentSources(c);
c.setLocale(localeCode);
c.setErrorHandling(renderErrs, errLog);
return c;
}
/**
* Creates a Chunk with a starting template. If templateName contains one
* or more hashes (#) it is assumed that the template definition is nested inside
* another template. Everything up to the first hash is part of the filename
* (appends the DEFAULT extension to find the file) and everything after
* refers to a location within the file where the template contents are
* defined.
*
* @param templateName the location of the template definition.
* @return a Chunk pre-initialized with a snippet of template.
*/
public Chunk makeChunk(String templateName)
{
Chunk c = new Chunk();
c.setMacroLibrary(this,this);
c.append( getSnippet(templateName) );
// make sure chunk inherits theme settings
shareContentSources(c);
c.setLocale(localeCode);
c.setErrorHandling(renderErrs, errLog);
return c;
}
/**
* Creates a Chunk with a starting template. If templateName contains one
* or more hashes (#) it is assumed that the template definition is nested inside
* another template. Everything up to the first hash is part of the filename
* (appends the PASSED extension to find the file) and everything after
* refers to a location within the file where the template contents are
* defined.
*
* @param templateName the location of the template definition.
* @param extension the nonstandard extension which forms the template filename.
* @return a Chunk pre-initialized with a snippet of template.
*/
public Chunk makeChunk(String templateName, String extension)
{
Chunk c = new Chunk();
c.setMacroLibrary(this,this);
c.append( getSnippet(templateName, extension) );
// make sure chunk inherits theme settings
shareContentSources(c);
c.setLocale(localeCode);
c.setErrorHandling(renderErrs, errLog);
return c;
}
// chunk factory now supports sharing content sources with its factory-created chunks
private HashSet altSources = null;
// theme can have custom translations provider which it shares with its chunks
private TranslationsProvider translationsProvider = null;
public void setTranslationsProvider(TranslationsProvider provider) {
this.translationsProvider = provider;
}
public void addProtocol(ContentSource src)
{
if (altSources == null) altSources = new HashSet();
altSources.add(src);
}
private void shareContentSources(Chunk c)
{
if (altSources != null) {
java.util.Iterator iter = altSources.iterator();
while (iter.hasNext()) {
ContentSource src = iter.next();
c.addProtocol(src);
}
}
if (translationsProvider != null) {
c.setTranslationsProvider(translationsProvider);
}
}
/**
* If your templates are packaged into a jar with your application code,
* then you should use this method to tell chunk where your templates are.
*
* Chunk might still be able to find your templates but it will work a
* lot harder. Without this info, it has to peek into every jar in the
* classpath every time it loads a new template.
*
* @param classInSameJar
*/
public void setJarContext(Class> classInSameJar)
{
ArrayList templateSets = getTemplateSets();
if (templateSets != null) {
for (TemplateSet layer : templateSets) {
layer.setJarContext(classInSameJar);
}
}
}
public void setJarContext(Object ctx)
{
ArrayList templateSets = getTemplateSets();
if (templateSets != null) {
for (TemplateSet layer : templateSets) {
layer.setJarContext(ctx);
}
}
}
private ArrayList getTemplateSets()
{
ArrayList sets = null;
for (ContentSource x : getThemeLayers()) {
if (x instanceof TemplateSet) {
if (sets == null) sets = new ArrayList();
sets.add((TemplateSet)x);
}
}
return sets;
}
// now supporting user-contributed filters
private Map customFilters;
public Map getFilters()
{
return customFilters;
}
public void registerFilter(ChunkFilter filter)
{
if (customFilters == null) {
customFilters = new HashMap();
}
customFilters.put(filter.getFilterName(),filter);
String[] aliases = filter.getFilterAliases();
if (aliases != null) {
for (String alias : aliases) {
customFilters.put(alias,filter);
}
}
}
public void setErrorHandling(boolean renderErrs, PrintStream errLog)
{
this.renderErrs = renderErrs;
this.errLog = errLog;
}
public void setAbortOnMissingTemplate(boolean hardFail)
{
this.hardFailMissingTemplate = hardFail;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy