JSF SelectOneMenu使用标签作为值的noSelectionOption?

时间:2011-01-04 15:27:04

标签: jsf jsf-2 converter selectonemenu

在“创建新用户”jsf页面中,我有一个带有自定义转换器的SelectOneMenu和一个noSelectionOption selectItem,如下所示:(无关代码无效)

NewUser.xhtml

<h:form>
<h:selectOneMenu value="#{newUserController.user.department}" 
                 required="true" converter="departmentConverter">
    <f:selectItem itemLabel="Select a department" noSelectionOption="true"/>
    <f:selectItems value="#{newUserController.departments}"
                   var="dep" itemLabel="#{dep.name}" itemValue="#{dep}"/>
</h:selectOneMenu>
<p:commandButton action="#{newUserController.saveUser}"
                 value="#{bundle.Save}"
                 ajax="false"/>
</h:form>

NewUserController.java

@ManagedBean
@ViewScoped
public class NewUserController implements Serializable {
private static final long serialVersionUID = 10L;

@EJB private UserBean userBean;
private List<Department> departments;
private User user;

public NewUserController () {
}

@PostConstruct
public void init(){
    user = new User();
    departments = userBean.findAllDepartments();
}

public User getUser() {
    return user;
}

public void setUser(User user) {
    this.user = user;
}

public List<Department> getDepartments(){
    return departments;
}

public String saveUser() {
    // Business logic
}
}

DepartmentConverter.java

@FacesConverter(value="departmentConverter")
public class DepartmentConverter extends EntityConverter {
    public DepartmentConverter(){
        super(Department.class);
    }
}

所有实体的超级转换器

public class EntityConverter<E> implements Converter{
protected Class<E> entityClass;

public EntityConverter(Class<E> type) {
    entityClass = type;
}

@Override
public Object getAsObject(FacesContext facesContext, UIComponent component, String value) {
    if (value == null || value.length() == 0) {
        return null;
    }
    try {
        InitialContext ic = new InitialContext();
        UserBean ub = (UserBean)ic.lookup("java:global/CompetenceRegister/UserBean");
        return ub.find(entityClass, getKey(value));
    } catch (NamingException e) {
        return null;
    }
}

Long getKey(String value) {
    Long key;
    key = Long.valueOf(value);
    return key;
}

String getStringKey(Long value) {
    StringBuilder sb = new StringBuilder();
    sb.append(value);
    return sb.toString();
}

@Override
public String getAsString(FacesContext facesContext, UIComponent component, Object object) {
    if (object == null) {
        return null;
    }
    if (object instanceof AbstractEntity) {
        AbstractEntity e = (AbstractEntity) object;
        return getStringKey(e.getId());
    }
    else
        throw new IllegalArgumentException("object " + object + " is of type " + object.getClass().getName() + "; expected type: " + entityClass.getName());
}

}

但是,当我发布选择了“选择部门”选项的表单时,它会将标签发送到转换器中的getAsObject而不是null,从而导致转换器在getKey中抛出异常(尝试将包含id的String转换为Long)。将selectItem的itemValue属性设置为null无效。该系列中的物品与转换器完美配合。有没有人知道造成这种情况的原因是什么?

更新我忘了提到一件有趣的事情;如果我从SelectOneMenu中删除转换器属性,则noSelectionAttribute可以正常工作,但由于默认转换器不知道如何转换我的对象,因此帖子无法选择真正的部门。这是否意味着noSelectionOption = true是 SUPPOSED 而是发送其标签而转换器是以某种方式预期处理它的?

2 个答案:

答案 0 :(得分:7)

我的问题是切换到使用SelectOneMenu的converter属性而不是使用FacesConverter的forClass属性。

开关

@FacesConverter(value="departmentConverter")
public class DepartmentConverter extends EntityConverter {
public DepartmentConverter(){
    super(Department.class);
}
}

@FacesConverter(forClass=Department.class)
public class DepartmentConverter extends EntityConverter {
public DepartmentConverter(){
    super(Department.class);
}
}

导致我自己的转换器用于实际值,而当NoSelectionOption属性设置为true时,使用默认转换器(空转换器?我无法找到它的源代码)。我的理论是,将此属性设置为true会将值的类型设置为null,并将标签作为值,使其转到特殊转换器,始终返回null或类似的值。使用converter属性而不是forClass会导致我自己的转换器总是被使用而不管类型如何,因此我必须自己处理作为值发送的标签。

答案 1 :(得分:1)

一个有效的解决方案是简单地添加

try{
    return ub.find(entityClass, getKey(value));
}catch(NumberFormatException e){ // Value isn't a long and thus not an id.
    return null;
}

到EntityConverter中的getAsObject,但这感觉就像我在错误的地方修复问题。将标签作为值发送根本没有意义,它应该真的发送NULL。