JSF selectOneMenu:扩展MenuRenderer,用于在有一个项目时显示纯文本

时间:2014-12-05 20:18:53

标签: jsf

我想显示类似

的内容
<span ...>An item</span>

而不是

<select ...>
 <option ...>An item</option>
</select>

当菜单的 plainText 选项设置为true且内部只有一个SelectItem时:

<h:selectOneMenu plainText="true" ...>
 <f:selectItems ... /> <!-- contains one item -->
</h:selectOneMenu>

我尝试过创建一个从MenuRenderer扩展的自定义渲染器,但它的代码实际上不是可附加的。你能帮忙吗?感谢。

1 个答案:

答案 0 :(得分:0)

我们需要扩展 com.sun.faces.renderkit.html_basic.MenuRenderer 并覆盖 renderSelect renderOption 方法。它的代码大多是从父代码中复制的。当我们需要将元素显示为span时, private boolean plainText 为true。 在 renderSelect 的开头,我们添加:

plainText = "true".equalsIgnoreCase((String) component.getAttributes().get("plainText"));
if (plainText) {
    SelectItemsIterator<SelectItem> items = RenderKitUtils.getSelectItems(context, component);
    items.next();
    plainText = !items.hasNext();
}

首先检查 plainText 选项是否设置为 true ,而不是检查是否只有selectItem。我认为这部分可以进行优化。 而不是

writer.startElement("select", component);

我们应该写

if (plainText) {
    writer.startElement("span", component);
} else {
    writer.startElement("select", component);
}

与writer.endElement一样。

renderOption 的中间,我们添加了一个代码,它只打印出没有任何标签的唯一选项并离开。

if (plainText) {
    if (curItem.isEscape()) {
        String label = curItem.getLabel();
        if (label == null) {
            label = valueString;
        }
        writer.writeText(label, component, "label");
    } else {
        writer.write(curItem.getLabel());
    }
    writer.writeText("\n", component, null);
    return true;
}

所有代码:

import com.sun.faces.io.FastStringWriter;
import com.sun.faces.renderkit.Attribute;
import com.sun.faces.renderkit.AttributeManager;
import com.sun.faces.renderkit.RenderKitUtils;
import com.sun.faces.renderkit.SelectItemsIterator;
import com.sun.faces.renderkit.html_basic.MenuRenderer;

import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import javax.faces.context.ResponseWriter;
import javax.faces.convert.Converter;
import javax.faces.model.SelectItem;
import java.io.IOException;
import java.util.logging.Level;

public class ExtendedMenuRenderer extends MenuRenderer {
    private boolean plainText;

    private static final Attribute[] ATTRIBUTES =
            AttributeManager.getAttributes(AttributeManager.Key.SELECTMANYMENU);

    @Override
    protected void renderSelect(FacesContext context,
                                UIComponent component) throws IOException {
        plainText = "true".equalsIgnoreCase((String) component.getAttributes().get("plainText"));
        if (plainText) {
            SelectItemsIterator<SelectItem> items = RenderKitUtils.getSelectItems(context, component);
            items.next();
            plainText = !items.hasNext();
        }

        ResponseWriter writer = context.getResponseWriter();
        assert (writer != null);

        if (logger.isLoggable(Level.FINER)) {
            logger.log(Level.FINER, "Rendering 'select'");
        }
        if (plainText) {
            writer.startElement("span", component);
        } else {
            writer.startElement("select", component);
        }
        writeIdAttributeIfNecessary(context, writer, component);
        writer.writeAttribute("name", component.getClientId(context),
                "clientId");
        // render styleClass attribute if present.
        String styleClass;
        if (null !=
                (styleClass =
                        (String) component.getAttributes().get("styleClass"))) {
            writer.writeAttribute("class", styleClass, "styleClass");
        }
        if (!getMultipleText(component).equals("")) {
            writer.writeAttribute("multiple", true, "multiple");
        }

        // Determine how many option(s) we need to render, and update
        // the component's "size" attribute accordingly;  The "size"
        // attribute will be rendered as one of the "pass thru" attributes
        SelectItemsIterator<SelectItem> items = RenderKitUtils.getSelectItems(context, component);

        // render the options to a buffer now so that we can determine
        // the size
        FastStringWriter bufferedWriter = new FastStringWriter(128);
        context.setResponseWriter(writer.cloneWithWriter(bufferedWriter));
        int count = renderOptions(context, component, items);
        context.setResponseWriter(writer);
        // If "size" is *not* set explicitly, we have to default it correctly
        Integer size = (Integer) component.getAttributes().get("size");
        if (size == null || size == Integer.MIN_VALUE) {
            size = count;
        }
        writeDefaultSize(writer, size);

        RenderKitUtils.renderPassThruAttributes(context,
                writer,
                component,
                ATTRIBUTES,
                getNonOnChangeBehaviors(component));
        RenderKitUtils.renderXHTMLStyleBooleanAttributes(writer,
                component);

        RenderKitUtils.renderOnchange(context, component, false);

        // Now, write the buffered option content
        writer.write(bufferedWriter.toString());

        if (plainText) {
            writer.endElement("span");
        } else {
            writer.endElement("select");
        }

    }

