diff options
author | Teemu Suo-Anttila <teemusa@vaadin.com> | 2015-09-16 13:56:34 +0300 |
---|---|---|
committer | Vaadin Code Review <review@vaadin.com> | 2015-09-16 12:43:45 +0000 |
commit | 89df22ae86f870b984838940d73fbab0790452ca (patch) | |
tree | 736718bda8bf6840cb27b198660e69a4780656fd | |
parent | 07a3c3c07ccff25c8afeb623b9fd3ffee6d02361 (diff) | |
download | vaadin-framework-89df22ae86f870b984838940d73fbab0790452ca.tar.gz vaadin-framework-89df22ae86f870b984838940d73fbab0790452ca.zip |
Fix Grid validation with row change race condition (#18908)
Change-Id: I628f6b2921f800218a2e65d866b2332a9c574bda
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 |