diff options
author | Leif Åstrand <legioth@gmail.com> | 2017-01-22 21:50:46 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2017-01-22 21:50:46 +0200 |
commit | 5447e7e05392483455f7df4bf2af5d5d74e5a2ae (patch) | |
tree | 5028592fea2d9f1dda0c23d1448b92b8a35dc6fd /server/src | |
parent | 19427ba7c7a51d3af61c170a7ba54137eb629ac8 (diff) | |
download | vaadin-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')
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) { |