JSF 2: использование перечислений в отображаемом атрибуте - PullRequest
19 голосов
/ 16 января 2011

Есть ли способ декларативно проверить, имеет ли перечисление указанное значение. Например:

<h:graphicImage name="error.png" library="images" 
  rendered="#{viewController.current.status == Status.ERROR}" />

Немного утомительно определять метод в управляемом beand, который проверяет это для каждого значения перечисления, например,

public boolean isStateIsError() {
  return current.getStatus() == Status.ERROR;
}

Есть ли более короткий / лучший способ сделать это?

Ответы [ 5 ]

35 голосов
/ 16 января 2011

До EL 3.0 невозможно импортировать перечисления в области EL. Однако вы можете просто рассматривать и сравнивать их как строки, то есть значение константы перечисления должно быть заключено в кавычки, как показано ниже.

<h:graphicImage name="error.png" library="images" 
  rendered="#{viewController.current.status eq 'ERROR'}" />
7 голосов
/ 27 сентября 2012

Я знаю, что этот вопрос немного старше, но у меня возникла та же проблема, и я нашел другое решение, которым хочу поделиться:

Создайте пользовательский EL-Resolver и используйте перечисления и java-константы в качестве объектов в jsf el:

<h:graphicImage name="error.png" library="images"  
      rendered="#{viewController.current.status == Status.ERROR}" />

Но прежде чем вы сможете использовать перечисления таким образом, вы должны сделать 3 шага.

1. шаг - скопируйте этот класс и замените «MY_ENUM» через ваш enumClass (в приведенном выше примере это будет «Status»)

public class EnumCache {
    private Map<String, Object>  propertCache = new HashMap<String, Object>();
    private Map<String, Class>  baseCache = new HashMap<String, Class>();
    private static EnumCache staticEnumCache = null;

    public static EnumCache instance() {
        if (staticEnumCache == null) { staticEnumCache = new EnumCache(); }
        return staticEnumCache;
    }
    private EnumCache() {
        List<Class<?>> classes = new ArrayList<Class<?>>();
        classes.add(MY_ENUM.class);

        for(Class clazz : classes) {
            try {
                baseCache.put(clazz.getSimpleName(), clazz);
                Method m = clazz.getMethod("values", (Class[]) null);
                Enum<?>[] valueList = (Enum[]) m.invoke(null, (Object[]) null);
                for (Enum<?> en : valueList) {
                    propertCache.put(clazz.getSimpleName() + "." + en.name(), en);
                }
            } catch (Exception e) {
                System.err.println(clazz.getSimpleName(), e);
            }
        }
    }
    public Object getValueForKey(String key)  {
        return propertCache.get(key);
    }
    public Class getClassForKey(String key) {
        return baseCache.get(key);
    }
}

2. шаг - добавьте этот EnumResolver - этот класс сопоставит ваше выражение JSF с перечислением в кэше (шаг 1)

public class MyEnumResolver extends ELResolver {

    public Object getValue(ELContext context, Object base, Object property) {
        Object result = null;
        if (base == null) {
            result = EnumCache.instance().getClassForKey(property + "");
        } else if (base instanceof Class) {
            result = EnumCache.instance().getValueForKey(((Class) base).getSimpleName() + "." + property);
        }
        if (result != null) {
            context.setPropertyResolved(true);
        }
        return result;
    }

    public Class<?> getCommonPropertyType(ELContext context, Object base) {
        return null;
    }
    public Iterator<FeatureDescriptor> getFeatureDescriptors(ELContext context, Object base) {
        return null;
    }
    public Class<?> getType(ELContext context, Object base, Object property) {
        return null;
    }
    public boolean isReadOnly(ELContext context, Object base, Object property) {
        return false;
    }
    public void setValue(ELContext context, Object base, Object property, Object arg3) {
    }
}

3. шаг - зарегистрируйте EnumResolver в face-config.xml

<faces-config>
    <application>
        <el-resolver>com.asd.MyEnumResolver</el-resolver>
    </application>
</faces-config>

Примечание: Если вы хотите получить доступ к своим Java-константам таким образом, вам просто нужно расширить конструктор класса enumCache. Этот (непроверенный) пример должен работать:

