From 838c1d949b6a902fc1bdcabe107158d401d27425 Mon Sep 17 00:00:00 2001 From: Leif Åstrand <legioth@gmail.com> Date: Mon, 23 Jan 2017 14:59:11 +0200 Subject: Add shorthands to ComboBox for setting a ListDataProvider --- server/src/main/java/com/vaadin/ui/ComboBox.java | 84 ++++++++++++++++++++-- .../component/combobox/ComboBoxFilteringTest.java | 44 +++++++++--- 2 files changed, 115 insertions(+), 13 deletions(-) (limited to 'server') diff --git a/server/src/main/java/com/vaadin/ui/ComboBox.java b/server/src/main/java/com/vaadin/ui/ComboBox.java index 20cc14b2da..4cfcca813d 100644 --- a/server/src/main/java/com/vaadin/ui/ComboBox.java +++ b/server/src/main/java/com/vaadin/ui/ComboBox.java @@ -22,6 +22,7 @@ import java.util.HashMap; import java.util.Map; import java.util.Objects; import java.util.Set; +import java.util.stream.Stream; import org.jsoup.nodes.Element; @@ -30,6 +31,7 @@ import com.vaadin.data.HasValue; import com.vaadin.data.provider.DataCommunicator; import com.vaadin.data.provider.DataKeyMapper; import com.vaadin.data.provider.DataProvider; +import com.vaadin.data.provider.ListDataProvider; import com.vaadin.event.FieldEvents; import com.vaadin.event.FieldEvents.BlurEvent; import com.vaadin.event.FieldEvents.BlurListener; @@ -209,15 +211,64 @@ public class ComboBox<T> extends AbstractSingleSelect<T> }); } + /** + * {@inheritDoc} + * <p> + * Filtering will use a case insensitive match to show all items where the + * filter text is a substring of the caption displayed for that item. + */ @Override public void setItems(Collection<T> items) { + ListDataProvider<T> listDataProvider = DataProvider.create(items); + + setDataProvider(listDataProvider); + } + + /** + * {@inheritDoc} + * <p> + * Filtering will use a case insensitive match to show all items where the + * filter text is a substring of the caption displayed for that item. + */ + @Override + public void setItems(Stream<T> streamOfItems) { + // Overridden only to add clarification to javadocs + super.setItems(streamOfItems); + } + + /** + * {@inheritDoc} + * <p> + * Filtering will use a case insensitive match to show all items where the + * filter text is a substring of the caption displayed for that item. + */ + @Override + public void setItems(T... items) { + // Overridden only to add clarification to javadocs + super.setItems(items); + } + + /** + * Sets a list data provider as the data provider of this combo box. + * Filtering will use a case insensitive match to show all items where the + * filter text is a substring of the caption displayed for that item. + * <p> + * Note that this is a shorthand that calls + * {@link #setDataProvider(DataProvider)} with a wrapper of the provided + * list data provider. This means that {@link #getDataProvider()} will + * return the wrapper instead of the original list data provider. + * + * @param listDataProvider + * the list data provider to use, not <code>null</code> + */ + public void setDataProvider(ListDataProvider<T> listDataProvider) { // 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); + setDataProvider(defaultCaptionFilter, listDataProvider); } /** @@ -234,11 +285,36 @@ public class ComboBox<T> extends AbstractSingleSelect<T> * the data items to display */ public void setItems(CaptionFilter captionFilter, Collection<T> items) { + ListDataProvider<T> listDataProvider = DataProvider.create(items); + + setDataProvider(captionFilter, listDataProvider); + } + + /** + * Sets a list data provider with an item caption filter as the data + * provider of this combo box. The caption filter is used to compare the + * displayed caption of each item to the filter text entered by the user. + * <p> + * Note that this is a shorthand that calls + * {@link #setDataProvider(DataProvider)} with a wrapper of the provided + * list data provider. This means that {@link #getDataProvider()} will + * return the wrapper instead of the original list data provider. + * + * @param captionFilter + * filter to check if an item is shown when user typed some text + * into the ComboBox + * @param listDataProvider + * the list data provider to use, not <code>null</code> + */ + public void setDataProvider(CaptionFilter captionFilter, + ListDataProvider<T> listDataProvider) { + Objects.requireNonNull(listDataProvider, + "List data provider cannot be null"); + // 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) - .filteringBy(item -> getItemCaptionGenerator().apply(item), - captionFilter); + DataProvider<T, String> provider = listDataProvider.filteringBy( + item -> getItemCaptionGenerator().apply(item), captionFilter); setDataProvider(provider); } diff --git a/server/src/test/java/com/vaadin/tests/server/component/combobox/ComboBoxFilteringTest.java b/server/src/test/java/com/vaadin/tests/server/component/combobox/ComboBoxFilteringTest.java index b1f350f447..d4681151d5 100644 --- a/server/src/test/java/com/vaadin/tests/server/component/combobox/ComboBoxFilteringTest.java +++ b/server/src/test/java/com/vaadin/tests/server/component/combobox/ComboBoxFilteringTest.java @@ -27,6 +27,7 @@ import org.junit.Test; import com.vaadin.data.provider.DataProvider; import com.vaadin.data.provider.ListDataProvider; import com.vaadin.data.provider.Query; +import com.vaadin.tests.data.bean.Address; import com.vaadin.tests.data.bean.Person; import com.vaadin.tests.data.bean.Sex; import com.vaadin.ui.ComboBox; @@ -98,29 +99,54 @@ public class ComboBoxFilteringTest { public void setItems_array_customFiltering() { comboBox.setItemCaptionGenerator(Person::getFirstName); - // Result: typing "en" into the search field finds "Enrique Iglesias" - // and "Henry Dunant", but not "Erwin Engelbrecht" + // Result: typing "En" into the search field finds "Enrique Iglesias" + // but not "Henry Dunant" or "Erwin Engelbrecht" comboBox.setItems(String::startsWith, getPersonArray()); - checkFiltering("Er", "er", 3, 1); + checkFiltering("En", "en", 3, 1); } @Test public void setItems_collection_customFiltering() { comboBox.setItemCaptionGenerator(Person::getFirstName); + // Result: typing "En" into the search field finds "Enrique Iglesias" + // but not "Henry Dunant" or "Erwin Engelbrecht" + comboBox.setItems(String::startsWith, getPersonCollection()); + + checkFiltering("En", "en", 3, 1); + } + + @Test + public void setListDataProvider_defaultFiltering() { + comboBox.setItemCaptionGenerator(Person::getFirstName); + // Result: typing "en" into the search field finds "Enrique Iglesias" // and "Henry Dunant", but not "Erwin Engelbrecht" - comboBox.setItems(String::startsWith, getPersonCollection()); + comboBox.setDataProvider(DataProvider.create(getPersonCollection())); - checkFiltering("Er", "er", 3, 1); + checkFiltering("en", "ennen", 3, 2); + } + + @Test + public void setListDataProvider_customFiltering() { + comboBox.setItemCaptionGenerator(Person::getFirstName); + + // Result: typing "En" into the search field finds "Enrique Iglesias" + // but not "Henry Dunant" or "Erwin Engelbrecht" + comboBox.setDataProvider(String::startsWith, + DataProvider.create(getPersonCollection())); + + checkFiltering("En", "en", 3, 1); } public void invalid_dataProvider_compile_error() { - // uncommenting this causes a compile time error - ListDataProvider<Person> ldp = DataProvider.create(getPersonArray()); - // Compile error, because of invalid data provider filter type - // comboBox.setDataProvider(ldp); + DataProvider<Person, Address> dp = DataProvider.create(getPersonArray()) + .filteringByEquals(Person::getAddress); + + // uncommenting this causes a compile time error because of invalid data + // provider filter type + // comboBox.setDataProvider(dp); } @Test -- cgit v1.2.3