summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--client/src/main/java/com/vaadin/client/WidgetUtil.java51
-rw-r--r--client/src/main/java/com/vaadin/client/connectors/grid/EditorConnector.java14
-rwxr-xr-xclient/src/main/java/com/vaadin/client/widgets/Grid.java96
-rw-r--r--uitest/src/main/java/com/vaadin/tests/components/grid/GridEditRow.java97
-rw-r--r--uitest/src/test/java/com/vaadin/tests/components/grid/GridEditRowTest.java116
5 files changed, 336 insertions, 38 deletions
diff --git a/client/src/main/java/com/vaadin/client/WidgetUtil.java b/client/src/main/java/com/vaadin/client/WidgetUtil.java
index 11a5ae363c..b995ef15c0 100644
--- a/client/src/main/java/com/vaadin/client/WidgetUtil.java
+++ b/client/src/main/java/com/vaadin/client/WidgetUtil.java
@@ -52,6 +52,37 @@ import com.vaadin.shared.util.SharedUtil;
public class WidgetUtil {
/**
+ * Simple object to store another object.
+ *
+ * @param <T>
+ * the object type to store
+ * @since
+ */
+ public static class Reference<T> {
+
+ T reference = null;
+
+ /**
+ * Gets the current object.
+ *
+ * @return the stored object
+ */
+ public T get() {
+ return reference;
+ }
+
+ /**
+ * Sets the current object.
+ *
+ * @param reference
+ * the object to store
+ */
+ public void set(T reference) {
+ this.reference = reference;
+ }
+ }
+
+ /**
* Helper method for debugging purposes.
*
* Stops execution on firefox browsers on a breakpoint.
@@ -778,7 +809,7 @@ public class WidgetUtil {
com.google.gwt.dom.client.Element el, String p)
/*-{
try {
-
+
if (el.currentStyle) {
// IE
return el.currentStyle[p];
@@ -793,7 +824,7 @@ public class WidgetUtil {
} catch (e) {
return "";
}
-
+
}-*/;
/**
@@ -807,7 +838,7 @@ public class WidgetUtil {
try {
el.focus();
} catch (e) {
-
+
}
}-*/;
@@ -1158,7 +1189,7 @@ public class WidgetUtil {
if ($wnd.document.activeElement) {
return $wnd.document.activeElement;
}
-
+
return null;
}-*/;
@@ -1229,11 +1260,11 @@ public class WidgetUtil {
/*-{
var top = elem.offsetTop;
var height = elem.offsetHeight;
-
+
if (elem.parentNode != elem.offsetParent) {
top -= elem.parentNode.offsetTop;
}
-
+
var cur = elem.parentNode;
while (cur && (cur.nodeType == 1)) {
if (top < cur.scrollTop) {
@@ -1242,12 +1273,12 @@ public class WidgetUtil {
if (top + height > cur.scrollTop + cur.clientHeight) {
cur.scrollTop = (top + height) - cur.clientHeight;
}
-
+
var offsetTop = cur.offsetTop;
if (cur.parentNode != cur.offsetParent) {
offsetTop -= cur.parentNode.offsetTop;
}
-
+
top += offsetTop - cur.scrollTop;
cur = cur.parentNode;
}
@@ -1696,7 +1727,7 @@ public class WidgetUtil {
}
var heightWithoutBorder = cloneElement.offsetHeight;
parentElement.removeChild(cloneElement);
-
+
return heightWithBorder - heightWithoutBorder;
}
}-*/;
@@ -1873,7 +1904,7 @@ public class WidgetUtil {
* ancestors has the style {@code display: none} applied.
*
* @param element
- * the element to test for visibility
+ * the element to test for visibility
* @return {@code true} if the element is displayed, {@code false} otherwise
* @since 8.3.2
*/
diff --git a/client/src/main/java/com/vaadin/client/connectors/grid/EditorConnector.java b/client/src/main/java/com/vaadin/client/connectors/grid/EditorConnector.java
index c8c60c9620..8e2d050d55 100644
--- a/client/src/main/java/com/vaadin/client/connectors/grid/EditorConnector.java
+++ b/client/src/main/java/com/vaadin/client/connectors/grid/EditorConnector.java
@@ -66,12 +66,7 @@ public class EditorConnector extends AbstractExtensionConnector {
public void bind(final int rowIndex) {
// call this deferred to avoid issues with editing on init
Scheduler.get().scheduleDeferred(() -> {
- currentEditedRow = rowIndex;
- // might need to wait for available data,
- // if data is available, ensureAvailability will immediately trigger the handler anyway,
- // so no need for alternative "immediately available" logic
- waitingForAvailableData = true;
- getParent().getDataSource().ensureAvailability(rowIndex, 1);
+ getParent().getWidget().editRow(rowIndex);
});
}
@@ -218,13 +213,6 @@ public class EditorConnector extends AbstractExtensionConnector {
protected void extend(ServerConnector target) {
Grid<JsonObject> grid = getParent().getWidget();
grid.getEditor().setHandler(new CustomEditorHandler());
- grid.addDataAvailableHandler((event) -> {
- Range range = event.getAvailableRows();
- if (waitingForAvailableData && currentEditedRow != null && range.contains(currentEditedRow)) {
- getParent().getWidget().editRow(currentEditedRow);
- waitingForAvailableData = false;
- }
- });
}
@Override
diff --git a/client/src/main/java/com/vaadin/client/widgets/Grid.java b/client/src/main/java/com/vaadin/client/widgets/Grid.java
index 1d24086b3b..5a31ec1f41 100755
--- a/client/src/main/java/com/vaadin/client/widgets/Grid.java
+++ b/client/src/main/java/com/vaadin/client/widgets/Grid.java
@@ -80,6 +80,7 @@ import com.vaadin.client.BrowserInfo;
import com.vaadin.client.DeferredWorker;
import com.vaadin.client.Focusable;
import com.vaadin.client.WidgetUtil;
+import com.vaadin.client.WidgetUtil.Reference;
import com.vaadin.client.data.DataChangeHandler;
import com.vaadin.client.data.DataSource;
import com.vaadin.client.data.DataSource.RowHandle;
@@ -1437,7 +1438,6 @@ public class Grid<T> extends ResizeComposite implements HasSelectionHandlers<T>,
private String styleName = null;
private HandlerRegistration hScrollHandler;
- private HandlerRegistration vScrollHandler;
private final Button saveButton;
private final Button cancelButton;
@@ -1680,20 +1680,10 @@ public class Grid<T> extends ResizeComposite implements HasSelectionHandlers<T>,
}
state = State.ACTIVATING;
- final Escalator escalator = grid.getEscalator();
- if (escalator.getVisibleRowRange().contains(rowIndex)) {
- show(rowIndex, columnIndexDOM);
- } else {
- vScrollHandler = grid.addScrollHandler(event -> {
- if (escalator.getVisibleRowRange().contains(rowIndex)) {
- show(rowIndex, columnIndexDOM);
- vScrollHandler.removeHandler();
- }
- });
- grid.scrollToRow(rowIndex,
- isBuffered() ? ScrollDestination.MIDDLE
- : ScrollDestination.ANY);
- }
+ grid.scrollToRow(rowIndex,
+ isBuffered() ? ScrollDestination.MIDDLE
+ : ScrollDestination.ANY,
+ () -> show(rowIndex, columnIndexDOM));
}
/**
@@ -7384,6 +7374,45 @@ public class Grid<T> extends ResizeComposite implements HasSelectionHandlers<T>,
}
/**
+ * Helper method for making sure desired row is visible and it is properly
+ * rendered.
+ *
+ * @param rowIndex
+ * the row to look for
+ * @param destination
+ * the desired scroll destination
+ * @param callback
+ * the callback command to execute when row is available
+ * @since
+ */
+ public void scrollToRow(int rowIndex, ScrollDestination destination,
+ Runnable callback) {
+ waitUntilVisible(rowIndex, destination, () -> {
+ Reference<HandlerRegistration> registration = new Reference<>();
+ registration.set(addDataAvailableHandler(event -> {
+ if (event.getAvailableRows().contains(rowIndex)) {
+ registration.get().removeHandler();
+ callback.run();
+ }
+ }));
+ });
+ }
+
+ /**
+ * Helper method for making sure desired row is visible and it is properly
+ * rendered.
+ *
+ * @param rowIndex
+ * the row to look for
+ * @param whenRendered
+ * the callback command to execute when row is available
+ * @since
+ */
+ public void scrollToRow(int rowIndex, Runnable whenRendered) {
+ scrollToRow(rowIndex, ScrollDestination.ANY, whenRendered);
+ }
+
+ /**
* Scrolls to a certain row using only user-specified parameters.
* <p>
* If the details for that row are visible, those will be taken into account
@@ -7421,6 +7450,43 @@ public class Grid<T> extends ResizeComposite implements HasSelectionHandlers<T>,
}
/**
+ * Helper method for scrolling and making sure row is visible.
+ *
+ * @param rowIndex
+ * the row index to make visible
+ * @param destination
+ * the desired scroll destination
+ * @param whenVisible
+ * the callback method to call when row is visible
+ */
+ private void waitUntilVisible(int rowIndex, ScrollDestination destination,
+ Runnable whenVisible) {
+ final Escalator escalator = getEscalator();
+ if (escalator.getVisibleRowRange().contains(rowIndex)) {
+ TableRowElement rowElement = escalator.getBody()
+ .getRowElement(rowIndex);
+ long bottomBorder = Math.round(WidgetUtil.getBorderBottomThickness(
+ rowElement.getFirstChildElement()) + 0.5d);
+ if (rowElement.getAbsoluteTop() >= escalator.getHeader()
+ .getElement().getAbsoluteBottom()
+ && rowElement.getAbsoluteBottom() <= escalator.getFooter()
+ .getElement().getAbsoluteTop() + bottomBorder) {
+ whenVisible.run();
+ return;
+ }
+ }
+
+ Reference<HandlerRegistration> registration = new Reference<>();
+ registration.set(addScrollHandler(event -> {
+ if (escalator.getVisibleRowRange().contains(rowIndex)) {
+ registration.get().removeHandler();
+ whenVisible.run();
+ }
+ }));
+ scrollToRow(rowIndex, destination);
+ }
+
+ /**
* Scrolls to the beginning of the very first row.
*/
public void scrollToStart() {
diff --git a/uitest/src/main/java/com/vaadin/tests/components/grid/GridEditRow.java b/uitest/src/main/java/com/vaadin/tests/components/grid/GridEditRow.java
new file mode 100644
index 0000000000..bd24f53380
--- /dev/null
+++ b/uitest/src/main/java/com/vaadin/tests/components/grid/GridEditRow.java
@@ -0,0 +1,97 @@
+package com.vaadin.tests.components.grid;
+
+import java.util.ArrayList;
+
+import com.vaadin.annotations.Widgetset;
+import com.vaadin.data.provider.ListDataProvider;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.tests.components.AbstractTestUI;
+import com.vaadin.ui.Button;
+import com.vaadin.ui.Grid;
+import com.vaadin.ui.TextField;
+
+@Widgetset("com.vaadin.DefaultWidgetSet")
+public class GridEditRow extends AbstractTestUI {
+
+ private ArrayList<TestBean> items = new ArrayList<>();
+ private TestBean bean = null;
+ private int counter = 0;
+
+ @SuppressWarnings("unchecked")
+ @Override
+ protected void setup(VaadinRequest request) {
+
+ Grid<TestBean> grid = new Grid<TestBean>(TestBean.class);
+ grid.setDataProvider(new ListDataProvider<>(items));
+ grid.setWidth("100%");
+ grid.setHeight("400px");
+ grid.getEditor().setEnabled(true);
+ grid.getColumn("name").setEditorComponent(new TextField());
+ grid.getColumn("value").setEditorComponent(new TextField());
+
+ getLayout().addComponent(new Button("Add", event -> {
+ bean = new TestBean();
+ items.add(bean);
+ grid.getDataProvider().refreshAll();
+ }));
+ getLayout().addComponent(new Button("Add, Select & Edit", event -> {
+ bean = new TestBean();
+ items.add(bean);
+ grid.getDataProvider().refreshAll();
+ grid.select(bean);
+ int row = ((ArrayList<TestBean>) ((ListDataProvider<TestBean>) grid
+ .getDataProvider()).getItems()).indexOf(bean);
+ grid.getEditor().editRow(row);
+ }));
+ getLayout().addComponent(new Button("Edit", event -> {
+ int row = ((ArrayList<TestBean>) ((ListDataProvider<TestBean>) grid
+ .getDataProvider()).getItems()).indexOf(bean);
+ grid.getEditor().editRow(row);
+ }));
+ getLayout().addComponent(new Button("Select & Edit", event -> {
+ grid.select(bean);
+ int row = ((ArrayList<TestBean>) ((ListDataProvider<TestBean>) grid
+ .getDataProvider()).getItems()).indexOf(bean);
+ grid.getEditor().editRow(row);
+ }));
+
+ getLayout().addComponent(grid);
+ }
+
+ @Override
+ protected Integer getTicketNumber() {
+ return 10558;
+ }
+
+ @Override
+ protected String getTestDescription() {
+ return "Calling editRow shouldn't cause any other rows to be emptied.";
+ }
+
+ public class TestBean {
+ private String name = "name" + counter;
+ private String value = "value" + counter;
+
+ public TestBean() {
+ ++counter;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public void setName(String inName) {
+ name = inName;
+ }
+
+ public String getValue() {
+ return value;
+ }
+
+ public void setValue(String inValue) {
+ value = inValue;
+ }
+
+ }
+
+}
diff --git a/uitest/src/test/java/com/vaadin/tests/components/grid/GridEditRowTest.java b/uitest/src/test/java/com/vaadin/tests/components/grid/GridEditRowTest.java
new file mode 100644
index 0000000000..d84a09782b
--- /dev/null
+++ b/uitest/src/test/java/com/vaadin/tests/components/grid/GridEditRowTest.java
@@ -0,0 +1,116 @@
+/*
+ * Copyright 2000-2016 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.tests.components.grid;
+
+import static org.hamcrest.CoreMatchers.is;
+import static org.hamcrest.CoreMatchers.not;
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.hamcrest.Matchers.greaterThan;
+
+import org.junit.Test;
+
+import com.vaadin.testbench.By;
+import com.vaadin.testbench.elements.ButtonElement;
+import com.vaadin.testbench.elements.GridElement;
+import com.vaadin.testbench.elements.GridElement.GridEditorElement;
+import com.vaadin.testbench.parallel.TestCategory;
+import com.vaadin.tests.tb3.MultiBrowserTest;
+
+/**
+ * Tests for ensuring that the furthest away visible rows don't get emptied when
+ * editRow is called, and that the editor doesn't open beyond the lower border
+ * of the Grid.
+ *
+ */
+@TestCategory("grid")
+public class GridEditRowTest extends MultiBrowserTest {
+
+ private GridElement grid;
+ private ButtonElement addButton;
+ private ButtonElement editButton;
+
+ @Override
+ public void setup() throws Exception {
+ super.setup();
+ openTestURL();
+ grid = $(GridElement.class).first();
+ addButton = $(ButtonElement.class).caption("Add").first();
+ editButton = $(ButtonElement.class).caption("Edit").first();
+ }
+
+ public void addRows(int count) {
+ for (int i = 0; i < count; ++i) {
+ addButton.click();
+ }
+ }
+
+ public void editLastRow() {
+ editButton.click();
+ }
+
+ private void assertRowContents(int rowIndex) {
+ assertThat(grid.getCell(rowIndex, 0).getText(), is("name" + rowIndex));
+ }
+
+ private void assertEditorWithinGrid() {
+ GridEditorElement editor = grid.getEditor();
+ assertThat(editor.getLocation().y + editor.getSize().height,
+ not(greaterThan(grid.getLocation().y + grid.getSize().height)));
+ }
+
+ @Test
+ public void testEditWhenAllRowsVisible() {
+ addRows(7);
+
+ assertRowContents(0);
+
+ editLastRow();
+
+ assertRowContents(0);
+
+ waitForElementVisible(By.className("v-grid-editor"));
+
+ assertEditorWithinGrid();
+ }
+
+ @Test
+ public void testEditWhenSomeRowsNotVisible() {
+ addRows(11);
+
+ assertRowContents(3);
+
+ editLastRow();
+
+ waitForElementVisible(By.className("v-grid-editor"));
+
+ assertRowContents(3);
+ assertEditorWithinGrid();
+ }
+
+ @Test
+ public void testEditWhenSomeRowsOutsideOfCache() {
+ addRows(100);
+
+ assertRowContents(91);
+
+ editLastRow();
+
+ waitForElementVisible(By.className("v-grid-editor"));
+
+ assertRowContents(91);
+ assertEditorWithinGrid();
+ }
+}