-
Notifications
You must be signed in to change notification settings - Fork 0
/
keyboard-nav.js
119 lines (113 loc) · 3.4 KB
/
keyboard-nav.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
var KEY_RIGHT = 39;
var KEY_UP = 38;
var KEY_LEFT = 37;
var KEY_DOWN = 40;
var FIELD_TYPES = 'input, textarea, button'; //don't include dropdowns, radio btns as they have thier own behaviour with arrow keys
var MARGIN = 10;
$(function(){
$(FIELD_TYPES).on('keydown', processKeyDown);
});
function processKeyDown(event){
var source = event.target || event.srcElement;
var left = $(source).offset().left
var top = $(source).offset().top
console.log(left, top);
var compareTop, compareLeft, sortOn;
if(event.keyCode == KEY_RIGHT){
if(allowNavigation(source, 'right')){
compareTop = isTopSame;
compareLeft = isOnRight;
sortOn = compareLeft;
focusElement(source, compareTop, compareLeft, sortOn);
}
}else if(event.keyCode == KEY_LEFT){
if(allowNavigation(source, 'left')){
compareTop = isTopSame;
compareLeft = isOnLeft;
sortOn = compareLeft;
focusElement(source, compareTop, compareLeft, sortOn);
}
}else if(event.keyCode == KEY_DOWN){
compareTop = isDown;
compareLeft = isLeftSame;
sortOn = compareTop;
focusElement(source, compareTop, compareLeft, sortOn);
}else if(event.keyCode == KEY_UP){
compareTop = isUp;
compareLeft = isLeftSame;
sortOn = compareTop;
focusElement(source, compareTop, compareLeft, sortOn);
}
}
function focusElement(source, compareTop, compareLeft, sortOn){
var element = getElement(source, compareTop, compareLeft, sortOn);
$(element).focus();
}
function getElement(source, compareTop, compareLeft, sortOn){
var candidates = [];
$(FIELD_TYPES).each(function(){
console.log(this.id);
console.log($(this).offset().top, $(this).offset().left);
if(compareTop(this, source) && compareLeft(this, source)){
candidates.push(this);
}
});
$(candidates).each(function(){
console.log(this.id);
});
return $(candidates).sort(sortOn).first();
}
function isOnRight(a, b){
return $(a).offset().left > $(b).offset().left;
}
function isOnLeft(a, b){
return $(a).offset().left < $(b).offset().left;
}
function isDown(a, b){
return $(a).offset().top > $(b).offset().top;
}
function isUp(a, b){
return $(a).offset().top < $(b).offset().top;
}
function isTopSame(a, b){
return Math.abs($(a).offset().top - $(b).offset().top) <= MARGIN;
}
function isLeftSame(a, b){
return Math.abs($(a).offset().left - $(b).offset().left) <= MARGIN;
}
function allowNavigation(source, direction){
if($(source).is("input:checkbox")) return true;
var cursorPos = doGetCaretPosition(source);
var length = $(source).val().length;
console.log("cursorPos and length = ", cursorPos, length);
if(direction=='right' && cursorPos == length) return true;
if(direction=='left' && cursorPos == 0) return true;
return false;
}
/*
** Returns the caret (cursor) position of the specified text field.
** Return value range is 0-oField.value.length.
*/
function doGetCaretPosition (oField) {
var iCaretPos = 0;
try{
// IE Support
if (document.selection) {
// Set focus on the element
oField.focus ();
// To get cursor position, get empty selection range
var oSel = document.selection.createRange ();
// Move selection start to 0 position
oSel.moveStart ('character', -oField.value.length);
// The caret position is selection length
iCaretPos = oSel.text.length;
}
// Firefox support
else if (oField.selectionStart || oField.selectionStart == '0'){
iCaretPos = oField.selectionStart;
}
}catch(error){
iCaretPos = 0;
}
return (iCaretPos);
}