From 89df22ae86f870b984838940d73fbab0790452ca Mon Sep 17 00:00:00 2001 From: Teemu Suo-Anttila Date: Wed, 16 Sep 2015 13:56:34 +0300 Subject: Fix Grid validation with row change race condition (#18908) Change-Id: I628f6b2921f800218a2e65d866b2332a9c574bda --- .../vaadin/client/connectors/GridConnector.java | 22 ++-- .../widget/grid/DefaultEditorEventHandler.java | 12 +- .../vaadin/client/widget/grid/EditorHandler.java | 7 ++ client/src/com/vaadin/client/widgets/Grid.java | 139 ++++++++++++++++----- 4 files changed, 136 insertions(+), 44 deletions(-) (limited to 'client') 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 implements Editor.EventHandler { // 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 implements Editor.EventHandler { // 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 @@ -53,6 +53,13 @@ public interface EditorHandler { */ 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. * 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 extends ResizeComposite implements } private Grid grid; - private int rowIndex; + private final int rowIndex; + private final int columnIndex; private RequestCallback callback; private boolean completed = false; - public EditorRequestImpl(Grid grid, int rowIndex, + public EditorRequestImpl(Grid grid, int rowIndex, int columnIndex, RequestCallback callback) { this.grid = grid; this.rowIndex = rowIndex; + this.columnIndex = columnIndex; this.callback = callback; } @@ -1067,6 +1069,11 @@ public class Grid extends ResizeComposite implements return rowIndex; } + @Override + public int getColumnIndex() { + return columnIndex; + } + @Override public T getRow() { return grid.getDataSource().getRow(rowIndex); @@ -1360,6 +1367,7 @@ public class Grid extends ResizeComposite implements bindTimeout.cancel(); rowIndex = request.getRowIndex(); + focusedColumnIndex = request.getColumnIndex(); showOverlay(); } } @@ -1371,6 +1379,7 @@ public class Grid extends ResizeComposite implements doCancel(); } else { state = State.ACTIVE; + // TODO: Maybe restore focus? } bindTimeout.cancel(); @@ -1482,15 +1491,22 @@ public class Grid 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 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 extends ResizeComposite implements throw new IllegalStateException( "Cannot cancel edit: editor is not in edit mode"); } - handler.cancel(new EditorRequestImpl(grid, rowIndex, null)); + handler.cancel(new EditorRequestImpl(grid, rowIndex, + focusedColumnIndex, null)); doCancel(); } @@ -1605,7 +1625,7 @@ public class Grid extends ResizeComposite implements setButtonsEnabled(false); saveTimeout.schedule(SAVE_TIMEOUT_MS); EditorRequest request = new EditorRequestImpl(grid, rowIndex, - saveRequestCallback); + focusedColumnIndex, saveRequestCallback); handler.save(request); updateSelectionCheckboxesAsNeeded(true); } @@ -1664,12 +1684,12 @@ public class Grid 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 request = new EditorRequestImpl(grid, - rowIndex, bindRequestCallback); + rowIndex, columnIndex, bindRequestCallback); handler.bind(request); grid.getEscalator().setScrollLocked(Direction.VERTICAL, isBuffered()); @@ -1798,7 +1818,9 @@ public class Grid 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 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 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 @@ -6834,8 +6877,30 @@ public class Grid 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 extends ResizeComposite implements } private boolean handleEditorEvent(Event event, RowContainer container) { - return getEditor().getEventHandler().handleEvent( - new EditorDomEvent(event, getEventCell(), editor - .getWidget(eventCell.getColumn()))); + Widget w; + if (editor.focusedColumnIndex < 0) { + w = null; + } else { + w = editor.getWidget(getColumn(editor.focusedColumnIndex)); + } + + EditorDomEvent editorEvent = new EditorDomEvent(event, + getEventCell(), w); + + return getEditor().getEventHandler().handleEvent(editorEvent); } private boolean handleRendererEvent(Event event, RowContainer container) { -- cgit v1.2.3