aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/src/com/vaadin/data/DataGenerator.java14
-rw-r--r--server/src/com/vaadin/data/RpcDataProviderExtension.java44
-rw-r--r--server/src/com/vaadin/ui/Grid.java10
-rw-r--r--uitest/src/com/vaadin/tests/components/grid/basicfeatures/GridBasicFeatures.java35
-rw-r--r--uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridDetailsServerTest.java39
5 files changed, 103 insertions, 39 deletions
diff --git a/server/src/com/vaadin/data/DataGenerator.java b/server/src/com/vaadin/data/DataGenerator.java
index a5333b8523..5e301d8151 100644
--- a/server/src/com/vaadin/data/DataGenerator.java
+++ b/server/src/com/vaadin/data/DataGenerator.java
@@ -18,7 +18,6 @@ package com.vaadin.data;
import java.io.Serializable;
import com.vaadin.ui.Grid.AbstractGridExtension;
-import com.vaadin.ui.Grid.AbstractRenderer;
import elemental.json.JsonObject;
@@ -26,8 +25,8 @@ import elemental.json.JsonObject;
* Interface for {@link AbstractGridExtension}s that allows adding data to row
* objects being sent to client by the {@link RpcDataProviderExtension}.
* <p>
- * {@link AbstractRenderer} implements this interface to provide encoded data to
- * client for {@link Renderer}s automatically.
+ * This class also provides a way to remove any unneeded data once the data
+ * object is no longer used on the client-side.
*
* @since 7.6
* @author Vaadin Ltd
@@ -46,4 +45,13 @@ public interface DataGenerator extends Serializable {
*/
public void generateData(Object itemId, Item item, JsonObject rowData);
+ /**
+ * Informs the DataGenerator that an item id has been dropped and is no
+ * longer needed.
+ *
+ * @param itemId
+ * removed item id
+ */
+ public void destroyData(Object itemId);
+
}
diff --git a/server/src/com/vaadin/data/RpcDataProviderExtension.java b/server/src/com/vaadin/data/RpcDataProviderExtension.java
index 78c87ab23d..45caf01587 100644
--- a/server/src/com/vaadin/data/RpcDataProviderExtension.java
+++ b/server/src/com/vaadin/data/RpcDataProviderExtension.java
@@ -98,7 +98,7 @@ public class RpcDataProviderExtension extends AbstractExtension {
// Remove still active rows that were "dropped"
droppedItems.removeAll(itemIds);
- internalDropActiveItems(droppedItems);
+ internalDropItems(droppedItems);
droppedItems.clear();
}
@@ -115,15 +115,6 @@ public class RpcDataProviderExtension extends AbstractExtension {
}
}
- private void internalDropActiveItems(Collection<Object> itemIds) {
- for (Object itemId : droppedItems) {
- assert activeItemMap.containsKey(itemId) : "Item ID should exist in the activeItemMap";
-
- activeItemMap.remove(itemId).removeListener();
- keyMapper.remove(itemId);
- }
- }
-
/**
* Gets a collection copy of currently active item ids.
*
@@ -147,6 +138,15 @@ public class RpcDataProviderExtension extends AbstractExtension {
rowData.put(GridState.JSONKEY_ROWKEY, keyMapper.key(itemId));
}
+ @Override
+ public void destroyData(Object itemId) {
+ keyMapper.remove(itemId);
+ GridValueChangeListener removed = activeItemMap.remove(itemId);
+
+ if (removed != null) {
+ removed.removeListener();
+ }
+ }
}
/**
@@ -372,6 +372,13 @@ public class RpcDataProviderExtension extends AbstractExtension {
.getConnectorId() : ""));
}
}
+
+ @Override
+ public void destroyData(Object itemId) {
+ if (visibleDetails.contains(itemId)) {
+ destroyDetails(itemId);
+ }
+ }
}
private final Indexed container;
@@ -734,8 +741,7 @@ public class RpcDataProviderExtension extends AbstractExtension {
public void setParent(ClientConnector parent) {
if (parent == null) {
// We're being detached, release various listeners
- activeItemHandler.internalDropActiveItems(activeItemHandler
- .getActiveItemIds());
+ internalDropItems(activeItemHandler.getActiveItemIds());
if (container instanceof ItemSetChangeNotifier) {
((ItemSetChangeNotifier) container)
@@ -750,6 +756,20 @@ public class RpcDataProviderExtension extends AbstractExtension {
}
/**
+ * Informs all DataGenerators than an item id has been dropped.
+ *
+ * @param droppedItemIds
+ * collection of dropped item ids
+ */
+ private void internalDropItems(Collection<Object> droppedItemIds) {
+ for (Object itemId : droppedItemIds) {
+ for (DataGenerator generator : dataGenerators) {
+ generator.destroyData(itemId);
+ }
+ }
+ }
+
+ /**
* Informs this data provider that given columns have been removed from
* grid.
*
diff --git a/server/src/com/vaadin/ui/Grid.java b/server/src/com/vaadin/ui/Grid.java
index f0e7b664e0..cee045128f 100644
--- a/server/src/com/vaadin/ui/Grid.java
+++ b/server/src/com/vaadin/ui/Grid.java
@@ -1131,6 +1131,11 @@ public class Grid extends AbstractFocusable implements SelectionNotifier,
}
@Override
+ public void destroyData(Object itemId) {
+ // NO-OP
+ }
+
+ @Override
protected Object getItemId(String rowKey) {
return rowKey != null ? super.getItemId(rowKey) : null;
}
@@ -1880,6 +1885,11 @@ public class Grid extends AbstractFocusable implements SelectionNotifier,
data.put(columnKeys.key(cell.getPropertyId()), AbstractRenderer
.encodeValue(modelValue, renderer, converter, getLocale()));
}
+
+ @Override
+ public void destroyData(Object itemId) {
+ // NO-OP
+ }
}
/**
diff --git a/uitest/src/com/vaadin/tests/components/grid/basicfeatures/GridBasicFeatures.java b/uitest/src/com/vaadin/tests/components/grid/basicfeatures/GridBasicFeatures.java
index 479ece71ca..598ac420fc 100644
--- a/uitest/src/com/vaadin/tests/components/grid/basicfeatures/GridBasicFeatures.java
+++ b/uitest/src/com/vaadin/tests/components/grid/basicfeatures/GridBasicFeatures.java
@@ -20,10 +20,12 @@ import java.text.DecimalFormatSymbols;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
+import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
+import java.util.Map;
import java.util.Random;
import com.vaadin.data.Container.Filter;
@@ -190,8 +192,6 @@ public class GridBasicFeatures extends AbstractComponentTest<Grid> {
}
};
- private Panel detailsPanel;
-
private final DetailsGenerator detailedDetailsGenerator = new DetailsGenerator() {
@Override
public Component getDetails(final RowReference rowReference) {
@@ -230,12 +230,18 @@ public class GridBasicFeatures extends AbstractComponentTest<Grid> {
}
};
- private final DetailsGenerator hierarchicalDetailsGenerator = new DetailsGenerator() {
+ private Map<Object, Panel> detailsMap = new HashMap<Object, Panel>();
+
+ private final DetailsGenerator persistingDetailsGenerator = new DetailsGenerator() {
@Override
public Component getDetails(RowReference rowReference) {
- detailsPanel = new Panel();
- detailsPanel.setContent(new Label("One"));
- return detailsPanel;
+ Object itemId = rowReference.getItemId();
+ if (!detailsMap.containsKey(itemId)) {
+ Panel panel = new Panel();
+ panel.setContent(new Label("One"));
+ detailsMap.put(itemId, panel);
+ }
+ return detailsMap.get(itemId);
}
};
@@ -1514,18 +1520,21 @@ public class GridBasicFeatures extends AbstractComponentTest<Grid> {
watchingDetailsGenerator);
createClickAction("Detailed", "Generators", swapDetailsGenerator,
detailedDetailsGenerator);
- createClickAction("Hierarchical", "Generators", swapDetailsGenerator,
- hierarchicalDetailsGenerator);
+ createClickAction("Persisting", "Generators", swapDetailsGenerator,
+ persistingDetailsGenerator);
createClickAction("- Change Component", "Generators",
new Command<Grid, Void>() {
@Override
public void execute(Grid c, Void value, Object data) {
- Label label = (Label) detailsPanel.getContent();
- if (label.getValue().equals("One")) {
- detailsPanel.setContent(new Label("Two"));
- } else {
- detailsPanel.setContent(new Label("One"));
+ for (Object id : detailsMap.keySet()) {
+ Panel panel = detailsMap.get(id);
+ Label label = (Label) panel.getContent();
+ if (label.getValue().equals("One")) {
+ panel.setContent(new Label("Two"));
+ } else {
+ panel.setContent(new Label("One"));
+ }
}
}
}, null);
diff --git a/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridDetailsServerTest.java b/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridDetailsServerTest.java
index a9ab7027db..f13ea9c073 100644
--- a/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridDetailsServerTest.java
+++ b/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridDetailsServerTest.java
@@ -23,9 +23,9 @@ import static org.junit.Assert.fail;
import org.junit.Before;
import org.junit.Test;
-import org.openqa.selenium.By;
import org.openqa.selenium.NoSuchElementException;
+import com.vaadin.testbench.By;
import com.vaadin.testbench.TestBenchElement;
import com.vaadin.testbench.elements.NotificationElement;
import com.vaadin.tests.components.grid.basicfeatures.GridBasicFeaturesTest;
@@ -48,8 +48,8 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
"Component", "Details", "Generators", "NULL" };
private static final String[] DETAILS_GENERATOR_WATCHING = new String[] {
"Component", "Details", "Generators", "\"Watching\"" };
- private static final String[] DETAILS_GENERATOR_HIERARCHICAL = new String[] {
- "Component", "Details", "Generators", "Hierarchical" };
+ private static final String[] DETAILS_GENERATOR_PERSISTING = new String[] {
+ "Component", "Details", "Generators", "Persisting" };
private static final String[] CHANGE_HIERARCHY = new String[] {
"Component", "Details", "Generators", "- Change Component" };
@@ -187,8 +187,8 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
}
@Test
- public void hierarchyChangesWorkInDetails() {
- selectMenuPath(DETAILS_GENERATOR_HIERARCHICAL);
+ public void persistingChangesWorkInDetails() {
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
assertEquals("One", getGridElement().getDetails(0).getText());
selectMenuPath(CHANGE_HIERARCHY);
@@ -196,8 +196,8 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
}
@Test
- public void hierarchyChangesWorkInDetailsWhileOutOfView() {
- selectMenuPath(DETAILS_GENERATOR_HIERARCHICAL);
+ public void persistingChangesWorkInDetailsWhileOutOfView() {
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
assertEquals("One", getGridElement().getDetails(0).getText());
scrollGridVerticallyTo(10000);
@@ -207,6 +207,23 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
}
@Test
+ public void persistingChangesWorkInDetailsWhenNotAttached() {
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
+ selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
+ assertEquals("One", getGridElement().getDetails(0).getText());
+
+ selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
+ assertFalse("Details should be detached", getGridElement()
+ .isElementPresent(By.vaadin("#details[0]")));
+
+ selectMenuPath(CHANGE_HIERARCHY);
+ selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
+
+ assertEquals("Two", getGridElement().getDetails(0).getText());
+
+ }
+
+ @Test
public void swappingDetailsGenerators_noDetailsShown() {
selectMenuPath(DETAILS_GENERATOR_WATCHING);
selectMenuPath(DETAILS_GENERATOR_NULL);
@@ -215,7 +232,7 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
@Test
public void swappingDetailsGenerators_shownDetails() {
- selectMenuPath(DETAILS_GENERATOR_HIERARCHICAL);
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
assertTrue("Details should contain 'One' at first", getGridElement()
.getDetails(0).getText().contains("One"));
@@ -237,7 +254,7 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
@Test
public void swappingDetailsGenerators_whileDetailsScrolledOut_showAfter() {
scrollGridVerticallyTo(1000);
- selectMenuPath(DETAILS_GENERATOR_HIERARCHICAL);
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
selectMenuPath(DETAILS_GENERATOR_WATCHING);
scrollGridVerticallyTo(0);
@@ -249,7 +266,7 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
@Test
public void swappingDetailsGenerators_whileDetailsScrolledOut_showBefore() {
- selectMenuPath(DETAILS_GENERATOR_HIERARCHICAL);
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
selectMenuPath(DETAILS_GENERATOR_WATCHING);
scrollGridVerticallyTo(1000);
@@ -261,7 +278,7 @@ public class GridDetailsServerTest extends GridBasicFeaturesTest {
@Test
public void swappingDetailsGenerators_whileDetailsScrolledOut_showBeforeAndAfter() {
- selectMenuPath(DETAILS_GENERATOR_HIERARCHICAL);
+ selectMenuPath(DETAILS_GENERATOR_PERSISTING);
selectMenuPath(OPEN_FIRST_ITEM_DETAILS);
selectMenuPath(DETAILS_GENERATOR_WATCHING);
scrollGridVerticallyTo(1000);