blob: 3cf1016bf8524c67310241bacb33c6b9fecf1328 [file] [log] [blame]
Simon Hunt1eecfa22014-12-16 14:46:29 -08001/*
Simon Hunt8ead3a22015-01-06 11:00:15 -08002 * Copyright 2014,2015 Open Networking Laboratory
Simon Hunt1eecfa22014-12-16 14:46:29 -08003 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17/*
Simon Huntdc6362a2014-12-18 19:55:23 -080018 ONOS GUI -- Util -- Key Handler Service
Simon Hunt1eecfa22014-12-16 14:46:29 -080019 */
Simon Huntdc6362a2014-12-18 19:55:23 -080020(function () {
Simon Hunt1eecfa22014-12-16 14:46:29 -080021 'use strict';
22
23 // references to injected services
Simon Hunt9f5e97d2015-12-11 10:32:09 -080024 var $log, $timeout, fs, ts, ns, qhs;
25
26 // constants
27 var eeggMin = 'shiftO',
28 eeggMax = 'shiftONOS';
Simon Hunt1eecfa22014-12-16 14:46:29 -080029
30 // internal state
Simon Hunt36fc15c2015-02-12 17:02:58 -080031 var enabled = true,
Simon Huntd5579252015-10-06 15:09:14 -070032 globalEnabled = true,
Simon Hunt36fc15c2015-02-12 17:02:58 -080033 keyHandler = {
Simon Hunt1eecfa22014-12-16 14:46:29 -080034 globalKeys: {},
35 maskedKeys: {},
36 viewKeys: {},
37 viewFn: null,
38 viewGestures: []
Simon Hunt9f5e97d2015-12-11 10:32:09 -080039 },
40 eegg = '';
Simon Hunt1eecfa22014-12-16 14:46:29 -080041
Simon Hunt9f5e97d2015-12-11 10:32:09 -080042 function layEgg(key) {
43 eegg += key;
44 if (eeggMax.indexOf(eegg) === 0) {
45 if (eegg === eeggMax) {
46 d3.select('body').append('div').attr('id', 'eegg')
47 .append('img').attr('src', 'data/img/eegg.png');
48 $timeout(function () { d3.select('#eegg').remove(); }, 3000);
Thomas Vachuskafe1f01a2015-12-10 23:56:02 -080049 eegg = '';
50 }
51 return true;
Simon Hunt9f5e97d2015-12-11 10:32:09 -080052 }
53 if (eegg !== eeggMin) {
Thomas Vachuskafe1f01a2015-12-10 23:56:02 -080054 eegg = '';
55 }
56 return false;
57 }
58
Simon Hunt1eecfa22014-12-16 14:46:29 -080059 function whatKey(code) {
60 switch (code) {
61 case 13: return 'enter';
62 case 16: return 'shift';
63 case 17: return 'ctrl';
64 case 18: return 'alt';
65 case 27: return 'esc';
66 case 32: return 'space';
67 case 37: return 'leftArrow';
68 case 38: return 'upArrow';
69 case 39: return 'rightArrow';
70 case 40: return 'downArrow';
71 case 91: return 'cmdLeft';
72 case 93: return 'cmdRight';
73 case 187: return 'equals';
Simon Hunt90dcc3e2015-03-25 15:01:27 -070074 case 188: return 'comma';
Simon Hunt1eecfa22014-12-16 14:46:29 -080075 case 189: return 'dash';
Simon Hunt90dcc3e2015-03-25 15:01:27 -070076 case 190: return 'dot';
Simon Hunt1eecfa22014-12-16 14:46:29 -080077 case 191: return 'slash';
78 case 192: return 'backQuote';
79 case 220: return 'backSlash';
80 default:
81 if ((code >= 48 && code <= 57) ||
82 (code >= 65 && code <= 90)) {
83 return String.fromCharCode(code);
84 } else if (code >= 112 && code <= 123) {
85 return 'F' + (code - 111);
86 }
87 return '.';
88 }
89 }
90
91 function keyIn() {
92 var event = d3.event,
93 keyCode = event.keyCode,
94 key = whatKey(keyCode),
95 kh = keyHandler,
96 gk = kh.globalKeys[key],
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -080097 gcb = fs.isF(gk) || (fs.isA(gk) && fs.isF(gk[0])),
Simon Hunt1eecfa22014-12-16 14:46:29 -080098 vk = kh.viewKeys[key],
Simon Hunt09060142015-03-18 20:23:32 -070099 kl = fs.isF(kh.viewKeys._keyListener),
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800100 vcb = fs.isF(vk) || (fs.isA(vk) && fs.isF(vk[0])) || fs.isF(kh.viewFn),
Simon Hunt1bf1e8c2015-04-08 10:12:43 -0700101 token = 'keyev'; // indicate this was a key-pressed event
Simon Hunt1eecfa22014-12-16 14:46:29 -0800102
Simon Huntdc6362a2014-12-18 19:55:23 -0800103 d3.event.stopPropagation();
104
Simon Hunt36fc15c2015-02-12 17:02:58 -0800105 if (enabled) {
Thomas Vachuskafe1f01a2015-12-10 23:56:02 -0800106 if (layEgg(key)) return;
107
Simon Hunt36fc15c2015-02-12 17:02:58 -0800108 // global callback?
109 if (gcb && gcb(token, key, keyCode, event)) {
110 // if the event was 'handled', we are done
111 return;
112 }
113 // otherwise, let the view callback have a shot
114 if (vcb) {
115 vcb(token, key, keyCode, event);
116 }
Simon Hunt09060142015-03-18 20:23:32 -0700117 if (kl) {
118 kl(key);
119 }
Simon Hunt1eecfa22014-12-16 14:46:29 -0800120 }
121 }
122
123 function setupGlobalKeys() {
Simon Hunt404f6b22015-01-21 14:00:56 -0800124 angular.extend(keyHandler, {
Simon Hunt1eecfa22014-12-16 14:46:29 -0800125 globalKeys: {
126 backSlash: [quickHelp, 'Show / hide Quick Help'],
127 slash: [quickHelp, 'Show / hide Quick Help'],
128 esc: [escapeKey, 'Dismiss dialog or cancel selections'],
129 T: [toggleTheme, "Toggle theme"]
130 },
131 globalFormat: ['backSlash', 'slash', 'esc', 'T'],
132
133 // Masked keys are global key handlers that always return true.
134 // That is, the view will never see the event for that key.
135 maskedKeys: {
136 slash: true,
137 backSlash: true,
138 T: true
139 }
140 });
141 }
142
143 function quickHelp(view, key, code, ev) {
Simon Huntd5579252015-10-06 15:09:14 -0700144 if (!globalEnabled) {
145 return false;
146 }
Simon Hunt639dc662015-02-18 14:19:20 -0800147 qhs.showQuickHelp(keyHandler);
Simon Hunt1eecfa22014-12-16 14:46:29 -0800148 return true;
149 }
150
Simon Hunta0eb0a82015-02-11 12:30:06 -0800151 // returns true if we 'consumed' the ESC keypress, false otherwise
Simon Hunt1eecfa22014-12-16 14:46:29 -0800152 function escapeKey(view, key, code, ev) {
Simon Hunt9d286562015-03-09 13:53:50 -0700153 return ns.hideIfShown() || qhs.hideQuickHelp();
Simon Hunt1eecfa22014-12-16 14:46:29 -0800154 }
155
156 function toggleTheme(view, key, code, ev) {
Simon Huntd5579252015-10-06 15:09:14 -0700157 if (!globalEnabled) {
158 return false;
159 }
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800160 ts.toggleTheme();
Simon Hunt1eecfa22014-12-16 14:46:29 -0800161 return true;
162 }
163
Simon Hunt59df0b22014-12-17 10:32:25 -0800164 function setKeyBindings(keyArg) {
165 var viewKeys,
166 masked = [];
167
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800168 if (fs.isF(keyArg)) {
Simon Hunt59df0b22014-12-17 10:32:25 -0800169 // set general key handler callback
170 keyHandler.viewFn = keyArg;
171 } else {
172 // set specific key filter map
173 viewKeys = d3.map(keyArg).keys();
174 viewKeys.forEach(function (key) {
175 if (keyHandler.maskedKeys[key]) {
Simon Huntaf322072014-12-18 13:23:40 -0800176 masked.push('setKeyBindings(): Key "' + key + '" is reserved');
Simon Hunt59df0b22014-12-17 10:32:25 -0800177 }
178 });
179
180 if (masked.length) {
Simon Huntaf322072014-12-18 13:23:40 -0800181 $log.warn(masked.join('\n'));
Simon Hunt59df0b22014-12-17 10:32:25 -0800182 }
183 keyHandler.viewKeys = keyArg;
184 }
185 }
186
187 function getKeyBindings() {
188 var gkeys = d3.map(keyHandler.globalKeys).keys(),
189 masked = d3.map(keyHandler.maskedKeys).keys(),
190 vkeys = d3.map(keyHandler.viewKeys).keys(),
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800191 vfn = !!fs.isF(keyHandler.viewFn);
Simon Hunt59df0b22014-12-17 10:32:25 -0800192
193 return {
194 globalKeys: gkeys,
195 maskedKeys: masked,
196 viewKeys: vkeys,
197 viewFunction: vfn
198 };
199 }
200
Bri Prebilic Cole9dcaea52015-07-21 14:39:48 -0700201 function unbindKeys() {
202 keyHandler.viewKeys = {};
203 keyHandler.viewFn = null;
204 keyHandler.viewGestures = [];
205 }
206
Simon Hunt71892222015-09-29 13:39:40 -0700207 function checkNotGlobal(o) {
208 var oops = [];
209 if (fs.isO(o)) {
210 angular.forEach(o, function (val, key) {
211 if (keyHandler.globalKeys[key]) {
212 oops.push(key);
213 }
214 });
215 if (oops.length) {
216 $log.warn('Ignoring reserved global key(s):', oops.join(','));
217 oops.forEach(function (key) {
218 delete o[key];
219 });
220 }
221 }
222 }
223
Simon Huntdc6362a2014-12-18 19:55:23 -0800224 angular.module('onosUtil')
Simon Hunt639dc662015-02-18 14:19:20 -0800225 .factory('KeyService',
Simon Hunt9f5e97d2015-12-11 10:32:09 -0800226 ['$log', '$timeout', 'FnService', 'ThemeService', 'NavService',
Simon Hunt639dc662015-02-18 14:19:20 -0800227
Simon Hunt9f5e97d2015-12-11 10:32:09 -0800228 function (_$log_, _$timeout_, _fs_, _ts_, _ns_) {
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800229 $log = _$log_;
Simon Hunt9f5e97d2015-12-11 10:32:09 -0800230 $timeout = _$timeout_;
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800231 fs = _fs_;
232 ts = _ts_;
Simon Hunt9d286562015-03-09 13:53:50 -0700233 ns = _ns_;
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800234
Simon Huntdc6362a2014-12-18 19:55:23 -0800235 return {
Simon Hunt639dc662015-02-18 14:19:20 -0800236 bindQhs: function (_qhs_) {
237 qhs = _qhs_;
238 },
Simon Huntdc6362a2014-12-18 19:55:23 -0800239 installOn: function (elem) {
240 elem.on('keydown', keyIn);
241 setupGlobalKeys();
242 },
Simon Huntdc6362a2014-12-18 19:55:23 -0800243 keyBindings: function (x) {
244 if (x === undefined) {
245 return getKeyBindings();
246 } else {
247 setKeyBindings(x);
248 }
249 },
Bri Prebilic Cole9dcaea52015-07-21 14:39:48 -0700250 unbindKeys: unbindKeys,
Simon Huntdc6362a2014-12-18 19:55:23 -0800251 gestureNotes: function (g) {
252 if (g === undefined) {
253 return keyHandler.viewGestures;
254 } else {
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800255 keyHandler.viewGestures = fs.isA(g) || [];
Simon Huntdc6362a2014-12-18 19:55:23 -0800256 }
Simon Hunt36fc15c2015-02-12 17:02:58 -0800257 },
258 enableKeys: function (b) {
259 enabled = b;
Simon Hunt71892222015-09-29 13:39:40 -0700260 },
Simon Huntd5579252015-10-06 15:09:14 -0700261 enableGlobalKeys: function (b) {
262 globalEnabled = b;
263 },
Simon Hunt71892222015-09-29 13:39:40 -0700264 checkNotGlobal: checkNotGlobal
Simon Huntdc6362a2014-12-18 19:55:23 -0800265 };
Simon Hunt1eecfa22014-12-16 14:46:29 -0800266 }]);
267
Simon Huntdc6362a2014-12-18 19:55:23 -0800268}());