aboutsummaryrefslogtreecommitdiffstats
path: root/server/src
diff options
context:
space:
mode:
authorLeif Åstrand <legioth@gmail.com>2017-01-22 21:50:46 +0200
committerGitHub <noreply@github.com>2017-01-22 21:50:46 +0200
commit5447e7e05392483455f7df4bf2af5d5d74e5a2ae (patch)
tree5028592fea2d9f1dda0c23d1448b92b8a35dc6fd /server/src
parent19427ba7c7a51d3af61c170a7ba54137eb629ac8 (diff)
downloadvaadin-framework-5447e7e05392483455f7df4bf2af5d5d74e5a2ae.tar.gz
vaadin-framework-5447e7e05392483455f7df4bf2af5d5d74e5a2ae.zip
Add ListDataProvider shorthands for filter conversion (#8279)
Also updates ComboBox.setItems to use these new shorthands This is one of many steps towards #8245
Diffstat (limited to 'server/src')
-rw-r--r--server/src/main/java/com/vaadin/data/provider/ListDataProvider.java195
-rw-r--r--server/src/main/java/com/vaadin/ui/ComboBox.java26
-rw-r--r--server/src/test/java/com/vaadin/data/provider/DataProviderTestBase.java7
-rw-r--r--server/src/test/java/com/vaadin/data/provider/ListDataProviderTest.java45
4 files changed, 260 insertions, 13 deletions
diff --git a/server/src/main/java/com/vaadin/data/provider/ListDataProvider.java b/server/src/main/java/com/vaadin/data/provider/ListDataProvider.java
index 470339bc2a..4269ddbeff 100644
--- a/server/src/main/java/com/vaadin/data/provider/ListDataProvider.java
+++ b/server/src/main/java/com/vaadin/data/provider/ListDataProvider.java
@@ -17,14 +17,18 @@ package com.vaadin.data.provider;
import java.util.Collection;
import java.util.Comparator;
+import java.util.Locale;
import java.util.Objects;
import java.util.Optional;
import java.util.stream.Stream;
import com.vaadin.data.ValueProvider;
+import com.vaadin.server.SerializableBiPredicate;
import com.vaadin.server.SerializableComparator;
import com.vaadin.server.SerializablePredicate;
+import com.vaadin.server.SerializableSupplier;
import com.vaadin.shared.data.sort.SortDirection;
+import com.vaadin.ui.UI;
/**
* {@link DataProvider} wrapper for {@link Collection}s. This class does not
@@ -38,6 +42,15 @@ public class ListDataProvider<T>
implements AppendableFilterDataProvider<T, SerializablePredicate<T>>,
ConfigurableFilterDataProvider<T, SerializablePredicate<T>, SerializablePredicate<T>> {
+ private static final SerializableSupplier<Locale> CURRENT_LOCALE_SUPPLIER = () -> {
+ UI currentUi = UI.getCurrent();
+ if (currentUi != null) {
+ return currentUi.getLocale();
+ } else {
+ return Locale.getDefault();
+ }
+ };
+
private SerializableComparator<T> sortOrder = null;
private SerializablePredicate<T> filter;
@@ -376,4 +389,186 @@ public class ListDataProvider<T>
SerializablePredicate<T> filter2) {
return t -> filter1.test(t) && filter2.test(t);
}
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by comparing an item to the filter value provided in the query.
+ * <p>
+ * The predicate receives the item as the first parameter and the query
+ * filter value as the second parameter, and should return <code>true</code>
+ * if the corresponding item should be included. The query filter value is
+ * never <code>null</code> – all items are included without running the
+ * predicate if the query doesn't define any filter.
+ *
+ * @param predicate
+ * a predicate to use for comparing the item to the query filter,
+ * not <code>null</code>
+ *
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public <Q> DataProvider<T, Q> filteringBy(
+ SerializableBiPredicate<T, Q> predicate) {
+ Objects.requireNonNull(predicate, "Predicate cannot be null");
+
+ return convertFilter(
+ filterValue -> item -> predicate.test(item, filterValue));
+ }
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by comparing an item property value to the filter value provided in the
+ * query.
+ * <p>
+ * The predicate receives the property value as the first parameter and the
+ * query filter value as the second parameter, and should return
+ * <code>true</code> if the corresponding item should be included. The query
+ * filter value is never <code>null</code> – all items are included without
+ * running either callback if the query doesn't define any filter.
+ *
+ * @param valueProvider
+ * a value provider that gets the property value, not
+ * <code>null</code>
+ * @param predicate
+ * a predicate to use for comparing the property value to the
+ * query filter, not <code>null</code>
+ *
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public <V, Q> DataProvider<T, Q> filteringBy(
+ ValueProvider<T, V> valueProvider,
+ SerializableBiPredicate<V, Q> predicate) {
+ Objects.requireNonNull(valueProvider, "Value provider cannot be null");
+ Objects.requireNonNull(predicate, "Predicate cannot be null");
+
+ return filteringBy((item, filterValue) -> predicate
+ .test(valueProvider.apply(item), filterValue));
+ }
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by testing whether the value of a property is equals to the filter value
+ * provided in the query. Equality is tested using
+ * {@link Objects#equals(Object, Object)}.
+ *
+ * @param valueProvider
+ * a value provider that gets the property value, not
+ * <code>null</code>
+ *
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public <V> DataProvider<T, V> filteringByEquals(
+ ValueProvider<T, V> valueProvider) {
+ return filteringBy(valueProvider, Objects::equals);
+ }
+
+ private <V, Q> DataProvider<T, Q> filteringByIgnoreNull(
+ ValueProvider<T, V> valueProvider,
+ SerializableBiPredicate<V, Q> predicate) {
+ Objects.requireNonNull(predicate, "Predicate cannot be null");
+
+ return filteringBy(valueProvider,
+ (itemValue, queryFilter) -> itemValue != null
+ && predicate.test(itemValue, queryFilter));
+ }
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by a string by checking whether the lower case representation of the
+ * filter value provided in the query is a substring of the lower case
+ * representation of an item property value. The filter never passes if the
+ * item property value is <code>null</code>.
+ *
+ * @param valueProvider
+ * a value provider that gets the string property value, not
+ * <code>null</code>
+ * @param locale
+ * the locale to use for converting the strings to lower case,
+ * not <code>null</code>
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public DataProvider<T, String> filteringBySubstring(
+ ValueProvider<T, String> valueProvider, Locale locale) {
+ Objects.requireNonNull(locale, "Locale cannot be null");
+ return filteringByCaseInsensitiveString(valueProvider, String::contains,
+ () -> locale);
+ }
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by a string by checking whether the lower case representation of the
+ * filter value provided in the query is a substring of the lower case
+ * representation of an item property value. Conversion to lower case is
+ * done using the locale of the {@link UI#getCurrent() current UI} if
+ * available, or otherwise {@link Locale#getDefault() the default locale}.
+ * The filter never passes if the item property value is <code>null</code>.
+ *
+ * @param valueProvider
+ * a value provider that gets the string property value, not
+ * <code>null</code>
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public DataProvider<T, String> filteringBySubstring(
+ ValueProvider<T, String> valueProvider) {
+ return filteringByCaseInsensitiveString(valueProvider, String::contains,
+ CURRENT_LOCALE_SUPPLIER);
+ }
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by a string by checking whether the lower case representation of an item
+ * property value starts with the lower case representation of the filter
+ * value provided in the query. The filter never passes if the item property
+ * value is <code>null</code>.
+ *
+ * @param valueProvider
+ * a value provider that gets the string property value, not
+ * <code>null</code>
+ * @param locale
+ * the locale to use for converting the strings to lower case,
+ * not <code>null</code>
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public DataProvider<T, String> filteringByPrefix(
+ ValueProvider<T, String> valueProvider, Locale locale) {
+ return filteringByCaseInsensitiveString(valueProvider,
+ String::startsWith, () -> locale);
+ }
+
+ /**
+ * Wraps this data provider to create a new data provider that is filtered
+ * by a string by checking whether the lower case representation of an item
+ * property value starts with the lower case representation of the filter
+ * value provided in the query. Conversion to lower case is done using the
+ * locale of the {@link UI#getCurrent() current UI} if available, or
+ * otherwise {@link Locale#getDefault() the default locale}. The filter
+ * never passes if the item property value is <code>null</code>.
+ *
+ * @param valueProvider
+ * a value provider that gets the string property value, not
+ * <code>null</code>
+ * @return a data provider that filters accordingly, not <code>null</code>
+ */
+ public DataProvider<T, String> filteringByPrefix(
+ ValueProvider<T, String> valueProvider) {
+ return filteringByCaseInsensitiveString(valueProvider,
+ String::startsWith, CURRENT_LOCALE_SUPPLIER);
+ }
+
+ private DataProvider<T, String> filteringByCaseInsensitiveString(
+ ValueProvider<T, String> valueProvider,
+ SerializableBiPredicate<String, String> predicate,
+ SerializableSupplier<Locale> localeSupplier) {
+ // Only assert since these are only passed from our own code
+ assert predicate != null;
+ assert localeSupplier != null;
+
+ return filteringByIgnoreNull(valueProvider,
+ (itemString, filterString) -> {
+ Locale locale = localeSupplier.get();
+ assert locale != null;
+
+ return predicate.test(itemString.toLowerCase(locale),
+ filterString.toLowerCase(locale));
+ });
+ }
}
diff --git a/server/src/main/java/com/vaadin/ui/ComboBox.java b/server/src/main/java/com/vaadin/ui/ComboBox.java
index a789e2990c..20cc14b2da 100644
--- a/server/src/main/java/com/vaadin/ui/ComboBox.java
+++ b/server/src/main/java/com/vaadin/ui/ComboBox.java
@@ -134,11 +134,6 @@ public class ComboBox<T> extends AbstractSingleSelect<T>
private StyleGenerator<T> itemStyleGenerator = item -> null;
- private final SerializableBiPredicate<String, T> defaultFilterMethod = (
- text, item) -> getItemCaptionGenerator().apply(item)
- .toLowerCase(getLocale())
- .contains(text.toLowerCase(getLocale()));
-
/**
* Constructs an empty combo box without a caption. The content of the combo
* box can be set with {@link #setDataProvider(DataProvider)} or
@@ -216,10 +211,13 @@ public class ComboBox<T> extends AbstractSingleSelect<T>
@Override
public void setItems(Collection<T> items) {
- DataProvider<T, String> provider = DataProvider.create(items)
- .convertFilter(filterText -> item -> defaultFilterMethod
- .test(filterText, item));
- setDataProvider(provider);
+ // Cannot use the case insensitive contains shorthand from
+ // ListDataProvider since it wouldn't react to locale changes
+ CaptionFilter defaultCaptionFilter = (itemText, filterText) -> itemText
+ .toLowerCase(getLocale())
+ .contains(filterText.toLowerCase(getLocale()));
+
+ setItems(defaultCaptionFilter, items);
}
/**
@@ -236,9 +234,11 @@ public class ComboBox<T> extends AbstractSingleSelect<T>
* the data items to display
*/
public void setItems(CaptionFilter captionFilter, Collection<T> items) {
+ // Must do getItemCaptionGenerator() for each operation since it might
+ // not be the same as when this method was invoked
DataProvider<T, String> provider = DataProvider.create(items)
- .convertFilter(filterText -> item -> captionFilter.test(
- getItemCaptionGenerator().apply(item), filterText));
+ .filteringBy(item -> getItemCaptionGenerator().apply(item),
+ captionFilter);
setDataProvider(provider);
}
@@ -300,8 +300,8 @@ public class ComboBox<T> extends AbstractSingleSelect<T>
/**
* Returns true if the user can enter text into the field to either filter
- * the selections or enter a new value if new item handler is set
- * (see {@link #setNewItemHandler(NewItemHandler)}. If text input is disabled,
+ * the selections or enter a new value if new item handler is set (see
+ * {@link #setNewItemHandler(NewItemHandler)}. If text input is disabled,
* the comboBox will work in the same way as a {@link NativeSelect}
*
* @return true if text input is allowed
diff --git a/server/src/test/java/com/vaadin/data/provider/DataProviderTestBase.java b/server/src/test/java/com/vaadin/data/provider/DataProviderTestBase.java
index 9d1115d2af..79426f8c6c 100644
--- a/server/src/test/java/com/vaadin/data/provider/DataProviderTestBase.java
+++ b/server/src/test/java/com/vaadin/data/provider/DataProviderTestBase.java
@@ -190,4 +190,11 @@ public abstract class DataProviderTestBase<D extends DataProvider<StrBean, Seria
protected long sizeWithUnfilteredQuery() {
return dataProvider.fetch(new Query<>()).count();
}
+
+ protected static <F> void assertSizeWithFilter(int expectedSize,
+ DataProvider<?, F> dataProvider, F filterValue) {
+ Assert.assertEquals(expectedSize,
+ dataProvider.size(new Query<>(filterValue)));
+ }
+
}
diff --git a/server/src/test/java/com/vaadin/data/provider/ListDataProviderTest.java b/server/src/test/java/com/vaadin/data/provider/ListDataProviderTest.java
index e10c071c41..6f98100e19 100644
--- a/server/src/test/java/com/vaadin/data/provider/ListDataProviderTest.java
+++ b/server/src/test/java/com/vaadin/data/provider/ListDataProviderTest.java
@@ -2,6 +2,7 @@ package com.vaadin.data.provider;
import java.util.Comparator;
import java.util.List;
+import java.util.Locale;
import java.util.stream.Collectors;
import org.junit.Assert;
@@ -232,6 +233,50 @@ public class ListDataProviderTest
Assert.assertEquals(0, size);
}
+ @Test
+ public void filteringBy_itemPredicate() {
+ DataProvider<StrBean, String> filteringBy = dataProvider.filteringBy(
+ (item, filterValue) -> item.getValue().equals(filterValue));
+
+ assertSizeWithFilter(36, filteringBy, "Foo");
+ }
+
+ @Test
+ public void filteringBy_equals() {
+ DataProvider<StrBean, String> filteringBy = dataProvider
+ .filteringByEquals(StrBean::getValue);
+
+ assertSizeWithFilter(36, filteringBy, "Foo");
+ }
+
+ @Test
+ public void filteringBy_propertyValuePredicate() {
+ DataProvider<StrBean, Integer> filteringBy = dataProvider.filteringBy(
+ StrBean::getId,
+ (propertyValue, filterValue) -> propertyValue >= filterValue);
+
+ assertSizeWithFilter(90, filteringBy, 10);
+ }
+
+ @Test
+ public void filteringBy_caseInsensitiveSubstring() {
+ DataProvider<StrBean, String> filteringBy = dataProvider
+ .filteringBySubstring(StrBean::getValue, Locale.ENGLISH);
+
+ assertSizeWithFilter(36, filteringBy, "oo");
+ assertSizeWithFilter(36, filteringBy, "Oo");
+ }
+
+ @Test
+ public void filterBy_caseInsensitivePrefix() {
+ DataProvider<StrBean, String> filteringBy = dataProvider
+ .filteringByPrefix(StrBean::getValue, Locale.ENGLISH);
+
+ assertSizeWithFilter(36, filteringBy, "Fo");
+ assertSizeWithFilter(36, filteringBy, "fo");
+ assertSizeWithFilter(0, filteringBy, "oo");
+ }
+
@Override
protected void setSortOrder(List<SortOrder<String>> sortOrder,
Comparator<StrBean> comp) {