p:selectOneMenu不通过List <String>上的p:列呈现自定义内容

Gin*_*sai 5 jsf selectonemenu primefaces jsf-2

我正在尝试使用Primefaces 5.2 selectOneMenu来显示图像及其文件名.这就是我的xhtml目前的样子:

<h:form>
<h:panelGrid id="createPanelGrid" columns="2">
    <p:outputLabel value="Service Logo:" />
    <p:selectOneMenu value="#{imageBean.selectedImage}" var="l">
        <f:selectItem itemLabel="Select a logo" itemValue="" />
        <f:selectItems value="#{imageBean.imageList}" var="logo" itemLabel="#{logo}" itemValue="#{logo}" />
        <p:column>
            <p:graphicImage value="#{imageBean.imageFolder}/#{l}" style="max-width:50px;max-height:50px;" />
        </p:column>
        <p:column>#{l}</p:column>
    </p:selectOneMenu>
</h:panelGrid>
Run Code Online (Sandbox Code Playgroud)

ManagedBean(imageBean)有

public List<String> getImageList () {
    List<String> imageList = new ArrayList<String>();
    File[] files = absoluteImageFolder.listFiles();
    for (File file : files) {
        imageList.add(file.getName());
    }        
    return imageList;
}
Run Code Online (Sandbox Code Playgroud)

private String selectedImage;

public String getSelectedImage() {
    return selectedImage;
}

public void setSelectedImage(String selectedImage) {
    this.selectedImage = selectedImage;
}
Run Code Online (Sandbox Code Playgroud)

但是,图像不会在网页上呈现,只是文件名(我发布截图但我没有足够的声誉).我没有得到两列(首先是图像,然后是文件名),我只是获取文件名本身.

当我将文件名字符串包装到POJO并使用转换器时它可以工作 - 但只是使用字符串它不会.

我怎样才能让它与Strings一起工作?

Bal*_*usC 9

这个尴尬的行为由SelectOneMenuRenderer源代码确认(行号匹配5.2):

260            if(itemValue instanceof String) {
261                writer.startElement("td", null);
262                writer.writeAttribute("colspan", columns.size(), null);
263                writer.writeText(selectItem.getLabel(), null);
264                writer.endElement("td");
265            } 
266            else {
267                for(Column column : columns) {
268                    writer.startElement("td", null);
269                    renderChildren(context, column);
270                    writer.endElement("td");
271                }
272            }
Run Code Online (Sandbox Code Playgroud)

因此,如果项目值是实例String,<p:column>则完全忽略自定义内容via .这确实没有任何意义.直观的期望是通过var属性和/或<p:column>子项的存在来切换自定义内容.你最好向PrimeFaces报告一个问题来解释/改进这个问题.

除了提供非String类型化的项值之外,解决方法是SelectOneMenuRenderer使用自定义渲染器覆盖该自定义渲染器,该渲染器将String另一个对象包装,该对象恰好返回其中相同的值toString(),例如StringBuilder.这样,呈现器将被欺骗,因为值不是实例String.很高兴他们没有检查instanceof CharSequence.

public class YourSelectOneMenuRenderer extends SelectOneMenuRenderer {

    @Override
    protected void encodeOptionsAsTable(FacesContext context, SelectOneMenu menu, List<SelectItem> selectItems) throws IOException {
        List<SelectItem> wrappedSelectItems = new ArrayList<>();

        for (SelectItem selectItem : selectItems) {
            Object value = selectItem.getValue();

            if (value instanceof String) {
                value = new StringBuilder((String) value);
            }

            wrappedSelectItems.add(new SelectItem(value, selectItem.getLabel()));
        }

        super.encodeOptionsAsTable(context, menu, wrappedSelectItems);
    }

}
Run Code Online (Sandbox Code Playgroud)

要使其运行,请按以下方式注册faces-config.xml:

<render-kit>
    <renderer>
        <component-family>org.primefaces.component</component-family>
        <renderer-type>org.primefaces.component.SelectOneMenuRenderer</renderer-type>
        <renderer-class>com.example.YourSelectOneMenuRenderer</renderer-class>
    </renderer>
</render-kit>
Run Code Online (Sandbox Code Playgroud)