EDIT: the problem is when the composite component is wrapped inside an ui:repeat or h:datatable and also in a p:dialog
Using http://balusc.blogspot.de/2013/01/composite-component-with-multiple-input.html I seem to have a problem with submitting a form with composite components inside a p:dialog.
Setters on normal p:boolean and such fields are called, but setter from composite component is not, although getSubmittedvalue is.
The initial value is null and the same thing works when NOT in modal p:dialog.
I can see it clearly in debug/profile mode.
INFO: Initializing Mojarra 2.1.6 (SNAPSHOT 20111206) for context '/webclient'
INFO: Running on PrimeFaces 3.5
INFO: Running on PrimeFaces Extensions 0.6.3
EDIT: have tried also initial values to be set to a default value i.e. not null. The same thing happens...setters don't get called.
I must stress out again that the SAME thing works when not in p:dialog (with appendtobody true if that matters).
SSCCE
Controller:
@ManagedBean
@ViewScoped
public class TestController implements Serializable {
List<TestSelect> testbeanList;
List<BeanObject> beanList;
@PostConstruct
void init() {
testbeanList = new ArrayList<TestSelect>() {
{
add(new TestSelect("km"));
add(new TestSelect("g"));
add(new TestSelect("lbs"));
}
};
beanList = new ArrayList<BeanObject>() {
{
add(new BeanObject(2.0d, "lbs"));
add(new BeanObject(3.0d, "g"));
add(new BeanObject(4.0d, "km"));
}
};
}
public List<TestSelect> getTestbeanList() {
return testbeanList;
}
public void setTestbeanList(List<TestSelect> testbeanList) {
this.testbeanList = testbeanList;
}
public BeanObject getBeanObject() {
return BeanObject;
}
public void setBeanObject(BeanObject BeanObject) {
this.BeanObject = BeanObject;
}
}
Bean:
public class BeanObject {
double val;
String type;
public BeanObject(double val, String type) {
super();
this.val = val;
this.type = type;
}
public double getVal() {
return val;
}
public void setVal(double val) {
this.val = val;
}
public String getType() {
return type;
}
public void setType(String type) {
this.type = type;
}
}
Facelet:
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:p="http://primefaces.org/ui"
xmlns:pe="http://primefaces.org/ui/extensions"
>
<h:head>
</h:head>
<h:body>
<p:dialog widgetVar="dlg" appendToBody="true">
<h:form>
<ui:repeat var="uom" value="#{testController.beanList}">
<usb:uom value="#{uom}" list="#{testController.testBeanList}"/>
</ui:repeat>
<p:commandButton process="@form" action="null" value="submit" />
</h:form>
</p:dialog>
<h:form>
<p:commandButton onclick="dlg.show();" value="show dlg" />
</h:form>
</h:body>
</html>
FacesComponent:
@FacesComponent("testComponent") public class TestComponent extends UIInput implements NamingContainer {
/** The Constant LOGGER. */
private static final Logger LOGGER = Logger.getLogger(TestComponent.class);
// Fields
// -------------------------------------------------------------------------------------
/** The measure. */
private UIInput measure;
/** The unit. */
private UISelectOne unit;
// Actions
// ------------------------------------------------------------------------------------
/*
* (non-Javadoc)
*
* @see javax.faces.component.UIInput#getFamily()
*/
/**
* Gets the family.
*
* @return the family
*/
@Override
public String getFamily() {
return UINamingContainer.COMPONENT_FAMILY;
}
/*
* (non-Javadoc)
*
* @see
* javax.faces.component.UIComponent#encodeAll(javax.faces.context.FacesContext
* )
*/
/**
* Encode all.
*
* @param context
* the context
* @throws IOException
* Signals that an I/O exception has occurred.
*/
@Override
public void encodeBegin(FacesContext context) throws IOException {
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("encodeAll BEGIN");
}
Object testTest = getAttributeValue("value", "");
String val = null;
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("testTest -> " + testTest.toString());
}
if (testTest instanceof BeanObject) {
val = testTest.toString();
} else {
val = (String) testTest;
}
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("val -> " + val);
}
if (!val.equals("")) {
String[] testSplitted = val.split(" ");
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("testSplitted -> " + testSplitted);
for (int i = 0; i < testSplitted.length; i++) {
LOGGER.debug("testSplitted it -> " + testSplitted[i]);
}
}
measure.setValue(testSplitted[0]);
unit.setValue(testSplitted[1]);
}
super.encodeBegin(context);
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("encodeAll END");
}
}
/**
* Gets the converted value.
*
* @param context
* the context
* @param submittedValue
* the submitted value
* @return the converted value
*/
@Override
protected Object getConvertedValue(FacesContext context, Object submittedValue) {
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("getConvertedValue()");
}
String[] tmpVal = ((String) getSubmittedValue()).split(" ");
if (LOGGER.isDebugEnabled()) {
for (int i = 0; i < tmpVal.length; i++) {
LOGGER.debug("unit and val -> " + i + " -> " + tmpVal[i]);
}
}
if (tmpVal == null || Strings.isNullOrEmpty(tmpVal[0])) {
return null;
}
BeanObject test = new BeanObject(Double.parseDouble(tmpVal[0]), tmpVal[1]);
return test;
}
/*
* (non-Javadoc)
*
* @see javax.faces.component.UIInput#getSubmittedValue()
*/
/**
* Gets the submitted value.
*
* @return the submitted value
*/
@Override
public Object getSubmittedValue() {
if (Strings.isNullOrEmpty((String) getMeasure().getValue())) {
return null;
}
if (Strings.isNullOrEmpty((String) getUnit().getValue())) {
Object value = getAttributeValue("value", "");
if (value instanceof AttributeObject) {
LOGGER.error("yes1");
}
if (value instanceof IAttributeObject) {
LOGGER.error("yes2");
}
LOGGER.error("getSubmittedValue -> " + value.toString());
}
String submitVal = (String) getMeasure().getValue() + " " + (String) getUnit().getValue();
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("getSubmittedValue()" + submitVal);
}
return submitVal;
}
// Helpers
// ------------------------------------------------------------------------------------
/**
* Return specified attribute value or otherwise the specified default if
* it's null.
*
* @param <T>
* the generic type
* @param key
* the key
* @param defaultValue
* the default value
* @return the attribute value
*/
@SuppressWarnings("unchecked")
private <T> T getAttributeValue(String key, T defaultValue) {
T value = (T) getAttributes().get(key);
return (value != null) ? value : defaultValue;
}
// Getters/setters
// ----------------------------------------------------------------------------
/**
* Gets the measure.
*
* @return the measure
*/
public UIInput getMeasure() {
return measure;
}
/**
* Gets the unit.
*
* @return the unit
*/
public UISelectOne getUnit() {
return unit;
}
/**
* Sets the unit.
*
* @param unit
* the new unit
*/
public void setUnit(UISelectOne unit) {
this.unit = unit;
}
/**
* Sets the measure.
*
* @param measure
* the new measure
*/
public void setMeasure(UIInput measure) {
this.measure = measure;
}
}
component facelet:
<ui:composition
xmlns="http://www.w3.org/1999/xhtml"
xmlns:f="http://java.sun.com/jsf/core"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:cc="http://java.sun.com/jsf/composite"
xmlns:p="http://primefaces.org/ui"
xmlns:pe="http://primefaces.org/ui/extensions">
<cc:interface componentType="testComponent">
<cc:attribute type="java.lang.Object" name="value" required="true" />
<cc:attribute type="java.util.List" name="list"/>
</cc:interface>
<cc:implementation>
<h:panelGrid columns="3">
<p:inputText id="measure" binding="#{cc.measure}">
<f:ajax render="unit" />
</p:inputText>
<p:selectOneMenu id="unit" binding="#{cc.unit}">
<f:selectItem noSelectionOption="true" itemLabel=" " itemValue="" />
<f:selectItems value="#{cc.attrs.list}" var="option" itemLabel="#{option.getShortName()}" itemValue="#{option.getShortName()}" />
<f:ajax render="measure" />
</p:selectOneMenu>
</h:panelGrid>
</cc:implementation>
</ui:composition>
Helper select bean:
public class TestSelect {
private static final Logger LOGGER = Logger.getLogger(TestController.class);
String ShortName;
public TestSelect(String shortName) {
super();
ShortName = shortName;
}
public String getShortName() {
return ShortName;
}
public void setShortName(String shortName) {
ShortName = shortName;
}
@Override
public boolean equals(Object obj) {
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("this " + this.ShortName);
LOGGER.debug("compared " + obj.toString());
}
if (this.ShortName.equals((String)obj)) {
return true;
}
return false;
}
@Override
public String toString() {
return "TestSelect [ShortName=" + ShortName + "]";
}
}