    @Override
    protected boolean renderOption(FacesContext context,
                                   UIComponent component,
                                   UIComponent selectComponent,
                                   Converter converter,
                                   SelectItem curItem,
                                   Object currentSelections,
                                   Object[] submittedValues,
                                   OptionComponentInfo optionInfo) throws IOException {

        Object valuesArray;
        Object itemValue;
        String valueString = getFormattedValue(context, component,
                curItem.getValue(), converter);
        boolean containsValue;
        if (submittedValues != null) {
            containsValue = containsaValue(submittedValues);
            if (containsValue) {
                valuesArray = submittedValues;
                itemValue = valueString;
            } else {
                valuesArray = currentSelections;
                itemValue = curItem.getValue();
            }
        } else {
            valuesArray = currentSelections;
            itemValue = curItem.getValue();
        }

        boolean isSelected = isSelected(context, component, itemValue, valuesArray, converter);
        if (optionInfo.isHideNoSelection()
                && curItem.isNoSelectionOption()
                && currentSelections != null
                && !isSelected) {
            return false;
        }

        ResponseWriter writer = context.getResponseWriter();
        assert (writer != null);
        writer.writeText("\t", component, null);

        if (plainText) {
            if (curItem.isEscape()) {
                String label = curItem.getLabel();
                if (label == null) {
                    label = valueString;
                }
                writer.writeText(label, component, "label");
            } else {
                writer.write(curItem.getLabel());
            }
            writer.writeText("\n", component, null);
            return true;
        }

        writer.startElement("option", (null != selectComponent) ? selectComponent : component);
        writer.writeAttribute("value", valueString, "value");

        if (isSelected) {
            writer.writeAttribute("selected", true, "selected");
        }

        // if the component is disabled, "disabled" attribute would be rendered
        // on "select" tag, so don't render "disabled" on every option.
        if ((!optionInfo.isDisabled()) && curItem.isDisabled()) {
            writer.writeAttribute("disabled", true, "disabled");
        }

        String labelClass;
        if (optionInfo.isDisabled() || curItem.isDisabled()) {
            labelClass = optionInfo.getDisabledClass();
        } else {
            labelClass = optionInfo.getEnabledClass();
        }
        if (labelClass != null) {
            writer.writeAttribute("class", labelClass, "labelClass");
        }

        if (curItem.isEscape()) {
            String label = curItem.getLabel();
            if (label == null) {
                label = valueString;
            }
            writer.writeText(label, component, "label");
        } else {
            writer.write(curItem.getLabel());
        }
        writer.endElement("option");
        writer.writeText("\n", component, null);
        return true;
    }
}

在faces-config.xml中添加:

<render-kit>
    <renderer>
        <component-family>javax.faces.SelectOne</component-family>
        <renderer-type>javax.faces.Menu</renderer-type>
        <renderer-class>ru.mycityseason.extentions.ExtendedMenuRenderer</renderer-class>
    </renderer>
</render-kit>