aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/src/main/java/com/vaadin/data/Binder.java214
-rw-r--r--server/src/test/java/com/vaadin/data/BinderTest.java5
2 files changed, 124 insertions, 95 deletions
diff --git a/server/src/main/java/com/vaadin/data/Binder.java b/server/src/main/java/com/vaadin/data/Binder.java
index a34db33e51..0fbacce994 100644
--- a/server/src/main/java/com/vaadin/data/Binder.java
+++ b/server/src/main/java/com/vaadin/data/Binder.java
@@ -1,12 +1,12 @@
/*
* 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
@@ -46,30 +46,33 @@ import com.vaadin.ui.AbstractComponent;
* view, where a select component is used to pick the bean to edit.
* <p>
* Unless otherwise specified, {@code Binder} method arguments cannot be null.
- *
+ *
* @author Vaadin Ltd.
*
- * @param <T>
+ * @param <BEAN>
* the bean type
- *
+ *
* @see Binding
* @see HasValue
- *
+ *
* @since
*/
-public class Binder<T> implements Serializable {
+public class Binder<BEAN> implements Serializable {
/**
- * Represents the binding between a single field and a property.
+ * Represents the binding between a field and a data property.
+ *
+ * @param <BEAN>
+ * the bean type
+ * @param <FIELDVALUE>
+ * the value type of the field
+ * @param <TARGET>
+ * the target data type of the binding, matches the field type
+ * until a converter has been set
*
- * @param <T>
- * the item type
- * @param <V>
- * the field value type
- *
* @see Binder#forField(HasValue)
*/
- public interface Binding<T, V> extends Serializable {
+ public interface Binding<BEAN, FIELDVALUE, TARGET> extends Serializable {
/**
* Completes this binding using the given getter and setter functions
@@ -77,8 +80,8 @@ public class Binder<T> implements Serializable {
* update the field value from the property and to store the field value
* to the property, respectively.
* <p>
- * When a bean is bound with {@link Binder#bind(T)}, the field value is
- * set to the return value of the given getter. The property value is
+ * When a bean is bound with {@link Binder#bind(BEAN)}, the field value
+ * is set to the return value of the given getter. The property value is
* then updated via the given setter whenever the field value changes.
* The setter may be null; in that case the property value is never
* updated and the binding is said to be <i>read-only</i>.
@@ -90,17 +93,17 @@ public class Binder<T> implements Serializable {
* implementing user-defined conversion or validation. However, in the
* most basic use case you can simply pass a pair of method references
* to this method as follows:
- *
+ *
* <pre>
* class Person {
* public String getName() { ... }
* public void setName(String name) { ... }
* }
- *
+ *
* TextField nameField = new TextField();
* binder.forField(nameField).bind(Person::getName, Person::setName);
* </pre>
- *
+ *
* @param getter
* the function to get the value of the property to the
* field, not null
@@ -110,21 +113,23 @@ public class Binder<T> implements Serializable {
* @throws IllegalStateException
* if {@code bind} has already been called on this binding
*/
- public void bind(Function<T, V> getter, BiConsumer<T, V> setter);
+ public void bind(Function<BEAN, TARGET> getter,
+ BiConsumer<BEAN, TARGET> setter);
/**
* Adds a validator to this binding. Validators are applied, in
* registration order, when the field value is saved to the backing
* property. If any validator returns a failure, the property value is
* not updated.
- *
+ *
* @param validator
* the validator to add, not null
* @return this binding, for chaining
* @throws IllegalStateException
* if {@code bind} has already been called
*/
- public Binding<T, V> withValidator(Validator<? super V> validator);
+ public Binding<BEAN, FIELDVALUE, TARGET> withValidator(
+ Validator<? super TARGET> validator);
/**
* A convenience method to add a validator to this binding using the
@@ -133,10 +138,10 @@ public class Binder<T> implements Serializable {
* Validators are applied, in registration order, when the field value
* is saved to the backing property. If any validator returns a failure,
* the property value is not updated.
- *
+ *
* @see #withValidator(Validator)
* @see Validator#from(Predicate, String)
- *
+ *
* @param predicate
* the predicate performing validation, not null
* @param message
@@ -145,51 +150,70 @@ public class Binder<T> implements Serializable {
* @throws IllegalStateException
* if {@code bind} has already been called
*/
- public Binding<T, V> withValidator(Predicate<? super V> predicate,
- String message);
+ public Binding<BEAN, FIELDVALUE, TARGET> withValidator(
+ Predicate<? super TARGET> predicate, String message);
+
+ /**
+ * Gets the field the binding uses.
+ *
+ * @return the field for the binding
+ */
+ public HasValue<FIELDVALUE> getField();
+
}
/**
* An internal implementation of {@code Binding}.
- *
- * @param <V>
- * the value type
+ *
+ * @param <BEAN>
+ * the bean type, must match the Binder bean type
+ * @param <FIELDVALUE>
+ * the value type of the field
+ * @param <TARGET>
+ * the target data type of the binding, matches the field type
+ * until a converter has been set
*/
- protected class BindingImpl<V> implements Binding<T, V> {
+ protected static class BindingImpl<BEAN, FIELDVALUE, TARGET>
+ implements Binding<BEAN, FIELDVALUE, TARGET> {
+
+ private Binder<BEAN> binder;
- private HasValue<V> field;
+ private HasValue<FIELDVALUE> field;
private Registration onValueChange;
- private Function<T, V> getter;
- private BiConsumer<T, V> setter;
+ private Function<BEAN, TARGET> getter;
+ private BiConsumer<BEAN, TARGET> setter;
- private List<Validator<? super V>> validators = new ArrayList<>();
+ private List<Validator<? super TARGET>> validators = new ArrayList<>();
/**
* Creates a new binding associated with the given field.
- *
+ *
+ * @param binder
+ * the binder this instance is connected to
* @param field
* the field to bind
*/
- protected BindingImpl(HasValue<V> field) {
+ protected BindingImpl(Binder<BEAN> binder, HasValue<FIELDVALUE> field) {
+ this.binder = binder;
this.field = field;
}
@Override
- public void bind(Function<T, V> getter, BiConsumer<T, V> setter) {
+ public void bind(Function<BEAN, TARGET> getter,
+ BiConsumer<BEAN, TARGET> setter) {
checkUnbound();
Objects.requireNonNull(getter, "getter cannot be null");
this.getter = getter;
this.setter = setter;
- bindings.add(this);
- if (bean != null) {
- bind(bean);
- }
+ binder.bindings.add(this);
+ binder.getBean().ifPresent(this::bind);
}
@Override
- public Binding<T, V> withValidator(Validator<? super V> validator) {
+ public Binding<BEAN, FIELDVALUE, TARGET> withValidator(
+ Validator<? super TARGET> validator) {
checkUnbound();
Objects.requireNonNull(validator, "validator cannot be null");
validators.add(validator);
@@ -197,20 +221,21 @@ public class Binder<T> implements Serializable {
}
@Override
- public Binding<T, V> withValidator(Predicate<? super V> predicate,
- String message) {
+ public Binding<BEAN, FIELDVALUE, TARGET> withValidator(
+ Predicate<? super TARGET> predicate, String message) {
return withValidator(Validator.from(predicate, message));
}
- private void bind(T bean) {
+ private void bind(BEAN bean) {
setFieldValue(bean);
onValueChange = field
.addValueChangeListener(e -> storeFieldValue(bean));
}
- private List<ValidationError<V>> validate() {
+ private List<ValidationError<FIELDVALUE>> validate() {
return validators.stream()
- .map(validator -> validator.apply(field.getValue()))
+ .map(validator -> validator
+ .apply((TARGET) field.getValue()))
.filter(Result::isError)
.map(result -> new ValidationError<>(field,
result.getMessage().orElse(null)))
@@ -224,26 +249,26 @@ public class Binder<T> implements Serializable {
/**
* Sets the field value by invoking the getter function on the given
* bean.
- *
+ *
* @param bean
* the bean to fetch the property value from
*/
- private void setFieldValue(T bean) {
+ private void setFieldValue(BEAN bean) {
assert bean != null;
- field.setValue(getter.apply(bean));
+ field.setValue((FIELDVALUE) getter.apply(bean));
}
/**
* Saves the field value by invoking the setter function on the given
* bean, if the value passes all registered validators.
- *
+ *
* @param bean
* the bean to set the property value to
*/
- private void storeFieldValue(T bean) {
+ private void storeFieldValue(BEAN bean) {
assert bean != null;
if (setter != null) {
- setter.accept(bean, field.getValue());
+ setter.accept(bean, (TARGET) field.getValue());
}
}
@@ -254,19 +279,23 @@ public class Binder<T> implements Serializable {
}
}
+ @Override
+ public HasValue<FIELDVALUE> getField() {
+ return field;
+ }
}
- private T bean;
+ private BEAN bean;
- private Set<BindingImpl<?>> bindings = new LinkedHashSet<>();
+ private Set<BindingImpl<BEAN, ?, ?>> bindings = new LinkedHashSet<>();
/**
* Returns an {@code Optional} of the bean that has been bound with
* {@link #bind}, or an empty optional if a bean is not currently bound.
- *
+ *
* @return the currently bound bean if any
*/
- public Optional<T> getBean() {
+ public Optional<BEAN> getBean() {
return Optional.ofNullable(bean);
}
@@ -276,14 +305,15 @@ public class Binder<T> implements Serializable {
* {@link Binding#bind(Function, BiConsumer) Binding.bind} which completes
* the binding. Until {@code Binding.bind} is called, the binding has no
* effect.
- *
- * @param <V>
+ *
+ * @param <FIELDVALUE>
* the value type of the field
* @param field
* the field to be bound, not null
* @return the new binding
*/
- public <V> Binding<T, V> forField(HasValue<V> field) {
+ public <FIELDVALUE> Binding<BEAN, FIELDVALUE, FIELDVALUE> forField(
+ HasValue<FIELDVALUE> field) {
return createBinding(field);
}
@@ -295,8 +325,8 @@ public class Binder<T> implements Serializable {
* Use the {@link #forField(HasValue)} overload instead if you want to
* further configure the new binding.
* <p>
- * When a bean is bound with {@link Binder#bind(T)}, the field value is set
- * to the return value of the given getter. The property value is then
+ * When a bean is bound with {@link Binder#bind(BEAN)}, the field value is
+ * set to the return value of the given getter. The property value is then
* updated via the given setter whenever the field value changes. The setter
* may be null; in that case the property value is never updated and the
* binding is said to be <i>read-only</i>.
@@ -308,18 +338,18 @@ public class Binder<T> implements Serializable {
* implementing user-defined conversion or validation. However, in the most
* basic use case you can simply pass a pair of method references to this
* method as follows:
- *
+ *
* <pre>
* class Person {
* public String getName() { ... }
* public void setName(String name) { ... }
* }
- *
+ *
* TextField nameField = new TextField();
* binder.bind(nameField, Person::getName, Person::setName);
* </pre>
- *
- * @param <V>
+ *
+ * @param <FIELDVALUE>
* the value type of the field
* @param field
* the field to bind, not null
@@ -330,8 +360,9 @@ public class Binder<T> implements Serializable {
* the function to save the field value to the property or null
* if read-only
*/
- public <V> void bind(HasValue<V> field, Function<T, V> getter,
- BiConsumer<T, V> setter) {
+ public <FIELDVALUE> void bind(HasValue<FIELDVALUE> field,
+ Function<BEAN, FIELDVALUE> getter,
+ BiConsumer<BEAN, FIELDVALUE> setter) {
forField(field).bind(getter, setter);
}
@@ -343,11 +374,11 @@ public class Binder<T> implements Serializable {
* corresponding getter functions. Any changes to field values are reflected
* back to their corresponding property values of the bean as long as the
* bean is bound.
- *
+ *
* @param bean
* the bean to edit, not null
*/
- public void bind(T bean) {
+ public void bind(BEAN bean) {
Objects.requireNonNull(bean, "bean cannot be null");
unbind();
this.bean = bean;
@@ -359,17 +390,17 @@ public class Binder<T> implements Serializable {
* validation as a set of validation errors.
* <p>
* Validation is successful if the resulting set is empty.
- *
+ *
* @return the validation result.
*/
public List<ValidationError<?>> validate() {
List<ValidationError<?>> resultErrors = new ArrayList<>();
- for (BindingImpl<?> binding : bindings) {
- clearError(binding.field);
+ for (BindingImpl<BEAN, ?, ?> binding : bindings) {
+ clearError(binding.getField());
List<? extends ValidationError<?>> errors = binding.validate();
resultErrors.addAll(errors);
if (!errors.isEmpty()) {
- handleError(binding.field, errors.get(0).getMessage());
+ handleError(binding.getField(), errors.get(0).getMessage());
}
}
return resultErrors;
@@ -390,16 +421,14 @@ public class Binder<T> implements Serializable {
* Reads the bound property values from the given bean to the corresponding
* fields. The bean is not otherwise associated with this binder; in
* particular its property values are not bound to the field value changes.
- * To achieve that, use {@link #bind(T)}.
- *
+ * To achieve that, use {@link #bind(BEAN)}.
+ *
* @param bean
* the bean whose property values to read, not null
*/
- public void load(T bean) {
+ public void load(BEAN bean) {
Objects.requireNonNull(bean, "bean cannot be null");
- bindings.forEach(
-
- binding -> binding.setFieldValue(bean));
+ bindings.forEach(binding -> binding.setFieldValue(bean));
}
@@ -413,26 +442,25 @@ public class Binder<T> implements Serializable {
* @throws BindingException
* if some of the bound field values fail to validate
*/
- public void save(T bean) {
+ public void save(BEAN bean) {
Objects.requireNonNull(bean, "bean cannot be null");
- bindings.forEach(
-
- binding -> binding.storeFieldValue(bean));
-
+ bindings.forEach(binding -> binding.storeFieldValue(bean));
}
/**
* Creates a new binding with the given field.
- *
- * @param <V>
- * the field value type
+ *
+ * @param <FIELDVALUE>
+ * the value type of the field
* @param field
* the field to bind
* @return the new incomplete binding
*/
- protected <V> BindingImpl<V> createBinding(HasValue<V> field) {
+ protected <FIELDVALUE> Binding<BEAN, FIELDVALUE, FIELDVALUE> createBinding(
+ HasValue<FIELDVALUE> field) {
Objects.requireNonNull(field, "field cannot be null");
- BindingImpl<V> b = new BindingImpl<>(field);
+ BindingImpl<BEAN, FIELDVALUE, FIELDVALUE> b = new BindingImpl<>(this,
+ field);
return b;
}
@@ -441,7 +469,7 @@ public class Binder<T> implements Serializable {
* implementation clears the
* {@link AbstractComponent#setComponentError(ErrorMessage) component error}
* of the field if it is a Component, otherwise does nothing.
- *
+ *
* @param field
* the field with an invalid value
*/
@@ -456,7 +484,7 @@ public class Binder<T> implements Serializable {
* given field. The default implementation sets the
* {@link AbstractComponent#setComponentError(ErrorMessage) component error}
* of the field if it is a Component, otherwise does nothing.
- *
+ *
* @param field
* the field with the invalid value
* @param error
diff --git a/server/src/test/java/com/vaadin/data/BinderTest.java b/server/src/test/java/com/vaadin/data/BinderTest.java
index c3193603ef..38d88bed27 100644
--- a/server/src/test/java/com/vaadin/data/BinderTest.java
+++ b/server/src/test/java/com/vaadin/data/BinderTest.java
@@ -209,7 +209,7 @@ public class BinderTest {
@Test
public void bound_validatorsAreOK_noErrors() {
Binder<Person> binder = new Binder<>();
- Binding<Person, String> binding = binder.forField(nameField);
+ Binding<Person, String, String> binding = binder.forField(nameField);
binding.withValidator(Validator.alwaysPass()).bind(Person::getFirstName,
Person::setFirstName);
@@ -224,7 +224,7 @@ public class BinderTest {
@Test
public void bound_validatorsFail_errors() {
Binder<Person> binder = new Binder<>();
- Binding<Person, String> binding = binder.forField(nameField);
+ Binding<Person, String, String> binding = binder.forField(nameField);
binding.withValidator(Validator.alwaysPass());
String msg1 = "foo";
String msg2 = "bar";
@@ -261,4 +261,5 @@ public class BinderTest {
binder.bind(nameField, Person::getFirstName, Person::setFirstName);
binder.bind(p);
}
+
}