web-dev-qa-db-ja.com

<p:pickList>のカスタムコンバーターを作成する方法

POJOのリストを使用するPrimeFacesコンポーネントを使用する場合、カスタムコンバーターをどのように作成できますか?私の特定の問題は<p:pickList>にあります

<p:pickList converter="????" value="#{bean.projects}" var="project" 
                             itemLabel="#{project.name}" itemValue="#{project}">

コンバーターがなければ、JSFは送信された値を未変換のJava.lang.ClassCastException送信値で設定するため、Java.lang.Stringが得られます。

22
Thang Pham

カスタムコンバーターの作成方法を調査した後の解決策を次に示します。
1。 Java javax.faces.convert.Converter;を実装するクラスを作成します

public class ProjectConverter implements Converter{

   @EJB
   DocumentSBean sBean;

   public ProjectConverter(){
   }

   public Object getAsObject(FacesContext context, UIComponent component, String value){
     return sBean.getProjectById(value);
     //If u look below, I convert the object into a unique string, which is its id.
     //Therefore, I just need to write a method that query the object back from the 
     //database if given a id. getProjectById, is a method inside my Session Bean that
     //does what I just described
   }

   public String getAsString(FacesContext context, UIComponent component, Object value)     
   {
     return ((Project) value).getId().toString(); //--> convert to a unique string.
   }
}

2. faces-config.xmlにカスタムコンバーターを登録します。

<converter>
    <converter-id>projectConverter</converter-id>
    <converter-class>org.xdrawing.converter.ProjectConverter</converter-class>
</converter>

3.したがって、Primefacesコンポーネント内で、converter="projectConverter"を実行します。 projectConverterは、先ほど作成した<convert-id>であることに注意してください。上記の私の問題を解決するために、私はこれを行います:

<p:pickList converter="projectConverter" value="#{bean.projects}" var="project" 
                            itemLabel="#{project.name}" itemValue="#{project}">
18
Thang Pham

他のデータベースへのアクセスがなくても可能ですが、最善の方法がわかりません。私は非常に特殊なコンバーターを使用しており、選択リストでのみ機能します。これを試して:

@FacesConverter(value = "primeFacesPickListConverter")public class PrimeFacesPickListConverter implements Converter {
@Override
public Object getAsObject(FacesContext arg0, UIComponent arg1, String arg2) {
    Object ret = null;
    if (arg1 instanceof PickList) {
        Object dualList = ((PickList) arg1).getValue();
        DualListModel dl = (DualListModel) dualList;
        for (Object o : dl.getSource()) {
            String id = "" + ((Project) o).getId();
            if (arg2.equals(id)) {
                ret = o;
                break;
            }
        }
        if (ret == null)
            for (Object o : dl.getTarget()) {
                String id = "" + ((Project) o).getId();
                if (arg2.equals(id)) {
                    ret = o;
                    break;
                }
            }
    }
    return ret;
}

@Override
public String getAsString(FacesContext arg0, UIComponent arg1, Object arg2) {
    String str = "";
    if (arg2 instanceof Project) {
        str = "" + ((Project) arg2).getId();
    }
    return str;
}

と選択リスト:

<p:pickList converter="primeFacesPickListConverter" value="#{bean.projects}" var="project" 
                        itemLabel="#{project.name}" itemValue="#{project}">

仕事は私にとって、改善が必要です。

39
Braully Rocha

はい、次のようにして、選択リスト内のオブジェクトをシリアル化/逆シリアル化するコンバーターを作成できます。

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

    private static final Logger log = Logger.getLogger(PositionMetricConverter.class.getName());

    @Override
    public Object getAsObject(FacesContext facesContext, UIComponent uiComponent, String value) {
        try {
            byte[] data = Base64.decodeBase64(value);
            ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(data));
            Object o = ois.readObject();
            ois.close();
            return o;
        } catch (Exception e) {
            log.log(Level.SEVERE, "Unable to decode PositionMetric!", e);
            return null;
        }
    }

    @Override
    public String getAsString(FacesContext facesContext, UIComponent uiComponent, Object value) {
        try {
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(baos);
            oos.writeObject(value);
            oos.close();

            return Base64.encodeBase64String(baos.toByteArray());
        } catch (IOException e) {
            log.log(Level.SEVERE, "Unable to encode PositionMetric!", e);
            return "";
        }
    }

}

