From eed3293c5a2f4a5b99d118ccdc5b2861250315b1 Mon Sep 17 00:00:00 2001 From: Teemu Suo-Anttila Date: Thu, 26 Jul 2018 10:33:32 +0300 Subject: Fix Grid not using getId on select all (cherry picked from commit 69c235f) --- .../components/grid/MultiSelectionModelImpl.java | 10 +++++-- .../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 extends AbstractSelectionModel } doUpdateSelection(set -> { + DataProvider dataProvider = getGrid().getDataProvider(); // order of add / remove does not matter since no duplicates - set.removeAll(removedItems); - set.addAll(addedItems); + Set 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 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 data = IntStream.range(0, 100).boxed() + .map(i -> "String " + i).collect(Collectors.toList()); + Grid> 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> model = (MultiSelectionModel>) proxyGrid + .setSelectionMode(SelectionMode.MULTI); + List>> 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()); } } -- cgit v1.2.3