summaryrefslogtreecommitdiffstats
path: root/server/src
diff options
context:
space:
mode:
authorLeif Åstrand <leif@vaadin.com>2015-08-29 15:53:20 +0300
committerLeif Åstrand <leif@vaadin.com>2015-09-09 18:42:26 +0300
commit22dfc6231cce451d171a1e6204dd75e815423cad (patch)
treef50f509731a955fa48d2b75f53e945d7247f1609 /server/src
parent8b9cb7b88e7bb442c058aa44bad454ad45460fec (diff)
downloadvaadin-framework-22dfc6231cce451d171a1e6204dd75e815423cad.tar.gz
vaadin-framework-22dfc6231cce451d171a1e6204dd75e815423cad.zip
Support custom Enum.toString in converter (#17301)
Change-Id: Icd3c164fb252bd048ffcd953f967a9c7acdc4514
Diffstat (limited to 'server/src')
-rw-r--r--server/src/com/vaadin/data/util/converter/StringToEnumConverter.java71
1 files changed, 46 insertions, 25 deletions
diff --git a/server/src/com/vaadin/data/util/converter/StringToEnumConverter.java b/server/src/com/vaadin/data/util/converter/StringToEnumConverter.java
index 96fdc7baa8..7b2f43f972 100644
--- a/server/src/com/vaadin/data/util/converter/StringToEnumConverter.java
+++ b/server/src/com/vaadin/data/util/converter/StringToEnumConverter.java
@@ -22,8 +22,15 @@ import java.util.Locale;
* A converter that converts from {@link String} to an {@link Enum} and back.
* <p>
* Designed to provide nice human readable strings for {@link Enum} classes
- * where the constants are named SOME_UPPERCASE_WORDS. Will not necessarily work
- * correctly for other cases.
+ * conforming to one of these patterns:
+ * <ul>
+ * <li>The constants are named SOME_UPPERCASE_WORDS and there's no toString
+ * implementation.</li>
+ * <li>toString() always returns the same human readable string that is not the
+ * same as its name() value. Each constant in the enum type returns a distinct
+ * toString() value.</li>
+ * </ul>
+ * Will not necessarily work correctly for other cases.
* </p>
*
* @author Vaadin Ltd
@@ -63,26 +70,35 @@ public class StringToEnumConverter implements Converter<String, Enum> {
locale = Locale.getDefault();
}
- // Foo -> FOO
- // Foo bar -> FOO_BAR
- String result = value.replace(" ", "_").toUpperCase(locale);
- try {
- return Enum.valueOf(enumType, result);
- } catch (Exception ee) {
- // There was no match. Try to compare the available values to see if
- // the constant is using something else than all upper case
- try {
- EnumSet<T> set = EnumSet.allOf(enumType);
- for (T e : set) {
- if (e.name().toUpperCase(locale).equals(result)) {
- return e;
- }
+ if (!enumType.isEnum()) {
+ throw new ConversionException(enumType.getName()
+ + " is not an enum type");
+ }
+
+ // First test for the human-readable value since that's the more likely
+ // input
+ String upperCaseValue = value.toUpperCase(locale);
+ T match = null;
+ for (T e : EnumSet.allOf(enumType)) {
+ String upperCase = enumToString(e, locale).toUpperCase(locale);
+ if (upperCase.equals(upperCaseValue)) {
+ if (match != null) {
+ throw new ConversionException("Both " + match.name()
+ + " and " + e.name()
+ + " are matching the input string " + value);
}
- } catch (Exception e) {
+ match = e;
}
+ }
- // Fallback did not work either, re-throw original exception so
- // user knows what went wrong
+ if (match != null) {
+ return match;
+ }
+
+ // Then fall back to using a strict match based on name()
+ try {
+ return Enum.valueOf(enumType, upperCaseValue);
+ } catch (Exception ee) {
throw new ConversionException(ee);
}
}
@@ -107,12 +123,17 @@ public class StringToEnumConverter implements Converter<String, Enum> {
}
String enumString = value.toString();
- // FOO -> Foo
- // FOO_BAR -> Foo bar
- // _FOO -> _foo
- String result = enumString.substring(0, 1).toUpperCase(locale);
- result += enumString.substring(1).toLowerCase(locale).replace('_', ' ');
- return result;
+ if (enumString.equals(value.name())) {
+ // FOO -> Foo
+ // FOO_BAR -> Foo bar
+ // _FOO -> _foo
+ String result = enumString.substring(0, 1).toUpperCase(locale);
+ result += enumString.substring(1).toLowerCase(locale)
+ .replace('_', ' ');
+ return result;
+ } else {
+ return enumString;
+ }
}
@Override