Skip to content

๐Ÿš€ Just install and get the navigation functionality by pressing the Tab key in Unity game

License

Notifications You must be signed in to change notification settings

dav-sea/EasyTab

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 

History

21 Commits
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

Repository files navigation

EasyTab

A simple package that allows you to easily implement Tab key navigation functionality between Selectables. This package is very easy to install and does not require any configuration to work, just add it to the project

Badge Badge Badge openupm Badge
Badge Badge GitHub Downloads (all assets, all releases)


Table of Contents

How to start

By default, the package is automatically embedded in PlayerLoop Unity. It is enough to install the package in any convenient way and Tab key navigation should work

Installation methods

Method 1. Downlaod and import unitypackage
๐Ÿ‘‰ Just download EasyTab.unitypackage on the release page and import it into your project
Method 2. Via Git URL

This installation method requires support for query parameters in the url to specify the path to the package.

Requires a unity version of at least 2019.3.4 (If you are using a later version of unity, just use a different installation method)

  1. Go to menu Window โ†’ PackageManager
  2. Click + button
  3. Paste url
    https://github.com/dav-sea/EasyTab.git?path=Assets/EasyTab
    
  4. Click Add button

image

[!TIP]

You can also specify the target version of the package by adding #1.0.0 at the end of the URL, for example: https://github.com/dav-sea/EasyTab.git?path=Assets/Package#1.0.0

Method 3. OpenUPM

You can install package via OpenUPM client:

openupm add com.dav-sea.easy-tab
Method 4. AssetStore
๐Ÿ‘‰ Download asset from AssetStore page

Everything is ready! If the installation of the package was successful, then now you can go to Play Mode and check how the Tab key navigation works in your project

gif

Features

Just use

โœ… Support for all components based on Selectable

๐Ÿš€ Zero configuration. Performing useful functionality immediately after installation

๐Ÿ”„ Automatic detection of the navigation sequence by traversing the Transform tree

โžก๏ธ Support for sequential navigation of the Selectable when pressing Tab key

โฉ Fast navigation by Tab key holding

๐Ÿ”™ As well as navigation in reverse order when using the Shift+Tab key combination

โ†ฉ๏ธ Processing of the Return key presses when the focus is on the text field (or custom component)

๐Ÿ”ค Additional support for input fields, including TextMeshPro (multiline case and Tab non-processing when typing)

โ˜ฐ Dropdown support. Items inside Dropdown are automatically navigated cyclically (Roll)

โฎ๏ธ Navigate to the last selected object when the focus is not set

๐Ÿ’ก Automatic checking of objects for accessibility (alive, active, intractable, canvas-group)

๐ŸŽจ Canvas support of all types (world-space, screen space overlay, screen space camera)

๐Ÿ’ซ Automatic navigation between multiple canvases and multiple scenes

โŒจ๏ธ Support for all input modes (InputSystem V1 (old), InputSystem V2 (new), Both mode)

๐Ÿƒ No allocations during navigation operation

๐Ÿค Minimal and simple code base


With EasyTab component

๐Ÿงฉ Easy navigation configuration using the EasyTab component for GameObject via inspector window

๐Ÿ”€ The ability to explicitly set the navigation sequence using jumps

โœ‚๏ธ The ability to exclude Selectable or children for navigation

๐Ÿ”  The ability to override navigation options

๐Ÿšง The ability to determine the behavior when reaching the boundaries (๐Ÿ” Roll / ๐Ÿ”› Clamp / โคด๏ธ Escape)


Advanced </>

