diff options
author | John Alhroos <john.ahlroos@itmill.com> | 2010-04-29 12:26:11 +0000 |
---|---|---|
committer | John Alhroos <john.ahlroos@itmill.com> | 2010-04-29 12:26:11 +0000 |
commit | 001309748a3836f64ffc5a73b62314934cc7c018 (patch) | |
tree | 985ecc8e1f8799ce20c8f8ae0f41b154b03294f5 /src | |
parent | b76d365acf27491b795c5872623e96f79d50c6b0 (diff) | |
download | vaadin-framework-001309748a3836f64ffc5a73b62314934cc7c018.tar.gz vaadin-framework-001309748a3836f64ffc5a73b62314934cc7c018.zip |
Added keyboard navigation to Table #2390
svn changeset:12925/svn branch:6.4
Diffstat (limited to 'src')
-rw-r--r-- | src/com/vaadin/terminal/gwt/client/ui/VScrollTable.java | 437 | ||||
-rw-r--r-- | src/com/vaadin/ui/Table.java | 12 |
2 files changed, 425 insertions, 24 deletions
diff --git a/src/com/vaadin/terminal/gwt/client/ui/VScrollTable.java b/src/com/vaadin/terminal/gwt/client/ui/VScrollTable.java index b04e5f998d..17b822a204 100644 --- a/src/com/vaadin/terminal/gwt/client/ui/VScrollTable.java +++ b/src/com/vaadin/terminal/gwt/client/ui/VScrollTable.java @@ -24,6 +24,15 @@ import com.google.gwt.dom.client.Style.Display; import com.google.gwt.dom.client.Style.Position; import com.google.gwt.dom.client.Style.Unit; import com.google.gwt.dom.client.Style.Visibility; +import com.google.gwt.event.dom.client.BlurEvent; +import com.google.gwt.event.dom.client.BlurHandler; +import com.google.gwt.event.dom.client.FocusEvent; +import com.google.gwt.event.dom.client.FocusHandler; +import com.google.gwt.event.dom.client.KeyCodes; +import com.google.gwt.event.dom.client.KeyDownEvent; +import com.google.gwt.event.dom.client.KeyDownHandler; +import com.google.gwt.event.dom.client.KeyPressEvent; +import com.google.gwt.event.dom.client.KeyPressHandler; import com.google.gwt.event.dom.client.ScrollEvent; import com.google.gwt.event.dom.client.ScrollHandler; import com.google.gwt.user.client.Command; @@ -34,6 +43,7 @@ import com.google.gwt.user.client.Event; import com.google.gwt.user.client.Timer; import com.google.gwt.user.client.Window; import com.google.gwt.user.client.ui.FlowPanel; +import com.google.gwt.user.client.ui.FocusPanel; import com.google.gwt.user.client.ui.Panel; import com.google.gwt.user.client.ui.RootPanel; import com.google.gwt.user.client.ui.ScrollPanel; @@ -81,9 +91,12 @@ import com.vaadin.terminal.gwt.client.ui.dd.VerticalDropLocation; * TODO implement unregistering for child components in Cells */ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, - VHasDropHandler { + VHasDropHandler, KeyPressHandler, KeyDownHandler, FocusHandler, + BlurHandler { public static final String CLASSNAME = "v-table"; + public static final String CLASSNAME_SELECTION_FOCUS = CLASSNAME + "-focus"; + public static final String ITEM_CLICK_EVENT_ID = "itemClick"; public static final String HEADER_CLICK_EVENT_ID = "handleHeaderClick"; public static final String FOOTER_CLICK_EVENT_ID = "handleFooterClick"; @@ -127,6 +140,17 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, private final HashSet<String> selectedRowKeys = new HashSet<String>(); + /* + * We need to store the selection head so we now what range to select if we + * are selecting a range which is over several pages long + */ + private int lastSelectedRowKey = -1; + + /* + * The currently focused row + */ + private VScrollTableRow focusedRow; + /** * Represents a select range of rows */ @@ -179,6 +203,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, private final TableFooter tFoot = new TableFooter(); private final ScrollPanel bodyContainer = new ScrollPanel(); + private final FocusPanel bodyContainerFocus = new FocusPanel(bodyContainer); private int totalRows; @@ -220,20 +245,166 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, private int dragmode; private int multiselectmode; - private int lastSelectedRowKey = -1; + + public VScrollTable() { + bodyContainerFocus.setStyleName(CLASSNAME + "-body-wrapper"); + + /* + * Firefox handler auto-repeat works correctly only if we use a key + * press handler, other browsers handle it correctly when using a key + * down handler + */ + if (BrowserInfo.get().isGecko()) { + bodyContainerFocus.addKeyPressHandler(this); + } else { + bodyContainerFocus.addKeyDownHandler(this); + } + + bodyContainerFocus.addFocusHandler(this); + bodyContainerFocus.addBlurHandler(this); + bodyContainer.addScrollHandler(this); bodyContainer.setStyleName(CLASSNAME + "-body"); setStyleName(CLASSNAME); + add(tHead); - add(bodyContainer); + add(bodyContainerFocus); add(tFoot); rowRequestHandler = new RowRequestHandler(); } + /** + * Moves the selection head one row downloads + * + * @return Returns true if succeeded, else false if the selection could not + * be move downwards + */ + private boolean moveSelectionDown() { + if (selectMode > VScrollTable.SELECT_MODE_NONE) { + if (focusedRow == null) { + setRowFocus((VScrollTableRow) scrollBody.iterator().next()); + return true; + } else { + VScrollTableRow next = getNextRow(focusedRow); + if (next != null) { + setRowFocus(next); + return true; + } + } + } + + return false; + } + + /** + * Moves the selection head one row upwards + * + * @return Returns true if succeeded, else false if the selection could not + * be move upwards + * + */ + private boolean moveSelectionUp() { + if (selectMode > VScrollTable.SELECT_MODE_NONE) { + if (focusedRow == null) { + setRowFocus((VScrollTableRow) scrollBody.iterator().next()); + return true; + } else { + VScrollTableRow prev = getPreviousRow(focusedRow); + if (prev != null) { + setRowFocus(prev); + return true; + } + } + } + + return false; + } + + /** + * Selects a row where the current selection head is + * + * @param ctrlSelect + * Is the selection a ctrl+selection + * @param shiftSelect + * Is the selection a shift+selection + * @return Returns truw + */ + private void selectFocusedRow(boolean ctrlSelect, boolean shiftSelect) { + if (focusedRow != null) { + if (ctrlSelect && !shiftSelect && selectMode == SELECT_MODE_MULTI) { + focusedRow.toggleSelection(ctrlSelect); + } else if (!ctrlSelect && shiftSelect + && selectMode == SELECT_MODE_MULTI) { + focusedRow.toggleShiftSelection(true); + } else if (ctrlSelect && shiftSelect + && selectMode == SELECT_MODE_MULTI) { + focusedRow.toggleShiftSelection(false); + } else { + deselectAll(); + focusedRow.toggleSelection(true); + } + } + } + + /** + * Get the key that moves the selection head upwards. By default it is the + * up arrow key but by overriding this you can change the key to whatever + * you want. + * + * @return The keycode of the key + */ + protected int getNavigationUpKey() { + return KeyCodes.KEY_UP; + } + + /** + * Get the key that moves the selection head downwards. By default it is the + * down arrow key but by overriding this you can change the key to whatever + * you want. + * + * @return The keycode of the key + */ + protected int getNavigationDownKey() { + return KeyCodes.KEY_DOWN; + } + + /** + * Get the key that scrolls to the left in the table. By default it is the + * left arrow key but by overriding this you can change the key to whatever + * you want. + * + * @return The keycode of the key + */ + protected int getNavigationLeftKey() { + return KeyCodes.KEY_LEFT; + } + + /** + * Get the key that scroll to the right on the table. By default it is the + * right arrow key but by overriding this you can change the key to whatever + * you want. + * + * @return The keycode of the key + */ + protected int getNavigationRightKey() { + return KeyCodes.KEY_RIGHT; + } + + /** + * Get the key that selects an item in the table. By default it is the space + * bar key but by overriding this you can change the key to whatever you + * want. + * + * @return + */ + protected int getNavigationSelectKey() { + return 32; + } + @SuppressWarnings("unchecked") public void updateFromUIDL(UIDL uidl, ApplicationConnection client) { rendering = true; @@ -314,10 +485,12 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, if (uidl.hasVariable("selected")) { final Set<String> selectedKeys = uidl .getStringArrayVariableAsSet("selected"); - selectedRowKeys.clear(); - selectedRowRanges.clear(); + deselectAll(); for (String string : selectedKeys) { - selectedRowKeys.add(string); + VScrollTableRow row = getRenderedRowByKey(string); + if (row != null && !row.isSelected()) { + row.toggleSelection(false); + } } } @@ -621,6 +794,13 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, return tHead.getHeaderCell(colKey).getWidth(); } + /** + * Get a rendered row by its key + * + * @param key + * The key to search with + * @return + */ private VScrollTableRow getRenderedRowByKey(String key) { final Iterator<Widget> it = scrollBody.iterator(); VScrollTableRow r = null; @@ -632,6 +812,52 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, } return null; } + + /** + * Returns the next row to the given row + * + * @param row + * The row to calculate from + * @return The next row or null if no row exists + */ + private VScrollTableRow getNextRow(VScrollTableRow row){ + final Iterator<Widget> it = scrollBody.iterator(); + VScrollTableRow r = null; + while (it.hasNext()) { + r = (VScrollTableRow) it.next(); + if(r == row){ + if (it.hasNext()) { + return (VScrollTableRow) it.next(); + } else { + break; + } + } + } + + return null; + } + + /** + * Returns the previous row from the given row + * + * @param row + * The row to calculate from + * @return The previous row or null if no row exists + */ + private VScrollTableRow getPreviousRow(VScrollTableRow row) { + final Iterator<Widget> it = scrollBody.iterator(); + VScrollTableRow r = null; + VScrollTableRow prev = null; + while (it.hasNext()) { + r = (VScrollTableRow) it.next(); + if (r == row) { + return prev; + } else { + prev = r; + } + } + return null; + } private void reOrderColumn(String columnKey, int newIndex) { @@ -1255,6 +1481,9 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, onResizeEvent(event); } else { handleCaptionEvent(event); + if (DOM.eventGetType(event) == Event.ONMOUSEUP) { + onFocus(null); + } } } } @@ -1289,10 +1518,11 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, } /** - * Fires a header click event after the user has clicked a column header cell + * Fires a header click event after the user has clicked a column header + * cell * * @param event - * The click event + * The click event */ private void fireHeaderClickedEvent(Event event) { if (client.hasEventListeners(VScrollTable.this, @@ -1300,8 +1530,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, MouseEventDetails details = new MouseEventDetails(event); client.updateVariable(paintableId, "headerClickEvent", details .toString(), false); - client - .updateVariable(paintableId, "headerClickCID", cid, + client.updateVariable(paintableId, "headerClickCID", cid, immediate); } } @@ -2122,6 +2351,10 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, public void onBrowserEvent(Event event) { if (enabled && event != null) { handleCaptionEvent(event); + + if (DOM.eventGetType(event) == Event.ONMOUSEUP) { + onFocus(null); + } } } @@ -2927,7 +3160,8 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, rowElement = Document.get().createTRElement(); setElement(rowElement); DOM.sinkEvents(getElement(), Event.MOUSEEVENTS - | Event.ONDBLCLICK | Event.ONCONTEXTMENU); + | Event.ONDBLCLICK | Event.ONCONTEXTMENU + | Event.ONKEYDOWN); } private void paintComponent(Paintable p, UIDL uidl) { @@ -3117,6 +3351,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, client.updateVariable(paintableId, "clickedKey", "" + rowKey, false); + if (getElement() == targetTdOrTr.getParentElement()) { /* A specific column was clicked */ int childIndex = DOM.getChildIndex(getElement(), @@ -3172,6 +3407,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, case Event.ONMOUSEUP: mDown = false; handleClickEvent(event, targetTdOrTr); + onFocus(null); if (event.getButton() == Event.BUTTON_LEFT && selectMode > Table.SELECT_MODE_NONE) { @@ -3181,6 +3417,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, && selectMode == SELECT_MODE_MULTI && multiselectmode == MULTISELECT_MODE_DEFAULT) { toggleShiftSelection(false); + setRowFocus(this); // Ctrl click } else if ((event.getCtrlKey() || event @@ -3188,6 +3425,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, && selectMode == SELECT_MODE_MULTI && multiselectmode == MULTISELECT_MODE_DEFAULT) { toggleSelection(true); + setRowFocus(this); // Shift click } else if (event.getShiftKey() @@ -3200,14 +3438,16 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, if (multiselectmode == MULTISELECT_MODE_DEFAULT) { deselectAll(); } + toggleSelection(multiselectmode == MULTISELECT_MODE_DEFAULT); + setRowFocus(this); } // Remove IE text selection hack if (BrowserInfo.get().isIE()) { ((Element) event.getEventTarget().cast()) - .setPropertyJSO( - "onselectstart", null); + .setPropertyJSO("onselectstart", + null); } // Note: changing the immediateness of this @@ -3229,6 +3469,7 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, false); } + // Send the selected rows client .updateVariable( paintableId, @@ -3301,9 +3542,9 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, // Prevent default text selection in IE if (BrowserInfo.get().isIE()) { ((Element) event.getEventTarget().cast()) - .setPropertyJSO( - "onselectstart", - applyDisableTextSelectionIEHack()); + .setPropertyJSO( + "onselectstart", + applyDisableTextSelectionIEHack()); } event.stopPropagation(); @@ -3407,8 +3648,8 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, selectedRowKeys.add(String.valueOf(rowKey)); addStyleName("v-selected"); } else { - selectedRowKeys.remove(String.valueOf(rowKey)); removeStyleName("v-selected"); + selectedRowKeys.remove(String.valueOf(rowKey)); } } @@ -3425,7 +3666,8 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, * Ensures that we are in multiselect mode and that we have a * previous selection which was not a deselection */ - if (selectedRowKeys.isEmpty() || lastSelectedRowKey < 0) { + if (selectedRowKeys.isEmpty() || lastSelectedRowKey < 0 + || selectMode == SELECT_MODE_SINGLE) { // No previous selection found deselectAll(); toggleSelection(true); @@ -3454,10 +3696,11 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, .valueOf(r)); if (row != null && !row.isSelected()) { row.toggleSelection(false); + selectedRowKeys.add(String.valueOf(r)); } } } - + // Toggle clicked rows selection toggleSelection(false); @@ -4128,4 +4371,160 @@ public class VScrollTable extends FlowPanel implements Table, ScrollHandler, } + /** + * Moves the selection head to a specific row + * + * @param row + * The row to where the selection head should move + */ + private void setRowFocus(VScrollTableRow row) { + + if (selectMode == SELECT_MODE_NONE) { + return; + } + + // Remove previous selection + if (focusedRow != null) { + focusedRow.removeStyleName(CLASSNAME_SELECTION_FOCUS); + } + + if (row != null) { + // Apply focus style to new selection + focusedRow = row; + focusedRow.addStyleName(CLASSNAME_SELECTION_FOCUS); + + // Scroll up or down if needed + int rowTop = focusedRow.getElement().getAbsoluteTop(); + int scrollTop = bodyContainer.getElement().getAbsoluteTop(); + int scrollBottom = scrollTop + + bodyContainer.getElement().getOffsetHeight(); + if (rowTop > scrollBottom - focusedRow.getOffsetHeight()) { + bodyContainer.setScrollPosition(bodyContainer + .getScrollPosition() + + focusedRow.getOffsetHeight()); + } else if (rowTop < scrollTop) { + bodyContainer.setScrollPosition(bodyContainer + .getScrollPosition() + - focusedRow.getOffsetHeight()); + } + } + } + + /** + * Handles the keyboard events handled by the table + * + * @param event + * The keyboard event received + */ + private void handleNavigation(Event event) { + + // Down navigation + if (selectMode == SELECT_MODE_NONE + && event.getKeyCode() == getNavigationDownKey()) { + bodyContainer + .setScrollPosition(bodyContainer.getScrollPosition() + 5); + event.preventDefault(); + } else if (event.getKeyCode() == getNavigationDownKey()) { + if (moveSelectionDown()) { + if (event.getShiftKey()) { + selectFocusedRow(false, true); + } + event.preventDefault(); + } + } + + // Up navigation + if (selectMode == SELECT_MODE_NONE + && event.getKeyCode() == getNavigationUpKey()) { + bodyContainer + .setScrollPosition(bodyContainer.getScrollPosition() - 5); + event.preventDefault(); + } else if (event.getKeyCode() == getNavigationUpKey()) { + if (moveSelectionUp()) { + if (event.getShiftKey()) { + selectFocusedRow(false, true); + } + event.preventDefault(); + } + + // Left navigation + } else if (event.getKeyCode() == getNavigationLeftKey()) { + bodyContainer.setHorizontalScrollPosition(bodyContainer + .getHorizontalScrollPosition() - 5); + event.preventDefault(); + + // Right navigation + } else if (event.getKeyCode() == getNavigationRightKey()) { + bodyContainer.setHorizontalScrollPosition(bodyContainer + .getHorizontalScrollPosition() + 5); + event.preventDefault(); + } + + // Select navigation + if (selectMode > SELECT_MODE_NONE + && event.getKeyCode() == getNavigationSelectKey()) { + selectFocusedRow(event.getCtrlKey() || event.getMetaKey(), + event.getShiftKey()); + + event.preventDefault(); + + // Send the selected rows + client + .updateVariable(paintableId, "selected", + selectedRowKeys + .toArray(new String[selectedRowKeys + .size()]), immediate); + } + } + + /* + * (non-Javadoc) + * + * @see + * com.google.gwt.event.dom.client.KeyPressHandler#onKeyPress(com.google + * .gwt.event.dom.client.KeyPressEvent) + */ + public void onKeyPress(KeyPressEvent event) { + handleNavigation((Event) event.getNativeEvent().cast()); + } + + /* + * (non-Javadoc) + * + * @see + * com.google.gwt.event.dom.client.KeyDownHandler#onKeyDown(com.google.gwt + * .event.dom.client.KeyDownEvent) + */ + public void onKeyDown(KeyDownEvent event) { + handleNavigation((Event) event.getNativeEvent().cast()); + } + + /* + * (non-Javadoc) + * + * @see + * com.google.gwt.event.dom.client.FocusHandler#onFocus(com.google.gwt.event + * .dom.client.FocusEvent) + */ + public void onFocus(FocusEvent event) { + // Move focus from wrapper to container in FF, ignored in other browsers + if (BrowserInfo.get().isFF3()) { + bodyContainer.getElement().focus(); + } + if (focusedRow == null) { + setRowFocus((VScrollTableRow) scrollBody.iterator().next()); + } + } + + /* + * (non-Javadoc) + * + * @see + * com.google.gwt.event.dom.client.BlurHandler#onBlur(com.google.gwt.event + * .dom.client.BlurEvent) + */ + public void onBlur(BlurEvent event) { + setRowFocus(null); + } + } diff --git a/src/com/vaadin/ui/Table.java b/src/com/vaadin/ui/Table.java index 234173c2f1..dcebfa2672 100644 --- a/src/com/vaadin/ui/Table.java +++ b/src/com/vaadin/ui/Table.java @@ -1926,11 +1926,13 @@ public class Table extends AbstractSelect implements Action.Container, } // Add range items - for (String range : ranges) { - String[] limits = range.split("-"); - int start = Integer.valueOf(limits[0]); - int end = Integer.valueOf(limits[1]); - s.addAll(getItemIdsInRange(start, end)); + if (ranges != null) { + for (String range : ranges) { + String[] limits = range.split("-"); + int start = Integer.valueOf(limits[0]); + int end = Integer.valueOf(limits[1]); + s.addAll(getItemIdsInRange(start, end)); + } } setValue(s, true); |