diff options
author | Miki <miki@vaadin.com> | 2015-01-16 14:02:19 +0200 |
---|---|---|
committer | Vaadin Code Review <review@vaadin.com> | 2015-02-03 11:15:41 +0000 |
commit | a508ed7b4aa062334ad84d7967cb2bdd5d8ecc26 (patch) | |
tree | 5215ac87b3063efd9d7c8a67d5b3b008ff931b70 /server/src/com/vaadin/ui/declarative | |
parent | 293b62ab6ed402b188685027b100025597a37d00 (diff) | |
download | vaadin-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')
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 |