baseCache.put(CLASS_WITH_CONSTANTS.getSimpleName(), clazz);
for (Field field : CLASS_WITH_CONSTANTS.getDeclaredFields()) {
    try {
        propertCache.put(CLASS_WITH_CONSTANTS.getSimpleName() + "." 
                  + field.getName(), field.get(null));
    } catch (Exception e) { }
}

Надеюсь, этот сокращенный, но работающий код может кому-нибудь помочь.


Обновление

Я вижу это преимущество:

  1. Если вы используете строки в jsf (viewController.current.status == 'ERROR_abcdefg'), вы можете неправильно ввести значение и не сможете его быстро распознать. С моим решением вы получите ошибку при загрузке файла jsf, потому что enum не может быть разрешен.

  2. Вы можете видеть в исходном коде, что "ERROR" является значением перечисления "STATUS".

  3. Когда вы сравниваете два значения в el, класс перечислений тоже будет сравниваться. Так, например, PersonState.ACTIV отличается от AccounState.ACTIV.

  4. Когда мне нужно изменить значение моего перечисления с PersonState.ACTIV на PersonState.ACTIVATED, я могу найти строку «PersonState.ACTIV» в моем исходном коде. поиск "ACTIV" будет иметь гораздо больше совпадений.

1 голос
/ 10 марта 2016

Я думаю, что это можно сделать следующим образом:

Создайте метод в вашем бине, который будет возвращать список перечислений, например

public Status[] getStatuses() {
  Status.values();
}

тогда вы можете использовать перечисление в EL следующим образом

<h:graphicImage name="error.png" library="images" 
  rendered="#{viewController.current.status == someBean.statuses[0]}" />

при условии, что порядок членов перечисления не будет изменен (например, здесь состояния [0] - ОШИБКА). Тем не менее, я бы исправил позиции так:

public Status[] getStatuses() {
  Status myStatuses = new Status [2]; // or whatever number of statuses you are going to use in UI
  myStatuses [0] = Status.ERROR;
  myStatuses [1] = Status.RUNNING;
  return myStatuses;
}

Это все еще не динамическое решение, но оно лучше, чем жесткое кодирование в EL. Это может быть особенно полезно, когда вы используете локализацию для своих статусов (перечисление значений в зависимости от локали / перевода).

1 голос
/ 28 марта 2015

Я решил аналогичную проблему, statically сбросив все ключи перечисления (которые используются в визуализированных компонентах пользовательского интерфейса) на карте, а затем я использую статический метод getByKey для преобразования значения из пользовательского интерфейса в фактическоесобственное перечисление в установщике, генерирующее исключение, если предоставленное значение недопустимо:

public enum ReportType {

    FILING("F", "Filings"),
    RESOLUTION("R", "Resolutions"),
    BASIS("B", "Bases"),
    STAFF("T", "Staff Counts"),
    COUNTS("I", "Counts");

    private String key;
    private String label;

    private static Map<String, ReportType> keyMap = new HashMap<String, ReportType>();  

    static {
        for(ReportType type : ReportType.values()) {
            keyMap.put(type.getKey(), type);
        }
    }

    private ReportType(String _key, String _label) {
        this.key = _key;
        this.label = _label;

    }

    public String getKey() {
        return this.key;
    }

    public String getLabel() {
        return this.label;
    }

    public static List<ReportType> getValueList() {
        return Arrays.asList(ReportType.values());
    }

    public static ReportType getByKey(String _key) {
        ReportType result = keyMap.get(_key);

        if(result == null) {
            throw new IllegalArgumentException("Invalid report type key: " + _key);
        }

        return result;
    }
}

В уровне пользовательского интерфейса ключ перечисления используется в качестве значения, а метка перечисления используется в качестве метки:

<f:selectItems var="rptTypeItem" value="#{reportController.allReportTypes}" 
    itemLabel="#{rptTypeItem.label}" itemValue="#{rptTypeItem.key}"/>

В managed bean я преобразую перечисление в список визуализации, используя getValueList() из перечисления:

public List<ReportType> getAllReportTypes() {
    return ReportType.getValueList();
}

Наконец, [g | s] выводит вуправляемый компонент выглядит следующим образом:

public String getReportType() {
    return this.crtRptType.getKey();
}

public void setReportType(String _val) {
    this.crtRptType = ReportType.getByKey(_val);
}
0 голосов
/ 27 января 2018

Я решаю подобную проблему, используя:

<p:graphicImage name="images/close.png" rendered="#{i.unityEnum.name().equals('DAY')}" />
...