diff options
author | Denis Anisimov <denis@vaadin.com> | 2016-10-25 17:15:35 +0300 |
---|---|---|
committer | Denis Anisimov <denis@vaadin.com> | 2016-10-26 12:32:16 +0300 |
commit | f981521a52d4ee386b6d2ba5133fd1c1cd0c5450 (patch) | |
tree | 27f420472e762ccd468d413a4435a57ed3e49517 | |
parent | 3d2c66fe402ac109ef121b8517860c6dcb770d3d (diff) | |
download | vaadin-framework-f981521a52d4ee386b6d2ba5133fd1c1cd0c5450.tar.gz vaadin-framework-f981521a52d4ee386b6d2ba5133fd1c1cd0c5450.zip |
Disable default null representation one way converter.
Fixes vaadin/framework8-issues/#404
Change-Id: I9e07a7de5f67bbd7a5a59cf10cc1a8579cdfbbad
-rw-r--r-- | server/src/main/java/com/vaadin/data/BeanBinder.java | 2 | ||||
-rw-r--r-- | server/src/main/java/com/vaadin/data/Binder.java | 116 | ||||
-rw-r--r-- | server/src/test/java/com/vaadin/data/BinderTest.java | 63 |
3 files changed, 161 insertions, 20 deletions
diff --git a/server/src/main/java/com/vaadin/data/BeanBinder.java b/server/src/main/java/com/vaadin/data/BeanBinder.java index cc45e93c2d..ab87643451 100644 --- a/server/src/main/java/com/vaadin/data/BeanBinder.java +++ b/server/src/main/java/com/vaadin/data/BeanBinder.java @@ -184,7 +184,7 @@ public class BeanBinder<BEAN> extends Binder<BEAN> { Binding<BEAN, FIELDVALUE, Object> finalBinding; - finalBinding = withConverter(createConverter()); + finalBinding = withConverter(createConverter(), false); if (BeanUtil.checkBeanValidationAvailable()) { finalBinding = finalBinding.withValidator(new BeanValidator( diff --git a/server/src/main/java/com/vaadin/data/Binder.java b/server/src/main/java/com/vaadin/data/Binder.java index 7cd4c4a873..138ec12b00 100644 --- a/server/src/main/java/com/vaadin/data/Binder.java +++ b/server/src/main/java/com/vaadin/data/Binder.java @@ -20,6 +20,7 @@ import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.HashMap; +import java.util.IdentityHashMap; import java.util.LinkedHashSet; import java.util.List; import java.util.Locale; @@ -27,7 +28,6 @@ import java.util.Map; import java.util.Objects; import java.util.Optional; import java.util.Set; -import java.util.function.BiConsumer; import java.util.stream.Collectors; import com.vaadin.data.util.converter.Converter; @@ -471,11 +471,7 @@ public class Binder<BEAN> implements Serializable { @Override public <NEWTARGET> Binding<BEAN, FIELDVALUE, NEWTARGET> withConverter( Converter<TARGET, NEWTARGET> converter) { - checkUnbound(); - Objects.requireNonNull(converter, "converter cannot be null"); - - return getBinder().createBinding(getField(), - converterValidatorChain.chain(converter), statusHandler); + return withConverter(converter, true); } @Override @@ -499,6 +495,40 @@ public class Binder<BEAN> implements Serializable { } /** + * Implements {@link #withConverter(Converter)} method with additional + * possibility to disable (reset) default null representation converter. + * <p> + * The method {@link #withConverter(Converter)} calls this method with + * {@code true} provided as the second argument value. + * + * @see #withConverter(Converter) + * + * @param converter + * the converter to use, not null + * @param resetNullRepresentation + * if {@code true} then default null representation will be + * deactivated (if not yet), otherwise it won't be removed + * @return a new binding with the appropriate type + * @param <NEWTARGET> + * the type to convert to + * @throws IllegalStateException + * if {@code bind} has already been called + */ + protected <NEWTARGET> Binding<BEAN, FIELDVALUE, NEWTARGET> withConverter( + Converter<TARGET, NEWTARGET> converter, + boolean resetNullRepresentation) { + checkUnbound(); + Objects.requireNonNull(converter, "converter cannot be null"); + + if (resetNullRepresentation) { + getBinder().initialConverters.get(getField()).setIdentity(); + } + + return getBinder().createBinding(getField(), + converterValidatorChain.chain(converter), statusHandler); + } + + /** * Returns the {@code Binder} connected to this {@code Binding} * instance. * @@ -693,12 +723,54 @@ public class Binder<BEAN> implements Serializable { } + /** + * Converter decorator-strategy pattern to use initially provided "delegate" + * converter to execute its logic until the {@code setIdentity()} method is + * called. Once the method is called the class changes its behavior to the + * same as {@link Converter#identity()} behavior. + */ + private static class ConverterDelegate<FIELDVALUE> + implements Converter<FIELDVALUE, FIELDVALUE> { + + private Converter<FIELDVALUE, FIELDVALUE> delegate; + + private ConverterDelegate(Converter<FIELDVALUE, FIELDVALUE> converter) { + delegate = converter; + } + + @Override + public Result<FIELDVALUE> convertToModel(FIELDVALUE value, + ValueContext context) { + if (delegate == null) { + return Result.ok(value); + } else { + return delegate.convertToModel(value, context); + } + } + + @Override + public FIELDVALUE convertToPresentation(FIELDVALUE value, + ValueContext context) { + if (delegate == null) { + return value; + } else { + return delegate.convertToPresentation(value, context); + } + } + + void setIdentity() { + delegate = null; + } + } + private BEAN bean; private final Set<BindingImpl<BEAN, ?, ?>> bindings = new LinkedHashSet<>(); private final List<Validator<? super BEAN>> validators = new ArrayList<>(); + private final Map<HasValue<?>, ConverterDelegate<?>> initialConverters = new IdentityHashMap<>(); + private EventRouter eventRouter; private Label statusLabel; @@ -719,21 +791,17 @@ public class Binder<BEAN> implements Serializable { /** * Creates a new binding for the given field. The returned binding may be - * further configured before invoking <<<<<<< Upstream, based on master - * {@link Binding#bind(Function, BiConsumer) Binding.bind} which completes - * the binding. Until {@code Binding.bind} is called, the binding has no - * effect. + * further configured before invoking + * {@link Binding#bind(SerializableFunction, SerializableBiConsumer)} which + * completes the binding. Until {@code Binding.bind} is called, the binding + * has no effect. * <p> * <strong>Note:</strong> Not all {@link HasValue} implementations support * passing {@code null} as the value. For these the Binder will * automatically change {@code null} to a null representation provided by * {@link HasValue#getEmptyValue()}. This conversion is one-way only, if you * want to have a two-way mapping back to {@code null}, use - * {@link Binding#withNullRepresentation(Object))}. ======= - * {@link Binding#bind(SerializableFunction, SerializableBiConsumer) - * Binding.bind} which completes the binding. Until {@code Binding.bind} is - * called, the binding has no effect. >>>>>>> 7d541b5 Correct serializable - * issues and test that components can be serialized + * {@link Binding#withNullRepresentation(Object))}. * * @param <FIELDVALUE> * the value type of the field @@ -750,10 +818,7 @@ public class Binder<BEAN> implements Serializable { clearError(field); getStatusLabel().ifPresent(label -> label.setValue("")); - return createBinding(field, Converter.from(fieldValue -> fieldValue, - modelValue -> Objects.isNull(modelValue) ? field.getEmptyValue() - : modelValue, - exception -> exception.getMessage()), + return createBinding(field, createNullRepresentationAdapter(field), this::handleValidationStatus); } @@ -1342,4 +1407,17 @@ public class Binder<BEAN> implements Serializable { .fireEvent(new StatusChangeEvent(this, hasValidationErrors)); } + private <FIELDVALUE> Converter<FIELDVALUE, FIELDVALUE> createNullRepresentationAdapter( + HasValue<FIELDVALUE> field) { + Converter<FIELDVALUE, FIELDVALUE> nullRepresentationConverter = Converter + .from(fieldValue -> fieldValue, + modelValue -> Objects.isNull(modelValue) + ? field.getEmptyValue() : modelValue, + exception -> exception.getMessage()); + ConverterDelegate<FIELDVALUE> converter = new ConverterDelegate<>( + nullRepresentationConverter); + initialConverters.put(field, converter); + return converter; + } + } diff --git a/server/src/test/java/com/vaadin/data/BinderTest.java b/server/src/test/java/com/vaadin/data/BinderTest.java index eb686acf9b..9d8e05feb1 100644 --- a/server/src/test/java/com/vaadin/data/BinderTest.java +++ b/server/src/test/java/com/vaadin/data/BinderTest.java @@ -10,6 +10,8 @@ import org.junit.Assert; import org.junit.Before; import org.junit.Test; +import com.vaadin.data.util.converter.StringToIntegerConverter; +import com.vaadin.data.validator.NotNullValidator; import com.vaadin.tests.data.bean.Person; import com.vaadin.tests.data.bean.Sex; import com.vaadin.ui.TextField; @@ -253,4 +255,65 @@ public class BinderTest extends BinderTestBase<Binder<Person>, Person> { Assert.assertEquals("Field value was not set correctly", item.getFirstName(), nameField.getValue()); } + + @Test + public void withConverter_disablesDefaulNullRepresentation() { + Integer customNullConverter = 0; + binder.forField(ageField).withNullRepresentation("foo") + .withConverter(new StringToIntegerConverter("")) + .withConverter(age -> age, + age -> age == null ? customNullConverter : age) + .bind(Person::getSalary, Person::setSalary); + binder.bind(item); + + Assert.assertEquals(customNullConverter.toString(), + ageField.getValue()); + + Integer salary = 11; + ageField.setValue(salary.toString()); + Assert.assertEquals(11, salary.intValue()); + } + + @Test + public void beanBinder_nullRepresentationIsNotDisabled() { + BeanBinder<Person> binder = new BeanBinder<>(Person.class); + binder.forField(nameField).bind("firstName"); + + Person person = new Person(); + binder.bind(person); + + Assert.assertEquals("", nameField.getValue()); + } + + @Test + public void beanBinder_withConverter_nullRepresentationIsNotDisabled() { + String customNullPointerRepresentation = "foo"; + BeanBinder<Person> binder = new BeanBinder<>(Person.class); + binder.forField(nameField) + .withConverter(value -> value, value -> value == null + ? customNullPointerRepresentation : value) + .bind("firstName"); + + Person person = new Person(); + binder.bind(person); + + Assert.assertEquals(customNullPointerRepresentation, + nameField.getValue()); + } + + @Test + public void withValidator_doesNotDisablesDefaulNullRepresentation() { + String nullRepresentation = "foo"; + binder.forField(nameField).withNullRepresentation(nullRepresentation) + .withValidator(new NotNullValidator("")) + .bind(Person::getFirstName, Person::setFirstName); + item.setFirstName(null); + binder.bind(item); + + Assert.assertEquals(nullRepresentation, nameField.getValue()); + + String newValue = "bar"; + nameField.setValue(newValue); + Assert.assertEquals(newValue, item.getFirstName()); + } }
\ No newline at end of file |