This is the src for the focus-finder demo website. I will be open sourcing focus-finder soon. This is just the demo website.
Find the demo here: https://duder-onomy.github.io/focus-finder-demo/
The intention here is to update the core lib and eventually, if compatible, send a pr to https://github.com/SolidDigital/magic-focus-finder The original repo I worked on with with some co-werkz in 2016?ish
Brain dump / Explanation / Making plans for refactors after demos are complete.
How focus finder works right now.
How it works right now:=
- We calculate a position for each element when it is inserted into the dom (via Mutation Observer) and append it to a list of known focusables.
- When a direction key is pressed, (lets say the person pressed the right arrow key)
- We check for any element level direction overrides, they have them, we use them to push focus and bail out.
- We toss out all elements whose center left is less than the currently focused elements center right.
- If the distance exceeds the maxDistance, we bail out
- If the element contains a special class, which bounce
- For elements beyond the maxDistance, we reduce weight drastically, but still keep it in the set, this is to enable any ‘last ditch effort’ to push focus
- We compute and sort the list by its weighted result. If there are per element weights defined, we use those.
- Grab the first item, push focus it to.
function _getWeightedResult(azimuth, maxAzimuth = 1, azimuthWeight, distance, maxDistance = 1, distanceWeight) {
return azimuthWeight * (Math.abs(azimuth) / maxAzimuth) +
distanceWeight * (Math.abs(distance) / maxDistance);
}
Todo (how we want it to work):
- Canonical focus should be maintained by the browser, abandon our own mgmt of currentFocus and ask the browser for it every time. Use real focus always.
- Lets change the way any maxDistance or maxAzimuth calcs work to hard filter on the max, if no elements are left after the hard filter, increase the maxDistance and try again. This will, I think, simplify the code to a more declarative
map, filter, reduce
- Lets change how elements are registered. Somehow, can we get away with only calling
getBoundingClientRect
on a certain subset of the DOM? - Lets make sure we run in a raf so that the getBoundingClientRect calls are cheaper.
- Because all the positions are precalculated, everything breaks down when elements positions are dynamic.
How to use it right now:
this.mff = MFF
.configure(this.config)
.start();
this.mff
.getContainer()
.addEventListener('focus-moved', (event) => {});
{
keymap : {}, //override the default browser keymap
weightOverrideAttribute : 'weight-override', // weight-override-up, down, left, and right can be set to 'distance' or 'azimuth' and the other will be disregarded
focusableAttribute : '', //override the default data attribute used to denote focusability
focusedClass : '', // a focus class for non focusable elements
defaultFocusedElement : '', // a element reference, or a selector
container : '', //the container in which this thing lives, default to the document.,
eventNamespace : '', //custom namespace for events, default to 'magicFocusFinder'
overrideDirectionAttribute : 'focus-overrides',
captureFocusAttribute : 'capture-focus',
dynamicPositionAttribute : 'dynamic-position',
useRealFocus : true, // Will trigger `blur` and `focus` on the actual elements, if set to false, bypass this.
azimuthWeight : 5, // Higher value means that it will prefer elements in the direction it is going
distanceWeight : 1, // Higher value means that it will prefer elements that are closer
debug : false, // Setting to true will replace the elements innerHTML with the computed distance (weighted azimuth + weighted distance),
attributeWatchInterval : 100, // If your browser does not support mutation observers. This is how often it will check the known elements for attribute changes.
useNativeMutationObserver : true // You can force the repo to use the non native mutation observer fallback.
maxDistance: 500, // The max distance focus will travel, useful to avoid unwanted large jumps
ignoreClass: 'focus-ignore', // A class you can give to elements, focus finder will ignore them
}
All elements with config.focusableAttribute
can be given focus to. After starting and a key press, focus is given to config.defaultFocusedElement
, or the first element found with config.focusdClass
, or the first element found with the focusable attribute.
It will fire the following events in the following order - events bubble up and are cancelable, so you can listen on mff.getContainer()
for all these events.
This example moves focus from Element One to Element Two
-
losing-focus
- on Element One -event.data
has{ "from" : domElementOne }
-
focus-lost
- on Element One -event.data
has{ "from" : domElementOne }
-
gaining-focus
- on Element Two -event.data
has{ "to" : domElementTwo }
-
focus-gained
- on Element Two -event.data
has{ "to" : domElementTwo }
-
focus-moved
- on Element Two -event.data
is the object:{ "direction" : "up|down|left|right", "from" : domElementOne, "to" : domElementTwo }
focus
normal focus eventblur
normal blur eventmagicFocusFinder:focus:from:<% direction %>
namespaced focus event telling you which direction the focus came frommagicFocusFinder:blur:to:<% direction %>
namespaced blur event telling you which direction the element was blurred to.
Implemented
Not Implemented:
magicFocusFinder:elementAdded
event telling you a focusable element was added to the container.magicFocusFinder:elementRemoved
event telling you a focusable element was removed from the container.
You can specify on a per element basis whether you want distance
or azimuth
to be the only factor in determining the
next focused element. You have to specify up, down, left, or right following the weightOverrideAttribute
. Element attributes will be watched for changes and appropriately updated.
<div weight-override-up='azimuth' focusable> </div>
You can specify on a per element basis if you want to override the default behavior of that direction fired. This will be recalculated on each movement so you can use this attribute to change the directional behavior in real time if you need. The direction overrides will take a normal single line selector. The order of the overrides occur 'up' 'right' 'down' and 'left'. You can also used the keys null and skip. Null will bypass the direction override for that direction. In the example below, when moving down it will use the default focus behavior. If you use the work 'skip' it will skip this direction entirely, essentially canceling the movement. In the example below, when you go left, nothing will happen. Please not, the selectors for the overrides need to be a simple word as we split the directions on a space. Also note, the selectors must exist in inside your configured container.
<div focus-overrides='.something-up #somethingRight null skip'></div>
configure the defaults, can be called at anytime to change the configuration
magicFocusFinder.configure(options)
returns the current configuration
magicFocusFinder.getConfig()
returns the overall container - events bubble up to this
starts the dang thing, if start is called before configure, then default options will be used.
magicFocusFinder.start();
Prevents all navigation from key events.
magicFocusFinder.lock();
Allows navigation from key events.
magicFocusFinder.unlock();
will refresh the element mapping (should only be used if your browser does not support mutation observers) Otherwise it will watch the DOM for new elements.
magicFocusFinder.refresh();
set the current focused element, element ref or selector
magicFocusFinder.setCurrent();
get the current focused element, element ref of selector
magicFocusFinder.getCurrent()
movedirection
Tell the focus to move in a direction, this will bypass a lock, if one was set via mff.lock()
.
mff.move.right()
mff.move.down({ events : false });
Can move up, down, left, or right. Can turn off all events for a move by setting the events key to false
on the options object.
Returns an array of all elements that can be focused by mff.
magicFocusFinder.getKnownElements();
destroy this thing and free up all memory
magicFocusFinder.destroy();
Rawk.