diff options
author | Leif Åstrand <leif@vaadin.com> | 2015-08-29 15:53:20 +0300 |
---|---|---|
committer | Leif Åstrand <leif@vaadin.com> | 2015-09-09 18:42:26 +0300 |
commit | 22dfc6231cce451d171a1e6204dd75e815423cad (patch) | |
tree | f50f509731a955fa48d2b75f53e945d7247f1609 /server/src | |
parent | 8b9cb7b88e7bb442c058aa44bad454ad45460fec (diff) | |
download | vaadin-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.java | 71 |
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 |