org.richfaces.cdk.templatecompiler.statements.ForEachStatement Maven / Gradle / Ivy
The newest version!
/*
* JBoss, Home of Professional Open Source
* Copyright 2009, Red Hat, Inc. and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.richfaces.cdk.templatecompiler.statements;
import java.util.Iterator;
import org.richfaces.cdk.Logger;
import org.richfaces.cdk.generate.freemarker.FreeMarkerRenderer;
import org.richfaces.cdk.templatecompiler.ELParser;
import org.richfaces.cdk.templatecompiler.TemplateModel;
import org.richfaces.cdk.templatecompiler.builder.model.JavaField;
import org.richfaces.cdk.templatecompiler.builder.model.JavaImport;
import org.richfaces.cdk.templatecompiler.el.ParsingException;
import org.richfaces.cdk.templatecompiler.el.types.ComplexType;
import org.richfaces.cdk.templatecompiler.el.types.ELType;
import org.richfaces.cdk.templatecompiler.el.types.TypesFactory;
import com.google.common.collect.Iterables;
import com.google.inject.Inject;
/**
* @author Nick Belaevski
* @author Lukas Fryc
*/
public class ForEachStatement extends FreeMarkerTemplateStatementBase {
private static final String FOR_EACH_LOOP_CLASS = "org.richfaces.ui.common.ForEachLoop";
private static final String FOR_EACH_LOOP_STATUS_CLASS = FOR_EACH_LOOP_CLASS + ".Status";
private static String LOOP_OBJECT_PREFIX = "forEach";
private static volatile int LOOP_OBJECT_COUNTER = 1;
private String itemsExpression;
private TypedTemplateStatement itemsStatement;
private ELType itemsType;
private String var;
private String loopObject;
private String varStatus;
private Integer begin;
private Integer end;
private Integer step;
private ELType varType;
private final ELParser parser;
private final Logger log;
private final TypesFactory typesFactory;
private boolean iterator = false;
private boolean iterable = false;
@Inject
public ForEachStatement(@TemplateModel FreeMarkerRenderer renderer, ELParser parser, Logger log, TypesFactory typesFactory) {
super(renderer, "for-each");
this.parser = parser;
this.log = log;
this.typesFactory = typesFactory;
}
/**
* @return the itemsExpression
*/
public TypedTemplateStatement getItemsExpression() {
return itemsStatement;
}
/**
* @return the var
*/
public String getVar() {
return var;
}
/**
* @return the varType
*/
public ELType getVarType() {
return varType;
}
public String getVarStatus() {
return varStatus;
}
public Integer getBegin() {
return begin;
}
public Integer getEnd() {
return end;
}
public Integer getStep() {
return step;
}
public String getLoopObject() {
return loopObject;
}
public boolean isLoopObjectRequired() {
return varStatus != null || begin != null || end != null || step != null;
}
/**
*
*
*
* @param itemsExpression the itemsExpression to set
* @param var
*/
public void setItemsExpression(String itemsExpression, String var, String varStatus, Integer begin, Integer end,
Integer step) {
try {
this.itemsExpression = itemsExpression;
this.itemsStatement = parser.parse(itemsExpression, this, Object.class.getName());
this.itemsType = this.itemsStatement.getType();
this.itemsStatement.setParent(this);
this.varType = this.itemsStatement.getType().getContainerType();
this.var = var;
this.varStatus = varStatus;
this.begin = begin;
this.end = end;
this.step = step;
setVariable(var, this.varType);
if (isLoopObjectRequired()) {
initializeLoopObjectIteration();
} else {
initializePrimitiveIteration();
}
} catch (ParsingException e) {
log.error("Error parsing expression for iteration in statement", e);
}
}
public void initializePrimitiveIteration() {
if (this.itemsType.isArray()) {
iterable = true;
} else {
if (this.itemsType instanceof ComplexType) {
this.itemsType = ((ComplexType) this.itemsType).getRawType();
}
if (typesFactory.getType(Iterable.class).isAssignableFrom(this.itemsType)) {
iterable = true;
} else if (typesFactory.getType(Iterator.class).isAssignableFrom(this.itemsType)) {
iterator = true;
}
}
if (!iterable && !iterator) {
log.info("The expression " + itemsExpression
+ " is not array, Iterable nor Iterator. It will be treated as single object.");
}
}
public void initializeLoopObjectIteration() {
this.loopObject = LOOP_OBJECT_PREFIX + LOOP_OBJECT_COUNTER++;
this.addImport(FOR_EACH_LOOP_CLASS);
setVariable(loopObject, typesFactory.getType(FOR_EACH_LOOP_CLASS));
if (varStatus != null) {
setVariable(varStatus, typesFactory.getType(FOR_EACH_LOOP_STATUS_CLASS));
}
}
@Override
public Iterable getRequiredImports() {
return Iterables.concat(super.getRequiredImports(), itemsStatement.getRequiredImports());
}
@Override
public Iterable getRequiredMethods() {
return Iterables.concat(super.getRequiredMethods(), itemsStatement.getRequiredMethods());
}
@Override
public Iterable getRequiredFields() {
return Iterables.concat(super.getRequiredFields(), itemsStatement.getRequiredFields());
}
public boolean isIterable() {
return iterable;
}
public boolean isIterator() {
return iterator;
}
}