summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTeemu Suo-Anttila <teemusa@vaadin.com>2015-09-16 13:56:34 +0300
committerVaadin Code Review <review@vaadin.com>2015-09-16 12:43:45 +0000
commit89df22ae86f870b984838940d73fbab0790452ca (patch)
tree736718bda8bf6840cb27b198660e69a4780656fd
parent07a3c3c07ccff25c8afeb623b9fd3ffee6d02361 (diff)
downloadvaadin-framework-89df22ae86f870b984838940d73fbab0790452ca.tar.gz
vaadin-framework-89df22ae86f870b984838940d73fbab0790452ca.zip
Fix Grid validation with row change race condition (#18908)
Change-Id: I628f6b2921f800218a2e65d866b2332a9c574bda
-rw-r--r--client/src/com/vaadin/client/connectors/GridConnector.java22
-rw-r--r--client/src/com/vaadin/client/widget/grid/DefaultEditorEventHandler.java12
-rw-r--r--client/src/com/vaadin/client/widget/grid/EditorHandler.java7
-rw-r--r--client/src/com/vaadin/client/widgets/Grid.java139
-rw-r--r--uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridEditorUnbufferedTest.java16
5 files changed, 151 insertions, 45 deletions
diff --git a/client/src/com/vaadin/client/connectors/GridConnector.java b/client/src/com/vaadin/client/connectors/GridConnector.java
index a00e7f5a10..bffef05f89 100644
--- a/client/src/com/vaadin/client/connectors/GridConnector.java
+++ b/client/src/com/vaadin/client/connectors/GridConnector.java
@@ -236,13 +236,21 @@ public class GridConnector extends AbstractHasComponentsConnector implements
error);
}
- // Editor should not be touched while there's a
- // request pending.
- if (editorHandler.currentRequest == null) {
- getWidget().getEditor().setEditorError(
- getColumnErrors(),
- columnToErrorMessage.keySet());
- }
+ // Handle Editor RPC before updating error status
+ Scheduler.get().scheduleFinally(
+ new ScheduledCommand() {
+
+ @Override
+ public void execute() {
+ updateErrorColumns();
+ }
+ });
+ }
+
+ public void updateErrorColumns() {
+ getWidget().getEditor().setEditorError(
+ getColumnErrors(),
+ columnToErrorMessage.keySet());
}
});
}
diff --git a/client/src/com/vaadin/client/widget/grid/DefaultEditorEventHandler.java b/client/src/com/vaadin/client/widget/grid/DefaultEditorEventHandler.java
index f5ddbccaf4..da1380702c 100644
--- a/client/src/com/vaadin/client/widget/grid/DefaultEditorEventHandler.java
+++ b/client/src/com/vaadin/client/widget/grid/DefaultEditorEventHandler.java
@@ -17,6 +17,7 @@ package com.vaadin.client.widget.grid;
import com.google.gwt.core.client.Duration;
import com.google.gwt.dom.client.BrowserEvents;
+import com.google.gwt.dom.client.Element;
import com.google.gwt.event.dom.client.KeyCodes;
import com.google.gwt.user.client.Event;
import com.google.gwt.user.client.ui.Widget;
@@ -270,7 +271,9 @@ public class DefaultEditorEventHandler<T> implements Editor.EventHandler<T> {
// Limit colIndex between 0 and colCount - 1
colIndex = Math.max(0, Math.min(colCount - 1, colIndex));
- triggerValueChangeEvent(event);
+ if (rowIndex != event.getRowIndex()) {
+ triggerValueChangeEvent(event);
+ }
event.getEditor().editRow(rowIndex, colIndex);
}
@@ -287,9 +290,10 @@ public class DefaultEditorEventHandler<T> implements Editor.EventHandler<T> {
// Force a blur to cause a value change event
Widget editorWidget = event.getEditorWidget();
if (editorWidget != null) {
- if (editorWidget.getElement().isOrHasChild(
- WidgetUtil.getFocusedElement())) {
- WidgetUtil.getFocusedElement().blur();
+ Element focusedElement = WidgetUtil.getFocusedElement();
+ if (editorWidget.getElement().isOrHasChild(focusedElement)) {
+ focusedElement.blur();
+ focusedElement.focus();
}
}
}
diff --git a/client/src/com/vaadin/client/widget/grid/EditorHandler.java b/client/src/com/vaadin/client/widget/grid/EditorHandler.java
index 970130737c..91198700ca 100644
--- a/client/src/com/vaadin/client/widget/grid/EditorHandler.java
+++ b/client/src/com/vaadin/client/widget/grid/EditorHandler.java
@@ -54,6 +54,13 @@ public interface EditorHandler<T> {
public int getRowIndex();
/**
+ * Returns the index of the column being focused.
+ *
+ * @return the column index
+ */
+ public int getColumnIndex();
+
+ /**
* Returns the row data related to the row being requested.
*
* @return the row data
diff --git a/client/src/com/vaadin/client/widgets/Grid.java b/client/src/com/vaadin/client/widgets/Grid.java
index e7f41fa923..f22f3ac506 100644
--- a/client/src/com/vaadin/client/widgets/Grid.java
+++ b/client/src/com/vaadin/client/widgets/Grid.java
@@ -1051,14 +1051,16 @@ public class Grid<T> extends ResizeComposite implements
}
private Grid<T> grid;
- private int rowIndex;
+ private final int rowIndex;
+ private final int columnIndex;
private RequestCallback<T> callback;
private boolean completed = false;
- public EditorRequestImpl(Grid<T> grid, int rowIndex,
+ public EditorRequestImpl(Grid<T> grid, int rowIndex, int columnIndex,
RequestCallback<T> callback) {
this.grid = grid;
this.rowIndex = rowIndex;
+ this.columnIndex = columnIndex;
this.callback = callback;
}
@@ -1068,6 +1070,11 @@ public class Grid<T> extends ResizeComposite implements
}
@Override
+ public int getColumnIndex() {
+ return columnIndex;
+ }
+
+ @Override
public T getRow() {
return grid.getDataSource().getRow(rowIndex);
}
@@ -1360,6 +1367,7 @@ public class Grid<T> extends ResizeComposite implements
bindTimeout.cancel();
rowIndex = request.getRowIndex();
+ focusedColumnIndex = request.getColumnIndex();
showOverlay();
}
}
@@ -1371,6 +1379,7 @@ public class Grid<T> extends ResizeComposite implements
doCancel();
} else {
state = State.ACTIVE;
+ // TODO: Maybe restore focus?
}
bindTimeout.cancel();
@@ -1482,15 +1491,22 @@ public class Grid<T> extends ResizeComposite implements
*
* @since 7.5
*/
- public void editRow(final int rowIndex, int columnIndex) {
+ public void editRow(final int rowIndex, final int columnIndex) {
if (!enabled) {
throw new IllegalStateException(
"Cannot edit row: editor is not enabled");
}
- if (state != State.INACTIVE) {
- if (isBuffered() && this.rowIndex != rowIndex) {
+ if (state != State.INACTIVE && this.rowIndex != rowIndex) {
+ if (isBuffered()) {
throw new IllegalStateException(
"Cannot edit row: editor already in edit mode");
+ } else if (!columnErrors.isEmpty()) {
+ // Don't move row if errors are present
+
+ // FIXME: Should attempt bind if error field values have
+ // changed.
+
+ return;
}
}
if (columnIndex >= grid.getVisibleColumns().size()) {
@@ -1504,32 +1520,35 @@ public class Grid<T> extends ResizeComposite implements
return;
}
- if (focusedColumnIndex != columnIndex
- && columnIndex >= grid.getFrozenColumnCount()) {
- // Scroll to new focused column.
- grid.getEscalator().scrollToColumn(columnIndex,
- ScrollDestination.ANY, 0);
- }
- this.focusedColumnIndex = columnIndex;
-
if (this.rowIndex == rowIndex) {
+ if (focusedColumnIndex != columnIndex
+ && columnIndex >= grid.getFrozenColumnCount()) {
+ // Scroll to new focused column.
+ grid.getEscalator().scrollToColumn(columnIndex,
+ ScrollDestination.ANY, 0);
+ focusedColumnIndex = columnIndex;
+ }
+
updateHorizontalScrollPosition();
+
+ // Update Grid internal focus and focus widget if possible
+ grid.focusCell(rowIndex, focusedColumnIndex);
focusColumn(focusedColumnIndex);
+
// No need to request anything from the editor handler.
return;
}
- this.rowIndex = rowIndex;
state = State.ACTIVATING;
final Escalator escalator = grid.getEscalator();
if (escalator.getVisibleRowRange().contains(rowIndex)) {
- show(rowIndex);
+ show(rowIndex, columnIndex);
} else {
vScrollHandler = grid.addScrollHandler(new ScrollHandler() {
@Override
public void onScroll(ScrollEvent event) {
if (escalator.getVisibleRowRange().contains(rowIndex)) {
- show(rowIndex);
+ show(rowIndex, columnIndex);
vScrollHandler.removeHandler();
}
}
@@ -1558,7 +1577,8 @@ public class Grid<T> extends ResizeComposite implements
throw new IllegalStateException(
"Cannot cancel edit: editor is not in edit mode");
}
- handler.cancel(new EditorRequestImpl<T>(grid, rowIndex, null));
+ handler.cancel(new EditorRequestImpl<T>(grid, rowIndex,
+ focusedColumnIndex, null));
doCancel();
}
@@ -1605,7 +1625,7 @@ public class Grid<T> extends ResizeComposite implements
setButtonsEnabled(false);
saveTimeout.schedule(SAVE_TIMEOUT_MS);
EditorRequest<T> request = new EditorRequestImpl<T>(grid, rowIndex,
- saveRequestCallback);
+ focusedColumnIndex, saveRequestCallback);
handler.save(request);
updateSelectionCheckboxesAsNeeded(true);
}
@@ -1664,12 +1684,12 @@ public class Grid<T> extends ResizeComposite implements
this.enabled = enabled;
}
- protected void show(int rowIndex) {
+ protected void show(int rowIndex, int columnIndex) {
if (state == State.ACTIVATING) {
state = State.BINDING;
bindTimeout.schedule(BIND_TIMEOUT_MS);
EditorRequest<T> request = new EditorRequestImpl<T>(grid,
- rowIndex, bindRequestCallback);
+ rowIndex, columnIndex, bindRequestCallback);
handler.bind(request);
grid.getEscalator().setScrollLocked(Direction.VERTICAL,
isBuffered());
@@ -1798,7 +1818,9 @@ public class Grid<T> extends ResizeComposite implements
}, FocusEvent.getType()));
}
- focusColumn(focusedColumnIndex);
+ if (i == focusedColumnIndex) {
+ focusColumn(focusedColumnIndex);
+ }
} else {
cell.addClassName(NOT_EDITABLE_CLASS_NAME);
cell.addClassName(tr.getCells().getItem(i).getClassName());
@@ -1907,13 +1929,13 @@ public class Grid<T> extends ResizeComposite implements
}
Widget editor = getWidget(grid.getVisibleColumn(colIndex));
- if (colIndex == focusedColumnIndex) {
- if (editor instanceof Focusable) {
- ((Focusable) editor).focus();
- } else if (editor instanceof com.google.gwt.user.client.ui.Focusable) {
- ((com.google.gwt.user.client.ui.Focusable) editor)
- .setFocus(true);
- }
+ if (editor instanceof Focusable) {
+ ((Focusable) editor).focus();
+ } else if (editor instanceof com.google.gwt.user.client.ui.Focusable) {
+ ((com.google.gwt.user.client.ui.Focusable) editor)
+ .setFocus(true);
+ } else {
+ grid.focus();
}
}
@@ -2187,6 +2209,27 @@ public class Grid<T> extends ResizeComposite implements
public boolean isWorkPending() {
return saveTimeout.isRunning() || bindTimeout.isRunning();
}
+
+ protected int getElementColumn(Element e) {
+ int frozenCells = frozenCellWrapper.getChildCount();
+ if (frozenCellWrapper.isOrHasChild(e)) {
+ for (int i = 0; i < frozenCells; ++i) {
+ if (frozenCellWrapper.getChild(i).isOrHasChild(e)) {
+ return i;
+ }
+ }
+ }
+
+ if (cellWrapper.isOrHasChild(e)) {
+ for (int i = 0; i < cellWrapper.getChildCount(); ++i) {
+ if (cellWrapper.getChild(i).isOrHasChild(e)) {
+ return i + frozenCells;
+ }
+ }
+ }
+
+ return -1;
+ }
}
public static abstract class AbstractGridKeyEvent<HANDLER extends AbstractGridKeyEventHandler>
@@ -6834,8 +6877,30 @@ public class Grid<T> extends ResizeComposite implements
cell = cellFocusHandler.getFocusedCell();
container = cellFocusHandler.containerWithFocus;
} else {
- // Click in a location that does not contain cells.
- return;
+ // Click might be in an editor cell, should still map.
+ if (editor.editorOverlay != null
+ && editor.editorOverlay.isOrHasChild(e)) {
+ container = escalator.getBody();
+ int rowIndex = editor.getRow();
+ int colIndex = editor.getElementColumn(e);
+
+ if (colIndex < 0) {
+ getLogger().warning(
+ "Did not find a suitable column index.");
+ // Click in editor, but not for any column.
+ return;
+ }
+
+ TableCellElement cellElement = container
+ .getRowElement(rowIndex).getCells()
+ .getItem(colIndex);
+
+ cell = new Cell(rowIndex, colIndex, cellElement);
+ } else {
+ getLogger().warning("Click is lost.");
+ // Click in a place that does not have a column.
+ return;
+ }
}
} else {
cell = container.getCell(e);
@@ -6934,9 +6999,17 @@ public class Grid<T> extends ResizeComposite implements
}
private boolean handleEditorEvent(Event event, RowContainer container) {
- return getEditor().getEventHandler().handleEvent(
- new EditorDomEvent<T>(event, getEventCell(), editor
- .getWidget(eventCell.getColumn())));
+ Widget w;
+ if (editor.focusedColumnIndex < 0) {
+ w = null;
+ } else {
+ w = editor.getWidget(getColumn(editor.focusedColumnIndex));
+ }
+
+ EditorDomEvent<T> editorEvent = new EditorDomEvent<T>(event,
+ getEventCell(), w);
+
+ return getEditor().getEventHandler().handleEvent(editorEvent);
}
private boolean handleRendererEvent(Event event, RowContainer container) {
diff --git a/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridEditorUnbufferedTest.java b/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridEditorUnbufferedTest.java
index 4fe88c6eac..984b03ac5b 100644
--- a/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridEditorUnbufferedTest.java
+++ b/uitest/src/com/vaadin/tests/components/grid/basicfeatures/server/GridEditorUnbufferedTest.java
@@ -114,12 +114,26 @@ public class GridEditorUnbufferedTest extends GridEditorTest {
selectMenuPath(EDIT_ITEM_5);
getEditorWidgets().get(1).click();
- getEditorWidgets().get(1).sendKeys("not a number");
+ String faultyInt = "not a number";
+ getEditorWidgets().get(1).sendKeys(faultyInt);
getGridElement().getCell(10, 0).click();
assertEquals("Editor should not move from row 5", "(5, 0)",
getEditorWidgets().get(0).getAttribute("value"));
+
+ for (int i = 0; i < faultyInt.length(); ++i) {
+ getEditorWidgets().get(1).sendKeys(Keys.BACK_SPACE);
+ }
+
+ // FIXME: Needs to trigger one extra validation round-trip for now
+ getGridElement().sendKeys(Keys.ENTER);
+
+ getGridElement().getCell(10, 0).click();
+
+ assertEquals("Editor should not to row 10", "(10, 0)",
+ getEditorWidgets().get(0).getAttribute("value"));
+
}
@Test