1

我在像这样的可编辑数据表中使用 Primefaces selectManyCheckbox 。当用户单击编辑按钮时,他可以在 Documents-Column 中的不同文档之间进行选择。那是我的代码。请注意我正在使用的转换器:

<p:dataTable id="tbl" styleClass="fixedLayoutTable no-highlight dblclick-edit edit-commit-others"
            value="#{cc.attrs.manager.mg}" var="mg"
             emptyMessage="#{msgs.noManager}" editable="true" widgetVar="tbl">
        <p:column  headerText="Documents">
            <p:cellEditor>
                <f:facet name="output">
                   <h:outputText value="#{msgs.noDocs}" rendered="#{mg.docs.size() == 0}" />
                <h:outputText value="#{mg.docs.toArray()[0].fileName}" rendered="#{mg.docs.size() > 0}" />
            </f:facet>
                <f:facet name="input">
                    <p:selectManyCheckbox id="chkDoc" 
                            value="#{mg.docs}" layout="pageDirection"
                            converter="myConverter" collectionType="java.util.HashSet">
                        <f:selectItems value="#{cc.attrs.manager.docs}"
                            var="d" itemValue="#{d}" itemLabel="#{d.fileName}" />
                    </p:selectManyCheckbox>
                </f:facet>
            </p:cellEditor>
        </p:column>
    </p:dataTable>

在应用程序的另一个选项卡(我们称之为 Tab1)中,用户可以上传一些文档。这些文档已添加到cc.attrs.manager.docs支持 bean 的 -List 中,但不会持久保存在数据库中。如果用户在 Tab1 中上传了一些文档并转到 Tab2(带有数据表的选项卡),他应该能够在所有文档之间进行选择,包括最近添加的文档。于是用户点击编辑按钮,所有的文档出现在SelectManyCheckbox中。如果用户选择这些文档并单击 ✔-Sign 我的转换器会被调用。

我按照这个例子来实现转换器。这是此类的代码片段:

@FacesConverter(value="myConverter")
public class MyConverter implements Converter {

    @Override
    public Object getAsObject(FacesContext context, UIComponent component, String value) {
        if (component instanceof PickList) {
            return getAsObject((PickList)component, value);
        } else {
            return SelectItemsUtils.findValueByStringConversion(context, component, value, this);
        }
    }
}

这工作正常,直到用户不检查最近添加的文档。如果他检查新文档之一,则getAsObject-方法的字符串参数为“null”。有没有人有一些建议,为什么这个参数是空的?是因为文档还没有持久化吗?是否可以在不保留 Tab1 中添加的文档的情况下使其正常工作?

4

2 回答 2

2

而不是你自己的 custom converter,试试这个generic converter对所有事情都有效的方法。

import java.util.Map;
import java.util.Map.Entry;
import java.util.UUID;
import java.util.WeakHashMap;

import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import javax.faces.convert.Converter;
import javax.faces.convert.FacesConverter;

@FacesConverter(value = "entityConverter")
public class EntityConverter implements Converter {

    private static Map<Object, String> entities = new WeakHashMap<Object, String>();

    @Override
    public String getAsString(FacesContext context, UIComponent component, Object entity) {
        synchronized (entities) {
            if (!entities.containsKey(entity)) {
                String uuid = UUID.randomUUID().toString();
                entities.put(entity, uuid);
                return uuid;
            } else {
                return entities.get(entity);
            }
        }
    }

    @Override
    public Object getAsObject(FacesContext context, UIComponent component, String uuid) {
        for (Entry<Object, String> entry : entities.entrySet()) {
            if (entry.getValue().equals(uuid)) {
                return entry.getKey();
            }
        }
        return null;
    }

}
于 2013-12-23T10:17:28.837 回答
1

我通过实现另一个扩展 EntityConverter 并检查 ID 是否为空的转换器来解决它。那就是我的代码,例如:

@FacesConverter(value = "anotherConverter")
public class AnotherConverter extends EntityConverter {

    @Override
    public String getAsString(FacesContext context, UIComponent component, Object value) {
        if (value instanceof Document && ((Document) value).getId() == null) {
            return String.valueOf(((Document) value).hashCode());
        }
        return super.getAsString(context, component, value);
    }
}
于 2013-12-27T20:26:35.907 回答