๐Ÿ”Œ The package code is defined in the EasyTab.Runtime assembly (don't forget to include it in your assembly)

๐Ÿ”ง The ability to change all fields of the EasyTab component through the code

โš™๏ธ The ability to customize global behavior using EasyTabIntegration.Globally

๐ŸŒฑ๏ธ The ability to create local instances of EasyTabIntegration and EasyTabSolver

๐Ÿ•น๏ธ The ability to independently control navigation through the code

โœจ The ability to define additional blocking conditions for the availability of objects (by decorate default Driver)

Configuration

The package provides several configuration levels: EasyTab (component), EasyTabIntegration, EasyTabSolver

Configuration via EasyTab component

Component configuration is a universal way to easily and quickly configure navigation, because for this you just need to add the component to the GameObject and set the necessary parameters using the inspector or code.

It is great in cases where you need to define behavior in a local area of your project (for example, on a specific prefab or window)


SelectableRecognition

You can ignore Selectable on the same GameObject.

BorderMode

You can determine what the solver should do in cases when the extreme children (first or last) are reached when traversing the Transform tree

image

Escape (default): The solver will go beyond the boundaries of this Transform and search in neighboring Transform`s

Roll: The solver will NOT go beyond the boundaries of this Transform, but simply start navigation from the first available child element (๐Ÿ‘‰ the best solution for pop-ups and windows)

Clamp: The solver will NOT go beyond the boundaries of this Transform, but simply stop at the last selected

ChildrenExtracting

You can exclude all the children of this Transform from the tree

JumpingPolicy

Allows you to set the behavior of jumping to other elements during navigation.

Tip

The specified jumps will be used only when the definition of the next Selectable starts with this GameObject (that is, when it is specified as EventSystem.current.currentSelectedGameObject). When traversing the Transform tree, jumps will be ignored

DontUseJumps: Disables jumping

UseOnlyJumps: When navigating from this GameObject, the specified jumps will be used to determine the next Selectable. But if the specified jumps are unavailable (for example, not interactable), then the transition to the next Selectable will not occur.

UseJumpsOrHierarchy: When navigating from this GameObject, the specified jumps will be used to determine the next Selectable. But if the specified jumps are unavailable (for example, not interactable), then the next Selectable will be searched by traversing the Transform tree relative to this GameObject.

UseJumpsOrTheirNext: When navigating from this GameObject, the specified jumps will be used to determine the next Selectable. But if the specified jumps are unavailable (for example, not interactable), then the process of finding for the next Selectable will be started again, but relative GameObject specified as a jump (if it is not destroyed).

NextJump & ReverseJump

Jumps to the next and previous GameObject to be used when navigating from this GameObject. (Jumps are used only when JumpingPolicy is not DontUseJumps)

NavigationLock

You can block navigation when this GameObject is current. This can be useful when you don't need navigation on this GameObject. In Auto mode, navigation is blocked automatically when this GameObject contains an InputField in the multiline and focused state

EnterHandling

You can determine whether navigation on this GameObject should occur when you press the Enter key. In auto mode, when you press Enter, navigation will occur if this GameObject contains an InputField in the single-line state

Configuration via EasyTabIntegration class

EasyTabIntegration is a class that uses InputSystem and EventSystem to define the input and the currently selected object. As well as processing auxiliary functions, such as remembering the last selected object.

For ease of integration, one instance of class EasyTabIntegration is located in EasyTabIntegration.Globally. The update logic of this instance is embedded in PlayerLoop, and is always processed.

If this does not suit you, you can disable the processing of the global object from PlayerLoop via EasyTabIntegration.GloballyEnabled = false

You can call the update logic yourself at the time you need via EasyTabIntegration.Globally.UpdateAll().

You can change the time to start fast navigation through the SecondsForStartFastNavigation field as well as the speed of fast navigation through the SecondsBetweenSelectablesInFastNavigation field

Or you can make your own instance of EasyTabIntegration and work with it, ignoring EasyTabIntegration.Globally

You can also configure EasyTabIntegration.Solver (see the following section)

Configuration EasyTabSolver

EasyTabSolver is a class whose task is to decide which Selectable will be next. To configure it globally, use EasyTabIntegration.Globally.Solver


EntryPoint

The GameObject to be used as the initial one. Combined with the FallbackNavigationPolicy specified in the WhenCurrentIsNotSet and WhenCurrentIsNotSelectable fields

LastSelected

The last selected object. If this solver is specified in EasyTabIntegration, then this field is updated by it. Combined with the FallbackNavigationPolicy specified in the WhenCurrentIsNotSet and WhenCurrentIsNotSelectable fields

FallbackNavigationPolicy

You can set NavigationPolicy in different cases: WhenCurrentIsNotSet, WhenCurrentIsNotSelectable

AllowNavigateToEntryPoint: Allow navigation on EntryPoint as a fallback option

AllowNavigateToClosest: Allow to find the nearest Selectable (works only if the current GameObject is not null) as a fallback option

AllowNavigateToLastSelected: Allow navigation to the last selected object (especially useful when focus is lost when clicking into the void) as a fallback option

Integration with your tools

The idea of EasyTab is that it works "out of the box" and does not require any configuration. However, if you use third-party tools and unusual approaches to hide user interface elements and windows, then navigation may not work as expected.

EasyTab needs to understand which objects are really active and which are hidden. To do this, EasyTab offers a simple API based on decorators.

In simple cases, you just need to describe in which cases the element is not Selectable, for example:

var solver = EasyTabIntergation.Globally.Solver;

// we decorate the default driver with a handler that defines elements with a Graphic component and a transparent color as not selectable 
solver.Driver = solver.Driver
    .WithSelectableBlocking(t => t.TryGetComponent<Graphic>(out var graphic) && graphic.color.a <= 0);

A common practice when developing a user interface in Unity is to hide elements by changing transparency, moving outside the screen space, zooming out, etc.

However, EasyTab does not know about your specific implementation of hiding windows/popups, so you can use the API too, but to exclude the hierarchy of objects from navigation, for example:

var solver = EasyTabIntergation.Globally.Solver;

// we decorate the default driver with a handler that excludes from navigation all children of elements that have a disabled window.
solver.Driver = solver.Driver
    .WithTraversingChildrenBlocking(t => t.TryGetComponent(out Window window) && !window.IsActive);

You can decorate any IEasyTabDriver method.

For example, you can decorate GetBorderMode and change the BorderMode for all windows through the decorator, so as not to override the BorderMode through the EasyTab component

var solver = EasyTabIntergation.Globally.Solver;

solver.Driver = solver.Driver.DecorateBorderMode((@base, target) =>
{
    // We want to navigate all the elements inside the window cyclically
    if (target.IsTransform(out var targetTransform) && targetTransform.TryGetComponent<Window>(out _))
        return BorderMode.Roll;

    // If the target is not a window, then use the base decorator
    return @base.GetBorderMode(target);
});