/* * Copyright (c) 2015 Ziver * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package zutil.parser; import zutil.log.LogUtil; import zutil.struct.MutableInt; import java.lang.reflect.Field; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; import java.util.List; import java.util.logging.Level; import java.util.logging.Logger; /** * Class for generating dynamic text/code from set data. * The syntax is similar to the javascript mustache library. * *

* Supported tags: * * * TODO: {{> file}}: include file * TODO: {{=<% %>=}}: change delimiter * * @author Ziver koc */ public class Templator { private static final Logger log = LogUtil.getLogger(); private HashMap data; private TemplateEntity tmplRoot; public Templator(String tmpl){ this.data = new HashMap(); parseTemplate(tmpl); } public void set(String key, Object data){ this.data.put(key, data); } /** * Will clear all data attributes */ public void clear(){ data.clear(); } public String compile(){ StringBuilder str = new StringBuilder(); if(tmplRoot != null) tmplRoot.compile(str); return str.toString(); } /** * Will pare or re-parse the source template. */ private void parseTemplate(String tmpl){ tmplRoot = parseTemplate(new TemplateNode(), tmpl, new MutableInt(), null); } private TemplateNode parseTemplate(TemplateNode root, String tmpl, MutableInt m, String parentTag){ StringBuilder data = new StringBuilder(); boolean tagOpen = false; for(; m.i 0) // Still some text left, add to node root.add(new TemplateStaticString(data.toString())); // If we get to this point means that this node is incorrectly close // or this is the end of the file, so we convert it to a normal node if(parentTag != null) { root = new TemplateNode(root); String tagName = "{{"+parentTag+"}}"; log.severe("Missing closure of tag: " + tagName); root.addFirst(new TemplateStaticString(tagName)); } return root; } /**************************** Template Helper Classes *************************************/ protected interface TemplateEntity { public void compile(StringBuilder str); } protected class TemplateNode implements TemplateEntity { private List entities; public TemplateNode(){ this.entities = new ArrayList(); } public TemplateNode(TemplateNode node){ this.entities = node.entities; } public void addFirst(TemplateEntity s){ entities.add(0, s); } public void add(TemplateEntity s){ entities.add(s); } public void compile(StringBuilder str) { for(TemplateEntity sec : entities) sec.compile(str); } } protected class TemplateCondition extends TemplateNode { private TemplateDataAttribute attrib; public TemplateCondition(String key){ this.attrib = new TemplateDataAttribute(key); } public void compile(StringBuilder str) { Object obj = attrib.getObject(); if(obj != null) { if(obj instanceof Boolean){ if ((Boolean) obj) super.compile(str); } else if(obj instanceof Iterable){ for(Object o : (Iterable)obj){ // Iterate through the whole list set(".", o); super.compile(str); } set(".", null); } else super.compile(str); } } } protected class TemplateNegativeCondition extends TemplateNode { private TemplateDataAttribute attrib; public TemplateNegativeCondition(String key){ this.attrib = new TemplateDataAttribute(key); } public void compile(StringBuilder str) { Object obj = attrib.getObject(); if(obj == null) super.compile(str); else { if(obj instanceof Boolean) { if ( ! (Boolean) obj) super.compile(str); } else if(obj instanceof Collection) { if (((Collection) obj).isEmpty()) super.compile(str); } } } } protected class TemplateStaticString implements TemplateEntity { private String text; public TemplateStaticString(String text){ this.text = text; } public void compile(StringBuilder str) { str.append(text); } } protected class TemplateDataAttribute implements TemplateEntity { private String tag; private String key; private String attrib; public TemplateDataAttribute(String tag){ this.tag = tag; String[] s = tag.trim().split("\\.", 2); this.key = s[0]; if(s.length > 1) this.attrib = s[1]; } public Object getObject(){ if (data.containsKey(tag)) return data.get(tag); else if (data.containsKey(key)) { if (attrib != null) { Object obj = getFieldValue(data.get(key), attrib); if(obj != null) return obj; } else return data.get(key); } return null; } protected Object getFieldValue(Object obj, String attrib){ try { for (Field field : obj.getClass().getDeclaredFields()) { if(field.getName().equals(attrib)) { field.setAccessible(true); return field.get(obj); } } }catch (IllegalAccessException e){ log.log(Level.WARNING, null, e); } return null; } public void compile(StringBuilder str) { Object obj = getObject(); if(obj != null) str.append(obj.toString()); else str.append("{{").append(tag).append("}}"); } } }