Primefaces selectManyCheckbox с редактируемыми данными, отправляющими NULL в Converter

Я использую Primefaces selectManyCheckbox в редактируемом Datatable, как этот. Когда пользователь нажимает кнопку редактирования, он может выбирать между различными документами в столбце "Документы". Это мой код. Обратите внимание на конвертер, который я использую:

<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-Список в компоненте поддержки, но пока не будет сохранен в базе данных. Если пользователь загружает некоторые документы в Tab1 и переходит в Tab2 (вкладка с данными), он должен иметь возможность выбирать между всеми документами, включая недавно добавленные. Таким образом, пользователь нажимает на кнопку редактирования, все документы появляются в SelectManyCheckbox. Если пользователь выбирает эти документы и нажимает кнопку called-Sign my Converter, вызывается.

Я следовал этому примеру для реализации конвертера. Вот фрагмент кода этого класса:

@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-Метод "нулевой". У кого-нибудь есть предложения, почему этот параметр нулевой? Это потому, что документ еще не сохранился? Можно ли заставить это работать без сохранения документов, добавленных в Tab1?

2 ответа

Решение

Я решил это путем реализации другого конвертера, который расширяет EntityConverter и проверяет, является ли идентификатор пустым. Вот мой код:

@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);
    }
}

Вместо вашего собственного обычая 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;
    }

}
Другие вопросы по тегам