summaryrefslogtreecommitdiffstats
path: root/server/src/com/vaadin/ui/declarative
diff options
context:
space:
mode:
authorMiki <miki@vaadin.com>2015-01-16 14:02:19 +0200
committerVaadin Code Review <review@vaadin.com>2015-02-03 11:15:41 +0000
commita508ed7b4aa062334ad84d7967cb2bdd5d8ecc26 (patch)
tree5215ac87b3063efd9d7c8a67d5b3b008ff931b70 /server/src/com/vaadin/ui/declarative
parent293b62ab6ed402b188685027b100025597a37d00 (diff)
downloadvaadin-framework-a508ed7b4aa062334ad84d7967cb2bdd5d8ecc26.tar.gz
vaadin-framework-a508ed7b4aa062334ad84d7967cb2bdd5d8ecc26.zip
Declarative for DateFields (and related) with ISO8601 (#16313)
DesignAttributeHandler supports method names that contains some words in uppercase DesignAttributeHandler and other components now use extensible Formatter with Converters rather than static methods Change-Id: I9f68414bd4821f47ff37a26375091d154cae9a93
Diffstat (limited to 'server/src/com/vaadin/ui/declarative')
-rw-r--r--server/src/com/vaadin/ui/declarative/DesignAttributeHandler.java408
-rw-r--r--server/src/com/vaadin/ui/declarative/DesignFormatter.java348
-rw-r--r--server/src/com/vaadin/ui/declarative/converters/DesignDateConverter.java67
-rw-r--r--server/src/com/vaadin/ui/declarative/converters/DesignFormatConverter.java72
-rw-r--r--server/src/com/vaadin/ui/declarative/converters/DesignResourceConverter.java87
-rw-r--r--server/src/com/vaadin/ui/declarative/converters/DesignShortcutActionConverter.java121
-rw-r--r--server/src/com/vaadin/ui/declarative/converters/DesignToStringConverter.java115
-rw-r--r--server/src/com/vaadin/ui/declarative/converters/ShortcutKeyMapper.java151
8 files changed, 1024 insertions, 345 deletions
diff --git a/server/src/com/vaadin/ui/declarative/DesignAttributeHandler.java b/server/src/com/vaadin/ui/declarative/DesignAttributeHandler.java
index be7d023ebf..3e2c01c881 100644
--- a/server/src/com/vaadin/ui/declarative/DesignAttributeHandler.java
+++ b/server/src/com/vaadin/ui/declarative/DesignAttributeHandler.java
@@ -19,35 +19,25 @@ import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
-import java.io.File;
import java.io.Serializable;
import java.lang.reflect.Method;
-import java.text.DecimalFormat;
-import java.text.DecimalFormatSymbols;
import java.util.ArrayList;
-import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
-import java.util.Locale;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
import org.jsoup.nodes.Attribute;
import org.jsoup.nodes.Attributes;
import org.jsoup.nodes.Element;
import org.jsoup.nodes.Node;
-import com.vaadin.event.ShortcutAction;
-import com.vaadin.event.ShortcutAction.KeyCode;
-import com.vaadin.event.ShortcutAction.ModifierKey;
-import com.vaadin.server.ExternalResource;
-import com.vaadin.server.FileResource;
-import com.vaadin.server.FontAwesome;
-import com.vaadin.server.Resource;
-import com.vaadin.server.ThemeResource;
+import com.vaadin.data.util.converter.Converter;
import com.vaadin.shared.util.SharedUtil;
import com.vaadin.ui.Component;
@@ -65,8 +55,21 @@ public class DesignAttributeHandler implements Serializable {
return Logger.getLogger(DesignAttributeHandler.class.getName());
}
- private static Map<Class, AttributeCacheEntry> cache = Collections
- .synchronizedMap(new HashMap<Class, AttributeCacheEntry>());
+ private static Map<Class<?>, AttributeCacheEntry> cache = Collections
+ .synchronizedMap(new HashMap<Class<?>, AttributeCacheEntry>());
+
+ // translates string <-> object
+ private static DesignFormatter FORMATTER = new DesignFormatter();
+
+ /**
+ * Returns the currently used formatter. All primitive types and all types
+ * needed by Vaadin components are handled by that formatter.
+ *
+ * @return An instance of the formatter.
+ */
+ public static DesignFormatter getFormatter() {
+ return FORMATTER;
+ }
/**
* Clears the children and attributes of the given element
@@ -111,8 +114,8 @@ public class DesignAttributeHandler implements Serializable {
success = false;
} else {
// we have a value from design attributes, let's use that
- Object param = fromAttributeValue(
- setter.getParameterTypes()[0], value);
+ Object param = getFormatter().parse(value,
+ setter.getParameterTypes()[0]);
setter.invoke(target, param);
success = true;
}
@@ -170,7 +173,7 @@ public class DesignAttributeHandler implements Serializable {
Method getter = descriptor.getReadMethod();
Method setter = descriptor.getWriteMethod();
if (getter != null && setter != null
- && isSupported(descriptor.getPropertyType())) {
+ && getFormatter().canConvert(descriptor.getPropertyType())) {
String attribute = toAttributeName(descriptor.getName());
entry.addAttribute(attribute, getter, setter);
}
@@ -229,10 +232,9 @@ public class DesignAttributeHandler implements Serializable {
* @return the attribute value or the default value if the attribute is not
* found
*/
- @SuppressWarnings("unchecked")
public static <T> T readAttribute(String attribute, Attributes attributes,
Class<T> outputType) {
- if (!isSupported(outputType)) {
+ if (!getFormatter().canConvert(outputType)) {
throw new IllegalArgumentException("output type: "
+ outputType.getName() + " not supported");
}
@@ -241,7 +243,7 @@ public class DesignAttributeHandler implements Serializable {
} else {
try {
String value = attributes.get(attribute);
- return (T) fromAttributeValue(outputType, value);
+ return getFormatter().parse(value, outputType);
} catch (Exception e) {
throw new DesignException("Failed to read attribute "
+ attribute, e);
@@ -266,7 +268,7 @@ public class DesignAttributeHandler implements Serializable {
*/
public static <T> void writeAttribute(String attribute,
Attributes attributes, T value, T defaultValue, Class<T> inputType) {
- if (!isSupported(inputType)) {
+ if (!getFormatter().canConvert(inputType)) {
throw new IllegalArgumentException("input type: "
+ inputType.getName() + " not supported");
}
@@ -277,101 +279,6 @@ public class DesignAttributeHandler implements Serializable {
}
/**
- * Formats the given design attribute value. The method is provided to
- * ensure consistent number formatting for design attribute values
- *
- * @param number
- * the number to be formatted
- * @return the formatted number
- */
- public static String formatFloat(float number) {
- return getDecimalFormat().format(number);
- }
-
- /**
- * Formats the given design attribute value. The method is provided to
- * ensure consistent number formatting for design attribute values
- *
- * @param number
- * the number to be formatted
- * @return the formatted number
- */
- public static String formatDouble(double number) {
- return getDecimalFormat().format(number);
- }
-
- /**
- * Convert ShortcutAction to attribute string presentation
- *
- * @param shortcut
- * the shortcut action
- * @return the action as attribute string presentation
- */
- private static String formatShortcutAction(ShortcutAction shortcut) {
- StringBuilder sb = new StringBuilder();
- // handle modifiers
- if (shortcut.getModifiers() != null) {
- for (int modifier : shortcut.getModifiers()) {
- sb.append(ShortcutKeyMapper.getStringForKeycode(modifier))
- .append("-");
- }
- }
- // handle keycode
- sb.append(ShortcutKeyMapper.getStringForKeycode(shortcut.getKeyCode()));
- return sb.toString();
- }
-
- /**
- * Reads shortcut action from attribute presentation
- *
- * @param attributeValue
- * attribute presentation of shortcut action
- * @return shortcut action with keycode and modifier keys from attribute
- * value
- */
- private static ShortcutAction readShortcutAction(String attributeValue) {
- if (attributeValue.length() == 0) {
- return null;
- }
- String[] parts = attributeValue.split("-");
- // handle keycode
- String keyCodePart = parts[parts.length - 1];
- int keyCode = ShortcutKeyMapper.getKeycodeForString(keyCodePart);
- if (keyCode < 0) {
- throw new IllegalArgumentException("Invalid shortcut definition "
- + attributeValue);
- }
- // handle modifiers
- int[] modifiers = null;
- if (parts.length > 1) {
- modifiers = new int[parts.length - 1];
- }
- for (int i = 0; i < parts.length - 1; i++) {
- int modifier = ShortcutKeyMapper.getKeycodeForString(parts[i]);
- if (modifier > 0) {
- modifiers[i] = modifier;
- } else {
- throw new IllegalArgumentException(
- "Invalid shortcut definition " + attributeValue);
- }
- }
- return new ShortcutAction(null, keyCode, modifiers);
- }
-
- /**
- * Creates the decimal format used when writing attributes to the design.
- *
- * @return the decimal format
- */
- private static DecimalFormat getDecimalFormat() {
- DecimalFormatSymbols symbols = new DecimalFormatSymbols(new Locale(
- "en_US"));
- DecimalFormat fmt = new DecimalFormat("0.###", symbols);
- fmt.setGroupingUsed(false);
- return fmt;
- }
-
- /**
* Returns the design attribute name corresponding the given method name.
* For example given a method name <code>setPrimaryStyleName</code> the
* return value would be <code>primary-style-name</code>
@@ -381,6 +288,7 @@ public class DesignAttributeHandler implements Serializable {
* @return the design attribute name corresponding the given method name
*/
private static String toAttributeName(String propertyName) {
+ propertyName = removeSubsequentUppercase(propertyName);
String[] words = propertyName.split("(?<!^)(?=[A-Z])");
StringBuilder builder = new StringBuilder();
for (int i = 0; i < words.length; i++) {
@@ -393,56 +301,43 @@ public class DesignAttributeHandler implements Serializable {
}
/**
- * Parses the given attribute value to specified target type
+ * Replaces subsequent UPPERCASE strings of length 2 or more followed either
+ * by another uppercase letter or an end of string. This is to generalise
+ * handling of method names like <tt>showISOWeekNumbers</tt>.
*
- * @param targetType
- * the target type for the value
- * @param value
- * the parsed value
- * @return the object of specified target type
+ * @param param
+ * Input string.
+ * @return Input string with sequences of UPPERCASE turned into Normalcase.
*/
- private static Object fromAttributeValue(Class<?> targetType, String value) {
- if (targetType == String.class) {
- return value;
- }
- // special handling for boolean type. The attribute evaluates to true if
- // it is present and the value is not "false" or "FALSE". Thus empty
- // value evaluates to true.
- if (targetType == Boolean.TYPE || targetType == Boolean.class) {
- return parseBoolean(value);
- }
- if (targetType == Integer.TYPE || targetType == Integer.class) {
- return Integer.valueOf(value);
- }
- if (targetType == Byte.TYPE || targetType == Byte.class) {
- return Byte.valueOf(value);
- }
- if (targetType == Short.TYPE || targetType == Short.class) {
- return Short.valueOf(value);
- }
- if (targetType == Long.TYPE || targetType == Long.class) {
- return Long.valueOf(value);
- }
- if (targetType == Character.TYPE || targetType == Character.class) {
- return value.charAt(0);
- }
- if (targetType == Float.TYPE || targetType == Float.class) {
- return Float.valueOf(value);
- }
- if (targetType == Double.TYPE || targetType == Double.class) {
- return Double.valueOf(value);
- }
- if (targetType == Resource.class) {
- return parseResource(value);
- }
- if (Enum.class.isAssignableFrom(targetType)) {
- return Enum.valueOf((Class<? extends Enum>) targetType,
- value.toUpperCase());
- }
- if (targetType == ShortcutAction.class) {
- return readShortcutAction(value);
+ private static String removeSubsequentUppercase(String param) {
+ StringBuffer result = new StringBuffer();
+ // match all two-or-more caps letters lead by a non-uppercase letter
+ // followed by either a capital letter or string end
+ Pattern pattern = Pattern.compile("(^|[^A-Z])([A-Z]{2,})([A-Z]|$)");
+ Matcher matcher = pattern.matcher(param);
+ while (matcher.find()) {
+ String matched = matcher.group(2);
+ // if this is a beginning of the string, the whole matched group is
+ // written in lower case
+ if (matcher.group(1).isEmpty()) {
+ matcher.appendReplacement(result, matched.toLowerCase()
+ + matcher.group(3));
+ // otherwise the first character of the group stays uppercase,
+ // while the others are lower case
+ } else {
+ matcher.appendReplacement(
+ result,
+ matcher.group(1) + matched.substring(0, 1)
+ + matched.substring(1).toLowerCase()
+ + matcher.group(3));
+ }
+ // in both cases the uppercase letter of the next word (or string's
+ // end) is added
+ // this implies there is at least one extra lowercase letter after
+ // it to be caught by the next call to find()
}
- return null;
+ matcher.appendTail(result);
+ return result.toString();
}
/**
@@ -460,57 +355,16 @@ public class DesignAttributeHandler implements Serializable {
// value is not null. How to represent null value in attributes?
return "";
}
- if (sourceType == Resource.class) {
- if (value instanceof ExternalResource) {
- return ((ExternalResource) value).getURL();
- } else if (value instanceof ThemeResource) {
- return "theme://" + ((ThemeResource) value).getResourceId();
- } else if (value instanceof FontAwesome) {
- return "font://" + ((FontAwesome) value).name();
- } else if (value instanceof FileResource) {
- String path = ((FileResource) value).getSourceFile().getPath();
- if (File.separatorChar != '/') {
- // make sure we use '/' as file separator in templates
- return path.replace(File.separatorChar, '/');
- } else {
- return path;
- }
- } else {
- getLogger().warning(
- "Unknown resource type " + value.getClass().getName());
- return null;
- }
- } else if (sourceType == Float.class || sourceType == Float.TYPE) {
- return formatFloat(((Float) value).floatValue());
- } else if (sourceType == Double.class || sourceType == Double.TYPE) {
- return formatDouble(((Double) value).doubleValue());
- } else if (sourceType == ShortcutAction.class) {
- return formatShortcutAction((ShortcutAction) value);
+ Converter<String, Object> converter = getFormatter().findConverterFor(
+ sourceType);
+ if (converter != null) {
+ return converter.convertToPresentation(value, String.class, null);
} else {
return value.toString();
}
}
/**
- * Parses the given attribute value as resource
- *
- * @param value
- * the attribute value to be parsed
- * @return resource instance based on the attribute value
- */
- private static Resource parseResource(String value) {
- if (value.startsWith("http://")) {
- return new ExternalResource(value);
- } else if (value.startsWith("theme://")) {
- return new ThemeResource(value.substring(8));
- } else if (value.startsWith("font://")) {
- return FontAwesome.valueOf(value.substring(7));
- } else {
- return new FileResource(new File(value));
- }
- }
-
- /**
* Returns a setter that can be used for assigning the given design
* attribute to the class
*
@@ -542,29 +396,6 @@ public class DesignAttributeHandler implements Serializable {
return cache.get(clazz).getGetter(attribute);
}
- // supported property types
- private static final List<Class<?>> supportedClasses = Arrays
- .asList(new Class<?>[] { String.class, Boolean.class,
- Integer.class, Byte.class, Short.class, Long.class,
- Character.class, Float.class, Double.class, Resource.class,
- ShortcutAction.class });
-
- /**
- * Returns true if the specified value type is supported by this class.
- * Currently the handler supports primitives, {@link Locale.class} and
- * {@link Resource.class}.
- *
- * @param valueType
- * the value type to be tested
- * @return true if the value type is supported, otherwise false
- */
- private static boolean isSupported(Class<?> valueType) {
- return valueType != null
- && (valueType.isPrimitive()
- || supportedClasses.contains(valueType) || Enum.class
- .isAssignableFrom(valueType));
- }
-
/**
* Cache object for caching supported attributes and their getters and
* setters
@@ -599,117 +430,4 @@ public class DesignAttributeHandler implements Serializable {
}
}
- /**
- * Provides mappings between shortcut keycodes and their representation in
- * design attributes
- *
- * @author Vaadin Ltd
- */
- private static class ShortcutKeyMapper implements Serializable {
-
- private static Map<Integer, String> keyCodeMap = Collections
- .synchronizedMap(new HashMap<Integer, String>());
- private static Map<String, Integer> presentationMap = Collections
- .synchronizedMap(new HashMap<String, Integer>());
-
- static {
- // map modifiers
- mapKey(ModifierKey.ALT, "alt");
- mapKey(ModifierKey.CTRL, "ctrl");
- mapKey(ModifierKey.META, "meta");
- mapKey(ModifierKey.SHIFT, "shift");
- // map keys
- mapKey(KeyCode.ENTER, "enter");
- mapKey(KeyCode.ESCAPE, "escape");
- mapKey(KeyCode.PAGE_UP, "pageup");
- mapKey(KeyCode.PAGE_DOWN, "pagedown");
- mapKey(KeyCode.TAB, "tab");
- mapKey(KeyCode.ARROW_LEFT, "left");
- mapKey(KeyCode.ARROW_UP, "up");
- mapKey(KeyCode.ARROW_RIGHT, "right");
- mapKey(KeyCode.ARROW_DOWN, "down");
- mapKey(KeyCode.BACKSPACE, "backspace");
- mapKey(KeyCode.DELETE, "delete");
- mapKey(KeyCode.INSERT, "insert");
- mapKey(KeyCode.END, "end");
- mapKey(KeyCode.HOME, "home");
- mapKey(KeyCode.F1, "f1");
- mapKey(KeyCode.F2, "f2");
- mapKey(KeyCode.F3, "f3");
- mapKey(KeyCode.F4, "f4");
- mapKey(KeyCode.F5, "f5");
- mapKey(KeyCode.F6, "f6");
- mapKey(KeyCode.F7, "f7");
- mapKey(KeyCode.F8, "f8");
- mapKey(KeyCode.F9, "f9");
- mapKey(KeyCode.F10, "f10");
- mapKey(KeyCode.F11, "f11");
- mapKey(KeyCode.F12, "f12");
- mapKey(KeyCode.NUM0, "0");
- mapKey(KeyCode.NUM1, "1");
- mapKey(KeyCode.NUM2, "2");
- mapKey(KeyCode.NUM3, "3");
- mapKey(KeyCode.NUM4, "4");
- mapKey(KeyCode.NUM5, "5");
- mapKey(KeyCode.NUM6, "6");
- mapKey(KeyCode.NUM7, "7");
- mapKey(KeyCode.NUM8, "8");
- mapKey(KeyCode.NUM9, "9");
- mapKey(KeyCode.SPACEBAR, "spacebar");
- mapKey(KeyCode.A, "a");
- mapKey(KeyCode.B, "b");
- mapKey(KeyCode.C, "c");
- mapKey(KeyCode.D, "d");
- mapKey(KeyCode.E, "e");
- mapKey(KeyCode.F, "f");
- mapKey(KeyCode.G, "g");
- mapKey(KeyCode.H, "h");
- mapKey(KeyCode.I, "i");
- mapKey(KeyCode.J, "j");
- mapKey(KeyCode.K, "k");
- mapKey(KeyCode.L, "l");
- mapKey(KeyCode.M, "m");
- mapKey(KeyCode.N, "n");
- mapKey(KeyCode.O, "o");
- mapKey(KeyCode.P, "p");
- mapKey(KeyCode.Q, "q");
- mapKey(KeyCode.R, "r");
- mapKey(KeyCode.S, "s");
- mapKey(KeyCode.T, "t");
- mapKey(KeyCode.U, "u");
- mapKey(KeyCode.V, "v");
- mapKey(KeyCode.X, "x");
- mapKey(KeyCode.Y, "y");
- mapKey(KeyCode.Z, "z");
- }
-
- private static void mapKey(int keyCode, String presentation) {
- keyCodeMap.put(keyCode, presentation);
- presentationMap.put(presentation, keyCode);
- }
-
- private static int getKeycodeForString(String attributePresentation) {
- Integer code = presentationMap.get(attributePresentation);
- return code != null ? code.intValue() : -1;
- }
-
- private static String getStringForKeycode(int keyCode) {
- return keyCodeMap.get(keyCode);
- }
- }
-
- /**
- * Converts the given string attribute value to its corresponding boolean.
- *
- * An empty string and "true" are considered to represent a true value and
- * "false" to represent a false value.
- *
- * @param booleanValue
- * the boolean value from an attribute
- * @return the parsed boolean
- */
- public static boolean parseBoolean(String booleanValue) {
- return !booleanValue.equalsIgnoreCase("false");
- }
-
} \ No newline at end of file
diff --git a/server/src/com/vaadin/ui/declarative/DesignFormatter.java b/server/src/com/vaadin/ui/declarative/DesignFormatter.java
new file mode 100644
index 0000000000..fdce563104
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/DesignFormatter.java
@@ -0,0 +1,348 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative;
+
+import java.io.Serializable;
+import java.math.BigDecimal;
+import java.text.DecimalFormat;
+import java.text.DecimalFormatSymbols;
+import java.util.Collections;
+import java.util.Date;
+import java.util.HashMap;
+import java.util.Locale;
+import java.util.Map;
+import java.util.Set;
+import java.util.TimeZone;
+
+import com.vaadin.data.util.converter.Converter;
+import com.vaadin.event.ShortcutAction;
+import com.vaadin.server.Resource;
+import com.vaadin.ui.declarative.converters.DesignDateConverter;
+import com.vaadin.ui.declarative.converters.DesignFormatConverter;
+import com.vaadin.ui.declarative.converters.DesignResourceConverter;
+import com.vaadin.ui.declarative.converters.DesignShortcutActionConverter;
+import com.vaadin.ui.declarative.converters.ShortcutKeyMapper;
+import com.vaadin.ui.declarative.converters.DesignToStringConverter;
+
+/**
+ * Class focused on flexible and consistent formatting and parsing of different
+ * values throughout reading and writing {@link Design}. An instance of this
+ * class is used by {@link DesignAttributeHandler}.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ */
+public class DesignFormatter implements Serializable {
+
+ private final Map<Class<?>, Converter<String, ?>> converterMap = Collections
+ .synchronizedMap(new HashMap<Class<?>, Converter<String, ?>>());
+
+ /**
+ * Creates the formatter with default types already mapped.
+ */
+ public DesignFormatter() {
+ mapDefaultTypes();
+ }
+
+ /**
+ * Maps default types to their converters.
+ *
+ */
+ protected void mapDefaultTypes() {
+ // numbers use standard toString/valueOf approach
+ for (Class<?> c : new Class<?>[] { Integer.class, Byte.class,
+ Short.class, Long.class, BigDecimal.class }) {
+ DesignToStringConverter<?> conv = new DesignToStringConverter(c);
+ converterMap.put(c, conv);
+ try {
+ converterMap.put((Class<?>) c.getField("TYPE").get(null), conv);
+ } catch (Exception e) {
+ ; // this will never happen
+ }
+ }
+ // booleans use a bit different converter than the standard one
+ // "false" is boolean false, everything else is boolean true
+ Converter<String, Boolean> booleanConverter = new Converter<String, Boolean>() {
+
+ @Override
+ public Boolean convertToModel(String value,
+ Class<? extends Boolean> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return !value.equalsIgnoreCase("false");
+ }
+
+ @Override
+ public String convertToPresentation(Boolean value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return String.valueOf(value.booleanValue());
+ }
+
+ @Override
+ public Class<Boolean> getModelType() {
+ return Boolean.class;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+ };
+ converterMap.put(Boolean.class, booleanConverter);
+ converterMap.put(boolean.class, booleanConverter);
+
+ // floats and doubles use formatters
+ DecimalFormatSymbols symbols = new DecimalFormatSymbols(new Locale(
+ "en_US"));
+ DecimalFormat fmt = new DecimalFormat("0.###", symbols);
+ fmt.setGroupingUsed(false);
+ converterMap.put(Float.class, new DesignFormatConverter<Float>(
+ Float.class, fmt));
+ converterMap.put(Float.TYPE, new DesignFormatConverter<Float>(
+ Float.class, fmt));
+ converterMap.put(Double.class, new DesignFormatConverter<Double>(
+ Double.class, fmt));
+ converterMap.put(Double.TYPE, new DesignFormatConverter<Double>(
+ Double.class, fmt));
+
+ // strings do nothing
+ converterMap.put(String.class, new Converter<String, String>() {
+
+ @Override
+ public String convertToModel(String value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return value;
+ }
+
+ @Override
+ public String convertToPresentation(String value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return value;
+ }
+
+ @Override
+ public Class<String> getModelType() {
+ return String.class;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+ });
+
+ // char takes the first character from the string
+ Converter<String, Character> charConverter = new DesignToStringConverter<Character>(
+ Character.class) {
+
+ @Override
+ public Character convertToModel(String value,
+ Class<? extends Character> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return value.charAt(0);
+ }
+
+ };
+ converterMap.put(Character.class, charConverter);
+ converterMap.put(Character.TYPE, charConverter);
+
+ // date conversion has its own class
+ converterMap.put(Date.class, new DesignDateConverter());
+
+ // as shortcut action and resource
+ converterMap.put(ShortcutAction.class,
+ new DesignShortcutActionConverter(ShortcutKeyMapper.DEFAULT));
+
+ converterMap.put(Resource.class, new DesignResourceConverter());
+
+ // timezones use different static method and do not use toString()
+ converterMap.put(TimeZone.class, new DesignToStringConverter<TimeZone>(
+ TimeZone.class, "getTimeZone") {
+ @Override
+ public String convertToPresentation(TimeZone value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return value.getID();
+ }
+ });
+ }
+
+ /**
+ * Adds a converter for a new type.
+ *
+ * @param converter
+ * Converter to add.
+ */
+ protected <T> void addConverter(Converter<String, T> converter) {
+ converterMap.put(converter.getModelType(), converter);
+ }
+
+ /**
+ * Adds a converter for a given type.
+ *
+ * @param type
+ * Type to convert to/from.
+ * @param converter
+ * Converter.
+ */
+ protected <T> void addConverter(Class<?> type,
+ Converter<String, ?> converter) {
+ converterMap.put(type, converter);
+ }
+
+ /**
+ * Removes the converter for given type, if it was present.
+ *
+ * @param type
+ * Type to remove converter for.
+ */
+ protected void removeConverter(Class<?> type) {
+ converterMap.remove(type);
+ }
+
+ /**
+ * Returns a set of classes that have a converter registered. This is <b>not
+ * the same</b> as the list of supported classes - subclasses of classes in
+ * this set are also supported.
+ *
+ * @return An unmodifiable set of classes that have a converter registered.
+ */
+ protected Set<Class<?>> getRegisteredClasses() {
+ return Collections.unmodifiableSet(converterMap.keySet());
+ }
+
+ /**
+ * Parses a given string as a value of given type
+ *
+ * @param value
+ * String value to convert.
+ * @param type
+ * Expected result type.
+ * @return String converted to the expected result type using a registered
+ * converter for that type.
+ */
+ public <T> T parse(String value, Class<? extends T> type) {
+ Converter<String, T> converter = findConverterFor(type);
+ if (converter != null) {
+ return converter.convertToModel(value, type, null);
+ } else {
+ return null;
+ }
+ }
+
+ /**
+ * Finds a formatter for a given object and attempts to format it.
+ *
+ * @param object
+ * Object to format.
+ * @return String representation of the object, as returned by the
+ * registered converter.
+ */
+ public String format(Object object) {
+ return format(object, object == null ? Object.class : object.getClass());
+ }
+
+ /**
+ * Formats an object according to a converter suitable for a given type.
+ *
+ * @param object
+ * Object to format.
+ * @param type
+ * Type of the object.
+ * @return String representation of the object, as returned by the
+ * registered converter.
+ */
+ public <T> String format(T object, Class<? extends T> type) {
+ if (object == null) {
+ return null;
+ } else {
+ return findConverterFor(object.getClass()).convertToPresentation(
+ object, String.class, null);
+ }
+ }
+
+ /**
+ * Checks whether or not a value of a given type can be converted. If a
+ * converter for a superclass is found, this will return true.
+ *
+ * @param type
+ * Type to check.
+ * @return <b>true</b> when either a given type or its supertype has a
+ * converter, <b>false</b> otherwise.
+ */
+ public boolean canConvert(Class<?> type) {
+ return findConverterFor(type) != null;
+ }
+
+ /**
+ * Finds a converter for a given type. May return a converter for a
+ * superclass instead, if one is found and {@code strict} is false.
+ *
+ * @param sourceType
+ * Type to find a converter for.
+ * @param strict
+ * Whether or not search should be strict. When this is
+ * <b>false</b>, a converter for a superclass of given type may
+ * be returned.
+ * @return A valid converter for a given type or its supertype, <b>null</b>
+ * if it was not found.
+ */
+ @SuppressWarnings("unchecked")
+ protected <T> Converter<String, T> findConverterFor(
+ Class<? extends T> sourceType, boolean strict) {
+ if (sourceType.isEnum()) {
+ // enums can be read in lowercase
+ return new DesignToStringConverter<T>(sourceType) {
+
+ @Override
+ public T convertToModel(String value,
+ Class<? extends T> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return super.convertToModel(value.toUpperCase(),
+ targetType, locale);
+ }
+ };
+ } else if (converterMap.containsKey(sourceType)) {
+ return ((Converter<String, T>) converterMap.get(sourceType));
+ } else if (!strict) {
+ for (Class<?> supported : converterMap.keySet()) {
+ if (supported.isAssignableFrom(sourceType)) {
+ return ((Converter<String, T>) converterMap.get(supported));
+ }
+ }
+ }
+ return null;
+ }
+
+ /**
+ * Finds a converter for a given type. May return a converter for a
+ * superclass instead, if one is found.
+ *
+ * @param sourceType
+ * Type to find a converter for.
+ * @return A valid converter for a given type or its subtype, <b>null</b> if
+ * it was not found.
+ */
+ protected <T> Converter<String, T> findConverterFor(
+ Class<? extends T> sourceType) {
+ return findConverterFor(sourceType, false);
+ }
+
+}
diff --git a/server/src/com/vaadin/ui/declarative/converters/DesignDateConverter.java b/server/src/com/vaadin/ui/declarative/converters/DesignDateConverter.java
new file mode 100644
index 0000000000..d2d63ad16e
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/converters/DesignDateConverter.java
@@ -0,0 +1,67 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative.converters;
+
+import java.text.ParseException;
+import java.text.SimpleDateFormat;
+import java.util.Date;
+import java.util.Locale;
+
+import com.vaadin.data.util.converter.Converter;
+import com.vaadin.ui.declarative.DesignAttributeHandler;
+
+/**
+ * A date converter to be used by {@link DesignAttributeHandler}. Provides
+ * ISO-compliant way of storing date and time.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ */
+public class DesignDateConverter implements Converter<String, Date> {
+
+ @Override
+ public Date convertToModel(String value, Class<? extends Date> targetType,
+ Locale locale) throws Converter.ConversionException {
+ for (String pattern : new String[] { "yyyy-MM-dd HH:mm:ssZ",
+ "yyyy-MM-dd HH:mm:ss", "yyyy-MM-dd HH:mm", "yyyy-MM-dd HH",
+ "yyyy-MM-dd", "yyyy-MM", "yyyy" }) {
+ try {
+ return new SimpleDateFormat(pattern).parse(value);
+ } catch (ParseException e) {
+ // not parseable, ignore and try another format
+ }
+ }
+ return null;
+ }
+
+ @Override
+ public String convertToPresentation(Date value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return new SimpleDateFormat("yyyy-MM-dd HH:mm:ssZ").format(value);
+ }
+
+ @Override
+ public Class<Date> getModelType() {
+ return Date.class;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+}
diff --git a/server/src/com/vaadin/ui/declarative/converters/DesignFormatConverter.java b/server/src/com/vaadin/ui/declarative/converters/DesignFormatConverter.java
new file mode 100644
index 0000000000..e9b26fce0b
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/converters/DesignFormatConverter.java
@@ -0,0 +1,72 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative.converters;
+
+import java.text.Format;
+import java.text.ParseException;
+import java.util.Locale;
+
+import com.vaadin.data.util.converter.Converter;
+
+/**
+ * Converter based on Java Formats rather than static methods.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ * @param <TYPE>
+ * Type of the object to format.
+ */
+public class DesignFormatConverter<TYPE> implements Converter<String, TYPE> {
+
+ private final Format format;
+ private final Class<? extends TYPE> type;
+
+ /**
+ * Constructs an instance of the converter.
+ */
+ public DesignFormatConverter(Class<? extends TYPE> type, Format format) {
+ this.type = type;
+ this.format = format;
+ }
+
+ @Override
+ public TYPE convertToModel(String value, Class<? extends TYPE> targetType,
+ Locale locale) throws Converter.ConversionException {
+ try {
+ return targetType.cast(this.format.parseObject(value));
+ } catch (ParseException e) {
+ throw new Converter.ConversionException(e);
+ }
+ }
+
+ @Override
+ public String convertToPresentation(TYPE value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ return this.format.format(value);
+ }
+
+ @Override
+ public Class<TYPE> getModelType() {
+ return (Class<TYPE>) this.type;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+}
diff --git a/server/src/com/vaadin/ui/declarative/converters/DesignResourceConverter.java b/server/src/com/vaadin/ui/declarative/converters/DesignResourceConverter.java
new file mode 100644
index 0000000000..70e46b8e7f
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/converters/DesignResourceConverter.java
@@ -0,0 +1,87 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative.converters;
+
+import java.io.File;
+import java.util.Locale;
+
+import com.vaadin.data.util.converter.Converter;
+import com.vaadin.server.ExternalResource;
+import com.vaadin.server.FileResource;
+import com.vaadin.server.FontAwesome;
+import com.vaadin.server.Resource;
+import com.vaadin.server.ThemeResource;
+import com.vaadin.ui.declarative.DesignAttributeHandler;
+
+/**
+ * A converter for {@link Resource} implementations supported by
+ * {@link DesignAttributeHandler}.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ */
+public class DesignResourceConverter implements Converter<String, Resource> {
+
+ @Override
+ public Resource convertToModel(String value,
+ Class<? extends Resource> targetType, Locale locale)
+ throws Converter.ConversionException {
+ if (value.startsWith("http://")) {
+ return new ExternalResource(value);
+ } else if (value.startsWith("theme://")) {
+ return new ThemeResource(value.substring(8));
+ } else if (value.startsWith("font://")) {
+ return FontAwesome.valueOf(value.substring(7));
+ } else {
+ return new FileResource(new File(value));
+ }
+ }
+
+ @Override
+ public String convertToPresentation(Resource value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ if (value instanceof ExternalResource) {
+ return ((ExternalResource) value).getURL();
+ } else if (value instanceof ThemeResource) {
+ return "theme://" + ((ThemeResource) value).getResourceId();
+ } else if (value instanceof FontAwesome) {
+ return "font://" + ((FontAwesome) value).name();
+ } else if (value instanceof FileResource) {
+ String path = ((FileResource) value).getSourceFile().getPath();
+ if (File.separatorChar != '/') {
+ // make sure we use '/' as file separator in templates
+ return path.replace(File.separatorChar, '/');
+ } else {
+ return path;
+ }
+ } else {
+ throw new Converter.ConversionException("unknown Resource type - "
+ + value.getClass().getName());
+ }
+ }
+
+ @Override
+ public Class<Resource> getModelType() {
+ return Resource.class;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+}
diff --git a/server/src/com/vaadin/ui/declarative/converters/DesignShortcutActionConverter.java b/server/src/com/vaadin/ui/declarative/converters/DesignShortcutActionConverter.java
new file mode 100644
index 0000000000..d9d84a1263
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/converters/DesignShortcutActionConverter.java
@@ -0,0 +1,121 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative.converters;
+
+import java.util.Locale;
+
+import com.vaadin.data.util.converter.Converter;
+import com.vaadin.event.ShortcutAction;
+
+/**
+ * Converter for {@link ShortcutActions}.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ */
+public class DesignShortcutActionConverter implements
+ Converter<String, ShortcutAction> {
+
+ /**
+ * Default instance of the shortcut key mapper.
+ */
+ private final ShortcutKeyMapper keyMapper;
+
+ /**
+ * Constructs the converter with given key mapper.
+ *
+ * @param mapper
+ * Key mapper to use.
+ */
+ public DesignShortcutActionConverter(ShortcutKeyMapper mapper) {
+ keyMapper = mapper;
+ }
+
+ @Override
+ public ShortcutAction convertToModel(String value,
+ Class<? extends ShortcutAction> targetType, Locale locale)
+ throws Converter.ConversionException {
+ if (value.length() == 0) {
+ return null;
+ }
+ String[] data = value.split(" ", 2);
+
+ String[] parts = data[0].split("-");
+ // handle keycode
+ String keyCodePart = parts[parts.length - 1];
+ int keyCode = getKeyMapper().getKeycodeForString(keyCodePart);
+ if (keyCode < 0) {
+ throw new IllegalArgumentException("Invalid shortcut definition "
+ + value);
+ }
+ // handle modifiers
+ int[] modifiers = null;
+ if (parts.length > 1) {
+ modifiers = new int[parts.length - 1];
+ }
+ for (int i = 0; i < parts.length - 1; i++) {
+ int modifier = getKeyMapper().getKeycodeForString(parts[i]);
+ if (modifier > 0) {
+ modifiers[i] = modifier;
+ } else {
+ throw new IllegalArgumentException(
+ "Invalid shortcut definition " + value);
+ }
+ }
+ return new ShortcutAction(data.length == 2 ? data[1] : null, keyCode,
+ modifiers);
+ }
+
+ @Override
+ public String convertToPresentation(ShortcutAction value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ StringBuilder sb = new StringBuilder();
+ // handle modifiers
+ if (value.getModifiers() != null) {
+ for (int modifier : value.getModifiers()) {
+ sb.append(getKeyMapper().getStringForKeycode(modifier)).append(
+ "-");
+ }
+ }
+ // handle keycode
+ sb.append(getKeyMapper().getStringForKeycode(value.getKeyCode()));
+ if (value.getCaption() != null) {
+ sb.append(" ").append(value.getCaption());
+ }
+ return sb.toString();
+ }
+
+ @Override
+ public Class<ShortcutAction> getModelType() {
+ return ShortcutAction.class;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+ /**
+ * Returns the currently used key mapper.
+ *
+ * @return Key mapper.
+ */
+ public ShortcutKeyMapper getKeyMapper() {
+ return keyMapper;
+ }
+
+}
diff --git a/server/src/com/vaadin/ui/declarative/converters/DesignToStringConverter.java b/server/src/com/vaadin/ui/declarative/converters/DesignToStringConverter.java
new file mode 100644
index 0000000000..d80119bea1
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/converters/DesignToStringConverter.java
@@ -0,0 +1,115 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative.converters;
+
+import java.lang.reflect.InvocationTargetException;
+import java.util.Locale;
+
+import com.vaadin.data.util.converter.Converter;
+import com.vaadin.ui.declarative.DesignAttributeHandler;
+
+/**
+ * Utility class for {@link DesignAttributeHandler} that deals with converting
+ * various types to string.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ * @param <TYPE>
+ * Type of the data being converted.
+ */
+public class DesignToStringConverter<TYPE> implements Converter<String, TYPE> {
+
+ private final Class<? extends TYPE> type;
+
+ private final String staticMethodName;
+
+ /**
+ * A string that corresponds to how a null value is stored.
+ */
+ public static final String NULL_VALUE_REPRESENTATION = "";
+
+ /**
+ * Constructs the converter for a given type. Implicitly requires that a
+ * static method {@code valueOf(String)} is present in the type to do the
+ * conversion.
+ *
+ * @param type
+ * Type of values to convert.
+ */
+ public DesignToStringConverter(Class<? extends TYPE> type) {
+ this(type, "valueOf");
+ }
+
+ /**
+ * Constructs the converter for a given type, giving the name of the public
+ * static method that does the conversion from String.
+ *
+ * @param type
+ * Type to convert.
+ * @param staticMethodName
+ * Method to call when converting from String to this type. This
+ * must be public and static method that returns an object of
+ * passed type.
+ */
+ public DesignToStringConverter(Class<? extends TYPE> type, String staticMethodName) {
+ this.type = type;
+ this.staticMethodName = staticMethodName;
+ }
+
+ @Override
+ public TYPE convertToModel(String value, Class<? extends TYPE> targetType,
+ Locale locale) throws Converter.ConversionException {
+ try {
+ return type.cast(type
+ .getMethod(this.staticMethodName, String.class).invoke(
+ null, value));
+ } catch (IllegalAccessException e) {
+ throw new Converter.ConversionException(e);
+ } catch (IllegalArgumentException e) {
+ throw new Converter.ConversionException(e);
+ } catch (InvocationTargetException e) {
+ throw new Converter.ConversionException(e);
+ } catch (NoSuchMethodException e) {
+ throw new Converter.ConversionException(e);
+ } catch (SecurityException e) {
+ throw new Converter.ConversionException(e);
+ } catch (RuntimeException e) {
+ throw new Converter.ConversionException(e);
+ }
+ }
+
+ @Override
+ public String convertToPresentation(TYPE value,
+ Class<? extends String> targetType, Locale locale)
+ throws Converter.ConversionException {
+ if (value == null) {
+ return NULL_VALUE_REPRESENTATION;
+ } else {
+ return value.toString();
+ }
+ }
+
+ @Override
+ public Class<TYPE> getModelType() {
+ return (Class<TYPE>) this.type;
+ }
+
+ @Override
+ public Class<String> getPresentationType() {
+ return String.class;
+ }
+
+}
diff --git a/server/src/com/vaadin/ui/declarative/converters/ShortcutKeyMapper.java b/server/src/com/vaadin/ui/declarative/converters/ShortcutKeyMapper.java
new file mode 100644
index 0000000000..46c38ce0e0
--- /dev/null
+++ b/server/src/com/vaadin/ui/declarative/converters/ShortcutKeyMapper.java
@@ -0,0 +1,151 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.ui.declarative.converters;
+
+import java.io.Serializable;
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.Map;
+
+import com.vaadin.event.ShortcutAction.KeyCode;
+import com.vaadin.event.ShortcutAction.ModifierKey;
+
+/**
+ * Provides mappings between shortcut keycodes and their representation in
+ * design attributes. Contains a default framework implementation as a field.
+ *
+ * @since 7.4
+ * @author Vaadin Ltd
+ */
+public interface ShortcutKeyMapper extends Serializable {
+
+ /**
+ * Gets the key code for a given string.
+ *
+ * @param attributePresentation
+ * String
+ * @return Key code.
+ */
+ public int getKeycodeForString(String attributePresentation);
+
+ /**
+ * Returns a string for a given key code.
+ *
+ * @param keyCode
+ * Key code.
+ * @return String.
+ */
+ public String getStringForKeycode(int keyCode);
+
+ /**
+ * An instance of a default keymapper.
+ */
+ public static final ShortcutKeyMapper DEFAULT = new ShortcutKeyMapper() {
+
+ private final Map<Integer, String> keyCodeMap = Collections
+ .synchronizedMap(new HashMap<Integer, String>());
+ private final Map<String, Integer> presentationMap = Collections
+ .synchronizedMap(new HashMap<String, Integer>());
+
+ {
+ // map modifiers
+ mapKey(ModifierKey.ALT, "alt");
+ mapKey(ModifierKey.CTRL, "ctrl");
+ mapKey(ModifierKey.META, "meta");
+ mapKey(ModifierKey.SHIFT, "shift");
+ // map keys
+ mapKey(KeyCode.ENTER, "enter");
+ mapKey(KeyCode.ESCAPE, "escape");
+ mapKey(KeyCode.PAGE_UP, "pageup");
+ mapKey(KeyCode.PAGE_DOWN, "pagedown");
+ mapKey(KeyCode.TAB, "tab");
+ mapKey(KeyCode.ARROW_LEFT, "left");
+ mapKey(KeyCode.ARROW_UP, "up");
+ mapKey(KeyCode.ARROW_RIGHT, "right");
+ mapKey(KeyCode.ARROW_DOWN, "down");
+ mapKey(KeyCode.BACKSPACE, "backspace");
+ mapKey(KeyCode.DELETE, "delete");
+ mapKey(KeyCode.INSERT, "insert");
+ mapKey(KeyCode.END, "end");
+ mapKey(KeyCode.HOME, "home");
+ mapKey(KeyCode.F1, "f1");
+ mapKey(KeyCode.F2, "f2");
+ mapKey(KeyCode.F3, "f3");
+ mapKey(KeyCode.F4, "f4");
+ mapKey(KeyCode.F5, "f5");
+ mapKey(KeyCode.F6, "f6");
+ mapKey(KeyCode.F7, "f7");
+ mapKey(KeyCode.F8, "f8");
+ mapKey(KeyCode.F9, "f9");
+ mapKey(KeyCode.F10, "f10");
+ mapKey(KeyCode.F11, "f11");
+ mapKey(KeyCode.F12, "f12");
+ mapKey(KeyCode.NUM0, "0");
+ mapKey(KeyCode.NUM1, "1");
+ mapKey(KeyCode.NUM2, "2");
+ mapKey(KeyCode.NUM3, "3");
+ mapKey(KeyCode.NUM4, "4");
+ mapKey(KeyCode.NUM5, "5");
+ mapKey(KeyCode.NUM6, "6");
+ mapKey(KeyCode.NUM7, "7");
+ mapKey(KeyCode.NUM8, "8");
+ mapKey(KeyCode.NUM9, "9");
+ mapKey(KeyCode.SPACEBAR, "spacebar");
+ mapKey(KeyCode.A, "a");
+ mapKey(KeyCode.B, "b");
+ mapKey(KeyCode.C, "c");
+ mapKey(KeyCode.D, "d");
+ mapKey(KeyCode.E, "e");
+ mapKey(KeyCode.F, "f");
+ mapKey(KeyCode.G, "g");
+ mapKey(KeyCode.H, "h");
+ mapKey(KeyCode.I, "i");
+ mapKey(KeyCode.J, "j");
+ mapKey(KeyCode.K, "k");
+ mapKey(KeyCode.L, "l");
+ mapKey(KeyCode.M, "m");
+ mapKey(KeyCode.N, "n");
+ mapKey(KeyCode.O, "o");
+ mapKey(KeyCode.P, "p");
+ mapKey(KeyCode.Q, "q");
+ mapKey(KeyCode.R, "r");
+ mapKey(KeyCode.S, "s");
+ mapKey(KeyCode.T, "t");
+ mapKey(KeyCode.U, "u");
+ mapKey(KeyCode.V, "v");
+ mapKey(KeyCode.X, "x");
+ mapKey(KeyCode.Y, "y");
+ mapKey(KeyCode.Z, "z");
+ }
+
+ private void mapKey(int keyCode, String presentation) {
+ keyCodeMap.put(keyCode, presentation);
+ presentationMap.put(presentation, keyCode);
+ }
+
+ @Override
+ public int getKeycodeForString(String attributePresentation) {
+ Integer code = presentationMap.get(attributePresentation);
+ return code != null ? code.intValue() : -1;
+ }
+
+ @Override
+ public String getStringForKeycode(int keyCode) {
+ return keyCodeMap.get(keyCode);
+ }
+
+ };
+} \ No newline at end of file