public class TabIndexManager
extends java.lang.Object
The TabIndexManager separates the logic required to maintain a sensible tab-order for a page's components from the logic to handle allocation of actual tab index values. It is common to have non-focusable components with an implied position in the page's tab order - for example Layouts containing focusable buttons, or DynamicForms containing focusable items, and this class handles maintaining relative tab order within such groups, and supplying explicit TabIndex values for the items which actually need them.
Entries are registered with the TabIndexManager via
the addTarget()
API. A numeric tab index for each entry
will be lazily generated when requested via getTabIndex()
. The class provides APIs to modify the position of entries in the tab tree. When a target is registered,
a couple of custom callback functions can be provided. The first is a notification method for the tab index being
updated (due to, for example, a parent being repositioned and all its children having new tab indices assigned), and can
be used to take an appropriate action such as updating the tab index of an element in the DOM. The second callback will
be fired when a call to the special focusInTarget()
or
shiftFocus()
API requests focus be passed to an entry.
This allows a developer to take an appropriate action (such as programmatically focussing in some DOM element).
See
the tab order overview
topic for more information on tab order
management for components in Smart GWT.
Constructor and Description |
---|
TabIndexManager() |
Modifier and Type | Method and Description |
---|---|
static void |
addTarget(java.lang.String ID,
boolean canFocus)
Register a target to have its tab order position managed by the TabIndexManager.
|
static void |
addTarget(java.lang.String ID,
boolean canFocus,
java.lang.String parentID) |
static void |
addTarget(java.lang.String ID,
boolean canFocus,
java.lang.String parentID,
java.lang.Integer position) |
static void |
addTarget(java.lang.String ID,
boolean canFocus,
java.lang.String parentID,
java.lang.Integer position,
TabIndexUpdatedCallback tabIndexUpdatedCallback) |
static void |
addTarget(java.lang.String ID,
boolean canFocus,
java.lang.String parentID,
java.lang.Integer position,
TabIndexUpdatedCallback tabIndexUpdatedCallback,
ShiftFocusCallback shiftFocusCallback)
Register a target to have its tab order position managed by the TabIndexManager.
|
static boolean |
focusInTarget(java.lang.String ID)
Request the TabIndexManager shift focus to a registered focus target.
|
static java.lang.String |
getAllocatedTabChain()
Get a report of the current hierarchy of targets passed to
addTarget() together with current canFocus state and tabIndex (if assigned). |
static java.lang.Integer |
getTabIndex(java.lang.String ID)
Returns a tabIndex number for some target ID registered via
addTarget() . |
static boolean |
hasTarget(java.lang.String ID)
Has the specified target been added to this TabIndexManager via
addTarget() ? |
static void |
moveTarget(java.lang.String ID)
Move a target to the newly specified parent / position.
|
static void |
moveTarget(java.lang.String ID,
java.lang.String parentID) |
static void |
moveTarget(java.lang.String ID,
java.lang.String parentID,
java.lang.Integer position)
Move a target to the newly specified parent / position.
|
static void |
moveTargets(java.lang.String[] IDs)
Move a list of targets to the newly specified parent / position.
|
static void |
moveTargets(java.lang.String[] IDs,
java.lang.String parentID) |
static void |
moveTargets(java.lang.String[] IDs,
java.lang.String parentID,
java.lang.Integer position)
Move a list of targets to the newly specified parent / position.
|
static void |
removeTarget(java.lang.String ID)
Removes a target from this TabIndexManager.
|
static void |
resumeCallbacks(java.lang.String[] targets)
Resume firing any callbacks suppressed by
suppressCallbacks() |
static void |
setCanFocus(java.lang.String ID,
boolean canFocus)
Modifies whether or not some specified target should be treated as focusable and provide a meaningful TabIndex on a call
to
getTabIndex() . |
static boolean |
shiftFocus(java.lang.String ID,
boolean forward)
Method to shift focus to the next registered focusable target.
|
static boolean |
shiftFocusAfterGroup(java.lang.String targetGroup,
boolean forward)
Method to shift focus to the next registered focusable target beyond some registered target and any targets registered
as children within its group via
addTarget() or moveTarget() . |
static boolean |
shiftFocusWithinGroup(java.lang.String targetGroup,
java.lang.String currentTarget,
boolean forward)
Method to shift focus to the next registered focusable target within some group.
|
static void |
showAllocatedTabChain()
Show the current hierarchy of targets passed to
addTarget() together with current canFocus state and tabIndex (if assigned). |
static void |
suppressCallbacks(java.lang.String[] targets)
Temporarily suppress firing any tabIndexChanged callback passed into
addTarget() for the specified targets should their tab index
change. |
public static boolean focusInTarget(java.lang.String ID)
This method does not directly change the focus
within the DOM - instead it invokes the shiftFocusCallback
registered for the specified target if it is
marked as canFocus:true
.
Returns false if the target had no no shiftFocusCallback
, the
shiftFocusCallback
returned false, or if the target is marked as not canFocus:true
ID
- target to shift focus topublic static java.lang.String getAllocatedTabChain()
addTarget()
together with current canFocus state and tabIndex (if assigned).public static java.lang.Integer getTabIndex(java.lang.String ID)
addTarget()
. Generated tab indices are guaranteed to be in order. As targets are added to, or moved within the
TabIndexManager, their tab index may become invalid. The tabIndexUpdated
notification will be fired when
this occurs, giving developers a way to pick up the new tab index, and assign it to the appropriate DOM element if
appropriate.
ID
- ID of the target for which you want to get a numeric tabIndex.public static boolean hasTarget(java.lang.String ID)
addTarget()
?ID
- Unique ID to test for.public static void moveTarget(java.lang.String ID)
ID
- ID of the target to movepublic static void moveTarget(java.lang.String ID, java.lang.String parentID)
moveTarget(java.lang.String)
public static void moveTarget(java.lang.String ID, java.lang.String parentID, java.lang.Integer position)
ID
- ID of the target to moveparentID
- ID of the new parent (if null, will move to the top level)position
- Position within the specified parent. If null will be the last entry.public static void moveTargets(java.lang.String[] IDs)
IDs
- IDs of the targets to movepublic static void moveTargets(java.lang.String[] IDs, java.lang.String parentID)
moveTargets(java.lang.String[])
public static void moveTargets(java.lang.String[] IDs, java.lang.String parentID, java.lang.Integer position)
IDs
- IDs of the targets to moveparentID
- ID of the new parent (if null, will move to the top level)position
- Position within the specified parent. If null will be added at the endpublic static void removeTarget(java.lang.String ID)
moveTarget()
to move
the children out of this parentID
- target to removepublic static void resumeCallbacks(java.lang.String[] targets)
suppressCallbacks()
targets
- targets for which callbacks should be resumedsuppressCallbacks(java.lang.String[])
public static void setCanFocus(java.lang.String ID, boolean canFocus)
getTabIndex()
.ID
- target IDcanFocus
- new value for canFocuspublic static boolean shiftFocus(java.lang.String ID, boolean forward)
This method does not directly change the focus within
the DOM - instead it finds the next target marked as canFocus:true
, and invokes the
shiftFocusCallback
registered for that target. This callback is expected to take the appropriate action
(typically shifting native focus to an element in the DOM), and return true (or return false, if the target could not
receieve focus for some reason, in which case we'll find the next canFocus:true
target and repeat the
action there.
Targets with no shiftFocusCallback
will be skipped entirely in this process.
ID
- current focus target. If null, focus will be applied to the first focusable target (or the last if the
forward
parameter is false).forward
- should focus move forward to the next focusable target, or backward to the previous focusable target.public static boolean shiftFocusAfterGroup(java.lang.String targetGroup, boolean forward)
addTarget()
or moveTarget()
. This method does not directly change the focus
within the DOM - instead it finds the next target marked as canFocus:true
, and invokes the
shiftFocusCallback
registered for that target. This callback is expected to take the appropriate action
(typically shifting native focus to an element in the DOM), and return true (or return false, if the target could not
receieve focus for some reason, in which case we'll find the next canFocus:true
target and repeat the
action there.
Targets with no shiftFocusCallback
will be skipped entirely in this process.
A return value of false indicates that this method was unable to shift focus to a new target.
targetGroup
- ID of registered target. Focus will be shifted to the next registered focusable element, skipping this group and its
descendants.forward
- should focus move forward to the next focusable target, or backward to the previous focusable target.public static boolean shiftFocusWithinGroup(java.lang.String targetGroup, java.lang.String currentTarget, boolean forward)
addTarget()
or moveTarget()
). The second parameter can be passed to specify an explicit starting position to shift focus from. If this is not present, this method will attempt to focus into the group target itself if moving forward (or its last child, if moving backward) and failing that, shift focus from there.
This method does not directly change the focus within the DOM - instead it finds the next target marked as
canFocus:true
, and invokes the shiftFocusCallback
registered for that target. This callback
is expected to take the appropriate action (typically shifting native focus to an element in the DOM), and return true
(or return false, if the target could not receieve focus for some reason, in which case we'll find the next
canFocus:true
target and repeat the action there.
Targets with no shiftFocusCallback
will
be skipped entirely in this process.
A return value of false indicates that this method was unable to shift focus to a new target.
targetGroup
- ID of registered target. Focus will be shifted within this target and its descendants only.currentTarget
- Optional ID of current focus target within the group focus will be shifted in the specified direction from this node.forward
- should focus move forward to the next focusable target, or backward to the previous focusable target.public static void showAllocatedTabChain()
addTarget()
together with current canFocus state and tabIndex (if assigned). Results are output to the developer
console.public static void suppressCallbacks(java.lang.String[] targets)
addTarget()
for the specified targets should their tab index
change. This is useful for cases where a developer is managing a list of items and wants to avoid any potential for multiple notifications until the entire list has been managed
targets
- targets for which callbacks should be suppressedresumeCallbacks(java.lang.String[])
public static void addTarget(java.lang.String ID, boolean canFocus)
ID
- Unique ID to associate with a tab position. For a Canvas this would typically be the Canvas.ID
but any unique string is valid.canFocus
- Is this target directly focusable? Governs whether an explicit tabIndex will be created for this target. This
parameter should be passed as false
for targets which do not require an explicit tabIndex as they are
not focusable, or not explicit tab-stops for the user tabbing through the page. They will still have an implicit tab
order position which governs where descendants appear, and would be used to generate a tabIndex if canFocus is
subsequently updated via setCanFocus()
.public static void addTarget(java.lang.String ID, boolean canFocus, java.lang.String parentID)
addTarget(java.lang.String, boolean)
public static void addTarget(java.lang.String ID, boolean canFocus, java.lang.String parentID, java.lang.Integer position)
addTarget(java.lang.String, boolean)
public static void addTarget(java.lang.String ID, boolean canFocus, java.lang.String parentID, java.lang.Integer position, TabIndexUpdatedCallback tabIndexUpdatedCallback)
addTarget(java.lang.String, boolean)
public static void addTarget(java.lang.String ID, boolean canFocus, java.lang.String parentID, java.lang.Integer position, TabIndexUpdatedCallback tabIndexUpdatedCallback, ShiftFocusCallback shiftFocusCallback)
ID
- Unique ID to associate with a tab position. For a Canvas this would typically be the Canvas.ID
but any unique string is valid.canFocus
- Is this target directly focusable? Governs whether an explicit tabIndex will be created for this target. This
parameter should be passed as false
for targets which do not require an explicit tabIndex as they are
not focusable, or not explicit tab-stops for the user tabbing through the page. They will still have an implicit tab
order position which governs where descendants appear, and would be used to generate a tabIndex if canFocus is
subsequently updated via setCanFocus()
.parentID
- For cases where the tab position should be treated part of a group to be moved together, the ID of the parent target
containing all members of this group. An example of this would be a Layout managing the tab order of all its members.
If present, the passed parentID must already be being managed by this TabIndexManager. May be updated for registered
targets via moveTarget()
.position
- Position in the tab-order within the specified parent [or within top level widgets]. Omitting this parameter will add
the target to the end of the specified parent's tab group. May be updated for registered targets via moveTarget()
.tabIndexUpdatedCallback
- This notification method will be fired when the tabIndex is actually updated, typically due to the target, or some
parent of it being re-positioned in the managed Tab order. In some cases tab indices may also be updated to make
space for unrelated entries being added to the TabIndexManager. This notification is typically used to update the
appropriate element in the DOM to reflect a new tab index.shiftFocusCallback
- This notification method will be when the special shiftFocus()
method is called to programmatically move focus through the registered targets (simulating the user
tabbing through elements in the tab index chain). The implementation should attempt to update the UI state by
focusing in the appropriate UI for this target -- typically this means putting browser focus into a DOM element, and
return true to indicate success.shiftFocus()
will simply skip this entry.