次に、このコンバーターを次のように選択リストに適用します。

<p:pickList converter="PositionMetricConverter" value="#{bean.positionMetrics}" 
    var="positionMetric" itemLabel="#{positionMetric.name}" itemValue="#{positionMetric}"/>

オブジェクトがシリアライズ可能であることを確認してください。

6
Rares Oltean

この問題は、primeface関連ではなく、一般的なJSF関連のみです。

なぜデータベースに再度アクセスする必要があるのですか? Beanには、コンポーネントに表示するオブジェクトのリストがすでに含まれていますか、それともリクエストスコープですか?

  • Idフィールドを含むHibernate Pojoのスーパークラスを作成します。スーパークラスを作成したくない場合は、pojoクラスを使用しますが、さらに多くのコンバータークラスが必要です。
  • そのスーパークラスを使用すると、コンストラクターで渡されたPojoのリストを含むすべてのPojoクラスの汎用コンバーターを作成できます。
  • コンバーターをセッションBeanのプロパティとして追加し、JSFコンポーネントでそのコンバーターを使用します。

Beanのgetプロパティを介して最終リストにアクセスする場合、またはコンバーターのネストされたリストを選択する場合。

public class SuperPojo
{
    protected Integer id;
    //constructor & getter
}

public class PojoTest extends SuperPojo 
{
    private String label = null;    
    //constructor & getter
}

public class SuperPojoConverter<T extends SuperPojo> implements Converter
{
    private Collection<T> superPojos;

    public IdEntityConverter(Collection<T> superPojos)
    {
        this.superPojos = superPojos;
    }

    @Override
    public Object getAsObject(FacesContext context, UIComponent component, String value)
    {
        //catch exceptions and empty or  null value!
        final int intValue = Integer.parseInt(value);
        for(SuperPojo superPojo : this.superPojos)
            if(superPojo.getId().intValue() == intValue)
                return superPojo;

        return null;
    }

    @Override
    public String getAsString(FacesContext context, UIComponent component, Object value)
    {
        //catch null and instanceof
        return String.valueOf(((SuperPojo)value).getId().intValue());
    }

    public Collection<T> getSuperPojos()
    {
        return this.superPojos;
    }
}

public class Bean 
{
    private SuperPojoConverter<PojoTest> pojoTestConverter = null;

    public Bean()
    {
        final List<PojoTest> pojoTests = //get list from hibernate
        this.pojoTestConverter = new SuperPojoConverter<PojoTest>(pojoTests);
    }

    public SuperPojoConverter<PojoTest> getPojoTestConverter()
    {
        return this.pojoTestConverter;
    }
}


<h:selectOneMenu value="#{selPojoTest}" converter="#{bean.getPojoTestConverter}">
    <f:selectItems value="#{bean.getPojoTestConverter.getSuperPojos}" var="varPojoTest" itemLabel="#{varPojoTest.label}" itemValue="#{varPojoTest}"/>
</h:selectOneMenu>
3
djmj

2つのデータベースヒットなしでそれを実装する方法はありますか?

つまり、あなたが持っているとき

#{bean.projects}

これはデータベースヒットです。

そしてコンバーターが置くとき

sBean.getProjectById(value);

bean.projectsにはすでにオブジェクトのIDと値があるため、不要なデータベースヒットになります。

2
bluefoot

はい、可能です:

public class DocumentSBean sBean implements Serializable{

private List<Document> projects;
// projects methods...
// ...

public Converter getDocumentConverter(){
 return docConverter;
}

private Converter docConverter = new Converter() {

        @Override
        public Object getAsObject(FacesContext context, UIComponent component, String value) {
            return projects.stream().filter(p -> p.getName().equals(value)).findFirst().orElse(null);
        }

        @Override
        public String getAsString(FacesContext context, UIComponent component, Object value) {
            return (value != null)
                    ? ((Document) value).toString()
                    : null;
        }
    };
}
<p:pickList converter="#{sBean.documentConverter}" value="#...
1
user3158918