blob: d22a33569770f0fbdd6ac9253484b50542d7b52c [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 Hunt9d286562015-03-09 13:53:50 -070024 var $log, fs, ts, ns, qhs;
Simon Hunt1eecfa22014-12-16 14:46:29 -080025
26 // internal state
Simon Hunt36fc15c2015-02-12 17:02:58 -080027 var enabled = true,
28 keyHandler = {
Simon Hunt1eecfa22014-12-16 14:46:29 -080029 globalKeys: {},
30 maskedKeys: {},
31 viewKeys: {},
32 viewFn: null,
33 viewGestures: []
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -080034 };
Simon Hunt1eecfa22014-12-16 14:46:29 -080035
Simon Hunt9d286562015-03-09 13:53:50 -070036 // TODO: do we still need to have the concept of view token here..?
Simon Hunt1eecfa22014-12-16 14:46:29 -080037 function getViewToken() {
38 return 'NotYetAViewToken';
39 }
40
41 function whatKey(code) {
42 switch (code) {
43 case 13: return 'enter';
44 case 16: return 'shift';
45 case 17: return 'ctrl';
46 case 18: return 'alt';
47 case 27: return 'esc';
48 case 32: return 'space';
49 case 37: return 'leftArrow';
50 case 38: return 'upArrow';
51 case 39: return 'rightArrow';
52 case 40: return 'downArrow';
53 case 91: return 'cmdLeft';
54 case 93: return 'cmdRight';
55 case 187: return 'equals';
56 case 189: return 'dash';
57 case 191: return 'slash';
58 case 192: return 'backQuote';
59 case 220: return 'backSlash';
60 default:
61 if ((code >= 48 && code <= 57) ||
62 (code >= 65 && code <= 90)) {
63 return String.fromCharCode(code);
64 } else if (code >= 112 && code <= 123) {
65 return 'F' + (code - 111);
66 }
67 return '.';
68 }
69 }
70
71 function keyIn() {
72 var event = d3.event,
73 keyCode = event.keyCode,
74 key = whatKey(keyCode),
75 kh = keyHandler,
76 gk = kh.globalKeys[key],
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -080077 gcb = fs.isF(gk) || (fs.isA(gk) && fs.isF(gk[0])),
Simon Hunt1eecfa22014-12-16 14:46:29 -080078 vk = kh.viewKeys[key],
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -080079 vcb = fs.isF(vk) || (fs.isA(vk) && fs.isF(vk[0])) || fs.isF(kh.viewFn),
Simon Hunt1eecfa22014-12-16 14:46:29 -080080 token = getViewToken();
81
Simon Huntdc6362a2014-12-18 19:55:23 -080082 d3.event.stopPropagation();
83
Simon Hunt36fc15c2015-02-12 17:02:58 -080084 if (enabled) {
85 // global callback?
86 if (gcb && gcb(token, key, keyCode, event)) {
87 // if the event was 'handled', we are done
88 return;
89 }
90 // otherwise, let the view callback have a shot
91 if (vcb) {
92 vcb(token, key, keyCode, event);
93 }
Simon Hunt1eecfa22014-12-16 14:46:29 -080094 }
95 }
96
97 function setupGlobalKeys() {
Simon Hunt404f6b22015-01-21 14:00:56 -080098 angular.extend(keyHandler, {
Simon Hunt1eecfa22014-12-16 14:46:29 -080099 globalKeys: {
100 backSlash: [quickHelp, 'Show / hide Quick Help'],
101 slash: [quickHelp, 'Show / hide Quick Help'],
102 esc: [escapeKey, 'Dismiss dialog or cancel selections'],
103 T: [toggleTheme, "Toggle theme"]
104 },
105 globalFormat: ['backSlash', 'slash', 'esc', 'T'],
106
107 // Masked keys are global key handlers that always return true.
108 // That is, the view will never see the event for that key.
109 maskedKeys: {
110 slash: true,
111 backSlash: true,
112 T: true
113 }
114 });
115 }
116
117 function quickHelp(view, key, code, ev) {
Simon Hunt639dc662015-02-18 14:19:20 -0800118 qhs.showQuickHelp(keyHandler);
Simon Hunt1eecfa22014-12-16 14:46:29 -0800119 return true;
120 }
121
Simon Hunta0eb0a82015-02-11 12:30:06 -0800122 // returns true if we 'consumed' the ESC keypress, false otherwise
Simon Hunt1eecfa22014-12-16 14:46:29 -0800123 function escapeKey(view, key, code, ev) {
Simon Hunt9d286562015-03-09 13:53:50 -0700124 return ns.hideIfShown() || qhs.hideQuickHelp();
Simon Hunt1eecfa22014-12-16 14:46:29 -0800125 }
126
127 function toggleTheme(view, key, code, ev) {
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800128 ts.toggleTheme();
Simon Hunt1eecfa22014-12-16 14:46:29 -0800129 return true;
130 }
131
Simon Hunt59df0b22014-12-17 10:32:25 -0800132 function setKeyBindings(keyArg) {
133 var viewKeys,
134 masked = [];
135
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800136 if (fs.isF(keyArg)) {
Simon Hunt59df0b22014-12-17 10:32:25 -0800137 // set general key handler callback
138 keyHandler.viewFn = keyArg;
139 } else {
140 // set specific key filter map
141 viewKeys = d3.map(keyArg).keys();
142 viewKeys.forEach(function (key) {
143 if (keyHandler.maskedKeys[key]) {
Simon Huntaf322072014-12-18 13:23:40 -0800144 masked.push('setKeyBindings(): Key "' + key + '" is reserved');
Simon Hunt59df0b22014-12-17 10:32:25 -0800145 }
146 });
147
148 if (masked.length) {
Simon Huntaf322072014-12-18 13:23:40 -0800149 $log.warn(masked.join('\n'));
Simon Hunt59df0b22014-12-17 10:32:25 -0800150 }
151 keyHandler.viewKeys = keyArg;
152 }
153 }
154
155 function getKeyBindings() {
156 var gkeys = d3.map(keyHandler.globalKeys).keys(),
157 masked = d3.map(keyHandler.maskedKeys).keys(),
158 vkeys = d3.map(keyHandler.viewKeys).keys(),
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800159 vfn = !!fs.isF(keyHandler.viewFn);
Simon Hunt59df0b22014-12-17 10:32:25 -0800160
161 return {
162 globalKeys: gkeys,
163 maskedKeys: masked,
164 viewKeys: vkeys,
165 viewFunction: vfn
166 };
167 }
168
Simon Huntdc6362a2014-12-18 19:55:23 -0800169 angular.module('onosUtil')
Simon Hunt639dc662015-02-18 14:19:20 -0800170 .factory('KeyService',
Simon Hunt9d286562015-03-09 13:53:50 -0700171 ['$log', 'FnService', 'ThemeService', 'NavService',
Simon Hunt639dc662015-02-18 14:19:20 -0800172
Simon Hunt9d286562015-03-09 13:53:50 -0700173 function (_$log_, _fs_, _ts_, _ns_) {
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800174 $log = _$log_;
175 fs = _fs_;
176 ts = _ts_;
Simon Hunt9d286562015-03-09 13:53:50 -0700177 ns = _ns_;
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800178
Simon Huntdc6362a2014-12-18 19:55:23 -0800179 return {
Simon Hunt639dc662015-02-18 14:19:20 -0800180 bindQhs: function (_qhs_) {
181 qhs = _qhs_;
182 },
Simon Huntdc6362a2014-12-18 19:55:23 -0800183 installOn: function (elem) {
184 elem.on('keydown', keyIn);
185 setupGlobalKeys();
186 },
Simon Huntdc6362a2014-12-18 19:55:23 -0800187 keyBindings: function (x) {
188 if (x === undefined) {
189 return getKeyBindings();
190 } else {
191 setKeyBindings(x);
192 }
193 },
194 gestureNotes: function (g) {
195 if (g === undefined) {
196 return keyHandler.viewGestures;
197 } else {
Yuta HIGUCHI4f39bcd2014-12-18 20:46:14 -0800198 keyHandler.viewGestures = fs.isA(g) || [];
Simon Huntdc6362a2014-12-18 19:55:23 -0800199 }
Simon Hunt36fc15c2015-02-12 17:02:58 -0800200 },
201 enableKeys: function (b) {
202 enabled = b;
Simon Hunt59df0b22014-12-17 10:32:25 -0800203 }
Simon Huntdc6362a2014-12-18 19:55:23 -0800204 };
Simon Hunt1eecfa22014-12-16 14:46:29 -0800205 }]);
206
Simon Huntdc6362a2014-12-18 19:55:23 -0800207}());