diff options
-rw-r--r-- | server/src/main/java/com/vaadin/ui/components/grid/MultiSelectionModelImpl.java | 10 | ||||
-rw-r--r-- | server/src/test/java/com/vaadin/tests/components/grid/GridMultiSelectionModelTest.java | 31 |
2 files changed, 39 insertions, 2 deletions
diff --git a/server/src/main/java/com/vaadin/ui/components/grid/MultiSelectionModelImpl.java b/server/src/main/java/com/vaadin/ui/components/grid/MultiSelectionModelImpl.java index 3979a25a1a..408b133426 100644 --- a/server/src/main/java/com/vaadin/ui/components/grid/MultiSelectionModelImpl.java +++ b/server/src/main/java/com/vaadin/ui/components/grid/MultiSelectionModelImpl.java @@ -445,9 +445,15 @@ public class MultiSelectionModelImpl<T> extends AbstractSelectionModel<T> } doUpdateSelection(set -> { + DataProvider<T, ?> dataProvider = getGrid().getDataProvider(); // order of add / remove does not matter since no duplicates - set.removeAll(removedItems); - set.addAll(addedItems); + Set<Object> removedItemIds = removedItems.stream() + .map(dataProvider::getId).collect(Collectors.toSet()); + set.removeIf( + item -> removedItemIds.contains(dataProvider.getId(item))); + addedItems.stream().filter( + item -> !selectionContainsId(dataProvider.getId(item))) + .forEach(set::add); // refresh method is NOOP for items that are not present client side DataCommunicator<T> dataCommunicator = getGrid() diff --git a/server/src/test/java/com/vaadin/tests/components/grid/GridMultiSelectionModelTest.java b/server/src/test/java/com/vaadin/tests/components/grid/GridMultiSelectionModelTest.java index 131dc2025b..7250f02dd3 100644 --- a/server/src/test/java/com/vaadin/tests/components/grid/GridMultiSelectionModelTest.java +++ b/server/src/test/java/com/vaadin/tests/components/grid/GridMultiSelectionModelTest.java @@ -17,8 +17,11 @@ import java.util.Optional; import java.util.Set; import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicReference; +import java.util.stream.Collectors; import java.util.stream.IntStream; +import java.util.stream.Stream; +import com.vaadin.data.provider.CallbackDataProvider; import org.easymock.Capture; import org.junit.Before; import org.junit.Test; @@ -703,6 +706,34 @@ public class GridMultiSelectionModelTest { assertFalse(model.isSelectAllCheckBoxVisible()); assertEquals(SelectAllCheckBoxVisibility.DEFAULT, model.getSelectAllCheckBoxVisibility()); + } + + @Test + public void testSelectAllWithProxyDataProvider() { + List<String> data = IntStream.range(0, 100).boxed() + .map(i -> "String " + i).collect(Collectors.toList()); + Grid<AtomicReference<String>> proxyGrid = new Grid<>(); + proxyGrid.setDataProvider(new CallbackDataProvider<>( + q -> data.stream().map(AtomicReference::new).skip(q.getOffset()) + .limit(q.getLimit()), + q -> data.size(), AtomicReference::get)); + MultiSelectionModel<AtomicReference<String>> model = (MultiSelectionModel<AtomicReference<String>>) proxyGrid + .setSelectionMode(SelectionMode.MULTI); + List<Set<AtomicReference<String>>> selections = new ArrayList<>(); + model.addSelectionListener(e -> { + selections.add(e.getAllSelectedItems()); + }); + model.selectAll(); + model.deselect(model.getFirstSelectedItem().orElseThrow( + () -> new IllegalStateException("Items should be selected"))); + model.selectAll(); + + assertEquals("Item count mismatch on first select all", 100, + selections.get(0).size()); + assertEquals("Item count mismatch on deselect", 99, + selections.get(1).size()); + assertEquals("Item count mismatch on second select all", 100, + selections.get(2).size()); } } |