summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDenis Anisimov <denis@vaadin.com>2016-10-25 17:15:35 +0300
committerDenis Anisimov <denis@vaadin.com>2016-10-26 12:32:16 +0300
commitf981521a52d4ee386b6d2ba5133fd1c1cd0c5450 (patch)
tree27f420472e762ccd468d413a4435a57ed3e49517
parent3d2c66fe402ac109ef121b8517860c6dcb770d3d (diff)
downloadvaadin-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.java2
-rw-r--r--server/src/main/java/com/vaadin/data/Binder.java116
-rw-r--r--server/src/test/java/com/vaadin/data/BinderTest.java63
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