##// END OF EJS Templates
Merge pull request #7068 from Carreau/close-pager...
Min RK -
r19123:4a95b5b0 merge
parent child Browse files
Show More
@@ -1,231 +1,232
1 // Copyright (c) IPython Development Team.
1 // Copyright (c) IPython Development Team.
2 // Distributed under the terms of the Modified BSD License.
2 // Distributed under the terms of the Modified BSD License.
3 /**
3 /**
4 *
4 *
5 *
5 *
6 * @module keyboardmanager
6 * @module keyboardmanager
7 * @namespace keyboardmanager
7 * @namespace keyboardmanager
8 * @class KeyboardManager
8 * @class KeyboardManager
9 */
9 */
10
10
11 define([
11 define([
12 'base/js/namespace',
12 'base/js/namespace',
13 'jquery',
13 'jquery',
14 'base/js/utils',
14 'base/js/utils',
15 'base/js/keyboard',
15 'base/js/keyboard',
16 ], function(IPython, $, utils, keyboard) {
16 ], function(IPython, $, utils, keyboard) {
17 "use strict";
17 "use strict";
18
18
19 // Main keyboard manager for the notebook
19 // Main keyboard manager for the notebook
20 var keycodes = keyboard.keycodes;
20 var keycodes = keyboard.keycodes;
21
21
22 var KeyboardManager = function (options) {
22 var KeyboardManager = function (options) {
23 /**
23 /**
24 * A class to deal with keyboard event and shortcut
24 * A class to deal with keyboard event and shortcut
25 *
25 *
26 * @class KeyboardManager
26 * @class KeyboardManager
27 * @constructor
27 * @constructor
28 * @param options {dict} Dictionary of keyword arguments :
28 * @param options {dict} Dictionary of keyword arguments :
29 * @param options.events {$(Events)} instance
29 * @param options.events {$(Events)} instance
30 * @param options.pager: {Pager} pager instance
30 * @param options.pager: {Pager} pager instance
31 */
31 */
32 this.mode = 'command';
32 this.mode = 'command';
33 this.enabled = true;
33 this.enabled = true;
34 this.pager = options.pager;
34 this.pager = options.pager;
35 this.quick_help = undefined;
35 this.quick_help = undefined;
36 this.notebook = undefined;
36 this.notebook = undefined;
37 this.last_mode = undefined;
37 this.last_mode = undefined;
38 this.bind_events();
38 this.bind_events();
39 this.env = {pager:this.pager};
39 this.env = {pager:this.pager};
40 this.actions = options.actions;
40 this.actions = options.actions;
41 this.command_shortcuts = new keyboard.ShortcutManager(undefined, options.events, this.actions, this.env );
41 this.command_shortcuts = new keyboard.ShortcutManager(undefined, options.events, this.actions, this.env );
42 this.command_shortcuts.add_shortcuts(this.get_default_common_shortcuts());
42 this.command_shortcuts.add_shortcuts(this.get_default_common_shortcuts());
43 this.command_shortcuts.add_shortcuts(this.get_default_command_shortcuts());
43 this.command_shortcuts.add_shortcuts(this.get_default_command_shortcuts());
44 this.edit_shortcuts = new keyboard.ShortcutManager(undefined, options.events, this.actions, this.env);
44 this.edit_shortcuts = new keyboard.ShortcutManager(undefined, options.events, this.actions, this.env);
45 this.edit_shortcuts.add_shortcuts(this.get_default_common_shortcuts());
45 this.edit_shortcuts.add_shortcuts(this.get_default_common_shortcuts());
46 this.edit_shortcuts.add_shortcuts(this.get_default_edit_shortcuts());
46 this.edit_shortcuts.add_shortcuts(this.get_default_edit_shortcuts());
47 Object.seal(this);
47 Object.seal(this);
48 };
48 };
49
49
50
50
51
51
52
52
53 /**
53 /**
54 * Return a dict of common shortcut
54 * Return a dict of common shortcut
55 * @method get_default_common_shortcuts
55 * @method get_default_common_shortcuts
56 *
56 *
57 * @example Example of returned shortcut
57 * @example Example of returned shortcut
58 * ```
58 * ```
59 * 'shortcut-key': 'action-name'
59 * 'shortcut-key': 'action-name'
60 * // a string representing the shortcut as dash separated value.
60 * // a string representing the shortcut as dash separated value.
61 * // e.g. 'shift' , 'shift-enter', 'cmd-t'
61 * // e.g. 'shift' , 'shift-enter', 'cmd-t'
62 *```
62 *```
63 */
63 */
64 KeyboardManager.prototype.get_default_common_shortcuts = function() {
64 KeyboardManager.prototype.get_default_common_shortcuts = function() {
65 return {
65 return {
66 'shift' : 'ipython.ignore',
66 'shift' : 'ipython.ignore',
67 'shift-enter' : 'ipython.run-select-next',
67 'shift-enter' : 'ipython.run-select-next',
68 'ctrl-enter' : 'ipython.execute-in-place',
68 'ctrl-enter' : 'ipython.execute-in-place',
69 'alt-enter' : 'ipython.execute-and-insert-after',
69 'alt-enter' : 'ipython.execute-and-insert-after',
70 // cmd on mac, ctrl otherwise
70 // cmd on mac, ctrl otherwise
71 'cmdtrl-s' : 'ipython.save-notebook',
71 'cmdtrl-s' : 'ipython.save-notebook',
72 };
72 };
73 };
73 };
74
74
75 KeyboardManager.prototype.get_default_edit_shortcuts = function() {
75 KeyboardManager.prototype.get_default_edit_shortcuts = function() {
76 return {
76 return {
77 'esc' : 'ipython.go-to-command-mode',
77 'esc' : 'ipython.go-to-command-mode',
78 'ctrl-m' : 'ipython.go-to-command-mode',
78 'ctrl-m' : 'ipython.go-to-command-mode',
79 'up' : 'ipython.move-cursor-up-or-previous-cell',
79 'up' : 'ipython.move-cursor-up-or-previous-cell',
80 'down' : 'ipython.move-cursor-down-or-next-cell',
80 'down' : 'ipython.move-cursor-down-or-next-cell',
81 'ctrl-shift--' : 'ipython.split-cell-at-cursor',
81 'ctrl-shift--' : 'ipython.split-cell-at-cursor',
82 'ctrl-shift-subtract' : 'ipython.split-cell-at-cursor'
82 'ctrl-shift-subtract' : 'ipython.split-cell-at-cursor'
83 };
83 };
84 };
84 };
85
85
86 KeyboardManager.prototype.get_default_command_shortcuts = function() {
86 KeyboardManager.prototype.get_default_command_shortcuts = function() {
87 return {
87 return {
88 'shift-space': 'ipython.scroll-up',
88 'shift-space': 'ipython.scroll-up',
89 'shift-v' : 'ipython.paste-cell-before',
89 'shift-v' : 'ipython.paste-cell-before',
90 'shift-m' : 'ipython.merge-selected-cell-with-cell-after',
90 'shift-m' : 'ipython.merge-selected-cell-with-cell-after',
91 'shift-o' : 'ipython.toggle-output-scrolling-selected-cell',
91 'shift-o' : 'ipython.toggle-output-scrolling-selected-cell',
92 'ctrl-j' : 'ipython.move-selected-cell-down',
92 'ctrl-j' : 'ipython.move-selected-cell-down',
93 'ctrl-k' : 'ipython.move-selected-cell-up',
93 'ctrl-k' : 'ipython.move-selected-cell-up',
94 'enter' : 'ipython.enter-edit-mode',
94 'enter' : 'ipython.enter-edit-mode',
95 'space' : 'ipython.scroll-down',
95 'space' : 'ipython.scroll-down',
96 'down' : 'ipython.select-next-cell',
96 'down' : 'ipython.select-next-cell',
97 'i,i' : 'ipython.interrupt-kernel',
97 'i,i' : 'ipython.interrupt-kernel',
98 '0,0' : 'ipython.restart-kernel',
98 '0,0' : 'ipython.restart-kernel',
99 'd,d' : 'ipython.delete-cell',
99 'd,d' : 'ipython.delete-cell',
100 'esc': 'ipython.close-pager',
100 'up' : 'ipython.select-previous-cell',
101 'up' : 'ipython.select-previous-cell',
101 'k' : 'ipython.select-previous-cell',
102 'k' : 'ipython.select-previous-cell',
102 'j' : 'ipython.select-next-cell',
103 'j' : 'ipython.select-next-cell',
103 'x' : 'ipython.cut-selected-cell',
104 'x' : 'ipython.cut-selected-cell',
104 'c' : 'ipython.copy-selected-cell',
105 'c' : 'ipython.copy-selected-cell',
105 'v' : 'ipython.paste-cell-after',
106 'v' : 'ipython.paste-cell-after',
106 'a' : 'ipython.insert-cell-before',
107 'a' : 'ipython.insert-cell-before',
107 'b' : 'ipython.insert-cell-after',
108 'b' : 'ipython.insert-cell-after',
108 'y' : 'ipython.change-selected-cell-to-code-cell',
109 'y' : 'ipython.change-selected-cell-to-code-cell',
109 'm' : 'ipython.change-selected-cell-to-markdown-cell',
110 'm' : 'ipython.change-selected-cell-to-markdown-cell',
110 'r' : 'ipython.change-selected-cell-to-raw-cell',
111 'r' : 'ipython.change-selected-cell-to-raw-cell',
111 '1' : 'ipython.change-selected-cell-to-heading-1',
112 '1' : 'ipython.change-selected-cell-to-heading-1',
112 '2' : 'ipython.change-selected-cell-to-heading-2',
113 '2' : 'ipython.change-selected-cell-to-heading-2',
113 '3' : 'ipython.change-selected-cell-to-heading-3',
114 '3' : 'ipython.change-selected-cell-to-heading-3',
114 '4' : 'ipython.change-selected-cell-to-heading-4',
115 '4' : 'ipython.change-selected-cell-to-heading-4',
115 '5' : 'ipython.change-selected-cell-to-heading-5',
116 '5' : 'ipython.change-selected-cell-to-heading-5',
116 '6' : 'ipython.change-selected-cell-to-heading-6',
117 '6' : 'ipython.change-selected-cell-to-heading-6',
117 'o' : 'ipython.toggle-output-visibility-selected-cell',
118 'o' : 'ipython.toggle-output-visibility-selected-cell',
118 's' : 'ipython.save-notebook',
119 's' : 'ipython.save-notebook',
119 'l' : 'ipython.toggle-line-number-selected-cell',
120 'l' : 'ipython.toggle-line-number-selected-cell',
120 'h' : 'ipython.show-keyboard-shortcut-help-dialog',
121 'h' : 'ipython.show-keyboard-shortcut-help-dialog',
121 'z' : 'ipython.undo-last-cell-deletion',
122 'z' : 'ipython.undo-last-cell-deletion',
122 'q' : 'ipython.close-pager',
123 'q' : 'ipython.close-pager',
123 };
124 };
124 };
125 };
125
126
126 KeyboardManager.prototype.bind_events = function () {
127 KeyboardManager.prototype.bind_events = function () {
127 var that = this;
128 var that = this;
128 $(document).keydown(function (event) {
129 $(document).keydown(function (event) {
129 if(event._ipkmIgnore===true||(event.originalEvent||{})._ipkmIgnore===true){
130 if(event._ipkmIgnore===true||(event.originalEvent||{})._ipkmIgnore===true){
130 return false;
131 return false;
131 }
132 }
132 return that.handle_keydown(event);
133 return that.handle_keydown(event);
133 });
134 });
134 };
135 };
135
136
136 KeyboardManager.prototype.set_notebook = function (notebook) {
137 KeyboardManager.prototype.set_notebook = function (notebook) {
137 this.notebook = notebook;
138 this.notebook = notebook;
138 this.actions.extend_env({notebook:notebook});
139 this.actions.extend_env({notebook:notebook});
139 };
140 };
140
141
141 KeyboardManager.prototype.set_quickhelp = function (notebook) {
142 KeyboardManager.prototype.set_quickhelp = function (notebook) {
142 this.actions.extend_env({quick_help:notebook});
143 this.actions.extend_env({quick_help:notebook});
143 };
144 };
144
145
145
146
146 KeyboardManager.prototype.handle_keydown = function (event) {
147 KeyboardManager.prototype.handle_keydown = function (event) {
147 /**
148 /**
148 * returning false from this will stop event propagation
149 * returning false from this will stop event propagation
149 **/
150 **/
150
151
151 if (event.which === keycodes.esc) {
152 if (event.which === keycodes.esc) {
152 // Intercept escape at highest level to avoid closing
153 // Intercept escape at highest level to avoid closing
153 // websocket connection with firefox
154 // websocket connection with firefox
154 event.preventDefault();
155 event.preventDefault();
155 }
156 }
156
157
157 if (!this.enabled) {
158 if (!this.enabled) {
158 if (event.which === keycodes.esc) {
159 if (event.which === keycodes.esc) {
159 this.notebook.command_mode();
160 this.notebook.command_mode();
160 return false;
161 return false;
161 }
162 }
162 return true;
163 return true;
163 }
164 }
164
165
165 if (this.mode === 'edit') {
166 if (this.mode === 'edit') {
166 return this.edit_shortcuts.call_handler(event);
167 return this.edit_shortcuts.call_handler(event);
167 } else if (this.mode === 'command') {
168 } else if (this.mode === 'command') {
168 return this.command_shortcuts.call_handler(event);
169 return this.command_shortcuts.call_handler(event);
169 }
170 }
170 return true;
171 return true;
171 };
172 };
172
173
173 KeyboardManager.prototype.edit_mode = function () {
174 KeyboardManager.prototype.edit_mode = function () {
174 this.last_mode = this.mode;
175 this.last_mode = this.mode;
175 this.mode = 'edit';
176 this.mode = 'edit';
176 };
177 };
177
178
178 KeyboardManager.prototype.command_mode = function () {
179 KeyboardManager.prototype.command_mode = function () {
179 this.last_mode = this.mode;
180 this.last_mode = this.mode;
180 this.mode = 'command';
181 this.mode = 'command';
181 };
182 };
182
183
183 KeyboardManager.prototype.enable = function () {
184 KeyboardManager.prototype.enable = function () {
184 this.enabled = true;
185 this.enabled = true;
185 };
186 };
186
187
187 KeyboardManager.prototype.disable = function () {
188 KeyboardManager.prototype.disable = function () {
188 this.enabled = false;
189 this.enabled = false;
189 };
190 };
190
191
191 KeyboardManager.prototype.register_events = function (e) {
192 KeyboardManager.prototype.register_events = function (e) {
192 var that = this;
193 var that = this;
193 var handle_focus = function () {
194 var handle_focus = function () {
194 that.disable();
195 that.disable();
195 };
196 };
196 var handle_blur = function () {
197 var handle_blur = function () {
197 that.enable();
198 that.enable();
198 };
199 };
199 e.on('focusin', handle_focus);
200 e.on('focusin', handle_focus);
200 e.on('focusout', handle_blur);
201 e.on('focusout', handle_blur);
201 // TODO: Very strange. The focusout event does not seem fire for the
202 // TODO: Very strange. The focusout event does not seem fire for the
202 // bootstrap textboxes on FF25&26... This works around that by
203 // bootstrap textboxes on FF25&26... This works around that by
203 // registering focus and blur events recursively on all inputs within
204 // registering focus and blur events recursively on all inputs within
204 // registered element.
205 // registered element.
205 e.find('input').blur(handle_blur);
206 e.find('input').blur(handle_blur);
206 e.on('DOMNodeInserted', function (event) {
207 e.on('DOMNodeInserted', function (event) {
207 var target = $(event.target);
208 var target = $(event.target);
208 if (target.is('input')) {
209 if (target.is('input')) {
209 target.blur(handle_blur);
210 target.blur(handle_blur);
210 } else {
211 } else {
211 target.find('input').blur(handle_blur);
212 target.find('input').blur(handle_blur);
212 }
213 }
213 });
214 });
214 // There are times (raw_input) where we remove the element from the DOM before
215 // There are times (raw_input) where we remove the element from the DOM before
215 // focusout is called. In this case we bind to the remove event of jQueryUI,
216 // focusout is called. In this case we bind to the remove event of jQueryUI,
216 // which gets triggered upon removal, iff it is focused at the time.
217 // which gets triggered upon removal, iff it is focused at the time.
217 // is_focused must be used to check for the case where an element within
218 // is_focused must be used to check for the case where an element within
218 // the element being removed is focused.
219 // the element being removed is focused.
219 e.on('remove', function () {
220 e.on('remove', function () {
220 if (utils.is_focused(e[0])) {
221 if (utils.is_focused(e[0])) {
221 that.enable();
222 that.enable();
222 }
223 }
223 });
224 });
224 };
225 };
225
226
226
227
227 // For backwards compatibility.
228 // For backwards compatibility.
228 IPython.KeyboardManager = KeyboardManager;
229 IPython.KeyboardManager = KeyboardManager;
229
230
230 return {'KeyboardManager': KeyboardManager};
231 return {'KeyboardManager': KeyboardManager};
231 });
232 });
General Comments 0
You need to be logged in to leave comments. Login now