##// END OF EJS Templates
Added automatic close of Brackets.
damianavila -
Show More
@@ -1,570 +1,571 b''
1 1 //----------------------------------------------------------------------------
2 2 // Copyright (C) 2008-2011 The IPython Development Team
3 3 //
4 4 // Distributed under the terms of the BSD License. The full license is in
5 5 // the file COPYING, distributed as part of this software.
6 6 //----------------------------------------------------------------------------
7 7
8 8 //============================================================================
9 9 // CodeCell
10 10 //============================================================================
11 11 /**
12 12 * An extendable module that provide base functionnality to create cell for notebook.
13 13 * @module IPython
14 14 * @namespace IPython
15 15 * @submodule CodeCell
16 16 */
17 17
18 18
19 19 /* local util for codemirror */
20 20 var posEq = function(a, b) {return a.line == b.line && a.ch == b.ch;};
21 21
22 22 /**
23 23 *
24 24 * function to delete until previous non blanking space character
25 25 * or first multiple of 4 tabstop.
26 26 * @private
27 27 */
28 28 CodeMirror.commands.delSpaceToPrevTabStop = function(cm){
29 29 var from = cm.getCursor(true), to = cm.getCursor(false), sel = !posEq(from, to);
30 30 if (!posEq(from, to)) { cm.replaceRange("", from, to); return; }
31 31 var cur = cm.getCursor(), line = cm.getLine(cur.line);
32 32 var tabsize = cm.getOption('tabSize');
33 33 var chToPrevTabStop = cur.ch-(Math.ceil(cur.ch/tabsize)-1)*tabsize;
34 34 from = {ch:cur.ch-chToPrevTabStop,line:cur.line};
35 35 var select = cm.getRange(from,cur);
36 36 if( select.match(/^\ +$/) !== null){
37 37 cm.replaceRange("",from,cur);
38 38 } else {
39 39 cm.deleteH(-1,"char");
40 40 }
41 41 };
42 42
43 43
44 44 var IPython = (function (IPython) {
45 45 "use strict";
46 46
47 47 var utils = IPython.utils;
48 48 var key = IPython.utils.keycodes;
49 49
50 50 /**
51 51 * A Cell conceived to write code.
52 52 *
53 53 * The kernel doesn't have to be set at creation time, in that case
54 54 * it will be null and set_kernel has to be called later.
55 55 * @class CodeCell
56 56 * @extends IPython.Cell
57 57 *
58 58 * @constructor
59 59 * @param {Object|null} kernel
60 60 * @param {object|undefined} [options]
61 61 * @param [options.cm_config] {object} config to pass to CodeMirror
62 62 */
63 63 var CodeCell = function (kernel, options) {
64 64 this.kernel = kernel || null;
65 65 this.collapsed = false;
66 66
67 67 // create all attributed in constructor function
68 68 // even if null for V8 VM optimisation
69 69 this.input_prompt_number = null;
70 70 this.celltoolbar = null;
71 71 this.output_area = null;
72 72 this.last_msg_id = null;
73 73 this.completer = null;
74 74
75 75
76 76 var cm_overwrite_options = {
77 77 onKeyEvent: $.proxy(this.handle_keyevent,this)
78 78 };
79 79
80 80 options = this.mergeopt(CodeCell, options, {cm_config:cm_overwrite_options});
81 81
82 82 IPython.Cell.apply(this,[options]);
83 83
84 84 // Attributes we want to override in this subclass.
85 85 this.cell_type = "code";
86 86
87 87 var that = this;
88 88 this.element.focusout(
89 89 function() { that.auto_highlight(); }
90 90 );
91 91 };
92 92
93 93 CodeCell.options_default = {
94 94 cm_config : {
95 95 extraKeys: {
96 96 "Tab" : "indentMore",
97 97 "Shift-Tab" : "indentLess",
98 98 "Backspace" : "delSpaceToPrevTabStop",
99 99 "Cmd-/" : "toggleComment",
100 100 "Ctrl-/" : "toggleComment"
101 101 },
102 102 mode: 'ipython',
103 103 theme: 'ipython',
104 matchBrackets: true
104 matchBrackets: true,
105 autoCloseBrackets: true
105 106 }
106 107 };
107 108
108 109 CodeCell.msg_cells = {};
109 110
110 111 CodeCell.prototype = new IPython.Cell();
111 112
112 113 /**
113 114 * @method auto_highlight
114 115 */
115 116 CodeCell.prototype.auto_highlight = function () {
116 117 this._auto_highlight(IPython.config.cell_magic_highlight);
117 118 };
118 119
119 120 /** @method create_element */
120 121 CodeCell.prototype.create_element = function () {
121 122 IPython.Cell.prototype.create_element.apply(this, arguments);
122 123
123 124 var cell = $('<div></div>').addClass('cell border-box-sizing code_cell');
124 125 cell.attr('tabindex','2');
125 126
126 127 var input = $('<div></div>').addClass('input');
127 128 var prompt = $('<div/>').addClass('prompt input_prompt');
128 129 var inner_cell = $('<div/>').addClass('inner_cell');
129 130 this.celltoolbar = new IPython.CellToolbar(this);
130 131 inner_cell.append(this.celltoolbar.element);
131 132 var input_area = $('<div/>').addClass('input_area');
132 133 this.code_mirror = CodeMirror(input_area.get(0), this.cm_config);
133 134 $(this.code_mirror.getInputField()).attr("spellcheck", "false");
134 135 inner_cell.append(input_area);
135 136 input.append(prompt).append(inner_cell);
136 137
137 138 var widget_area = $('<div/>')
138 139 .addClass('widget-area')
139 140 .hide();
140 141 this.widget_area = widget_area;
141 142 var widget_prompt = $('<div/>')
142 143 .addClass('prompt')
143 144 .appendTo(widget_area);
144 145 var widget_subarea = $('<div/>')
145 146 .addClass('widget-subarea')
146 147 .appendTo(widget_area);
147 148 this.widget_subarea = widget_subarea;
148 149 var widget_clear_buton = $('<button />')
149 150 .addClass('close')
150 151 .html('&times;')
151 152 .click(function() {
152 153 widget_area.slideUp('', function(){ widget_subarea.html(''); });
153 154 })
154 155 .appendTo(widget_prompt);
155 156
156 157 var output = $('<div></div>');
157 158 cell.append(input).append(widget_area).append(output);
158 159 this.element = cell;
159 160 this.output_area = new IPython.OutputArea(output, true);
160 161 this.completer = new IPython.Completer(this);
161 162 };
162 163
163 164 /** @method bind_events */
164 165 CodeCell.prototype.bind_events = function () {
165 166 IPython.Cell.prototype.bind_events.apply(this);
166 167 var that = this;
167 168
168 169 this.element.focusout(
169 170 function() { that.auto_highlight(); }
170 171 );
171 172 };
172 173
173 174 CodeCell.prototype.handle_keyevent = function (editor, event) {
174 175
175 176 // console.log('CM', this.mode, event.which, event.type)
176 177
177 178 if (this.mode === 'command') {
178 179 return true;
179 180 } else if (this.mode === 'edit') {
180 181 return this.handle_codemirror_keyevent(editor, event);
181 182 }
182 183 };
183 184
184 185 /**
185 186 * This method gets called in CodeMirror's onKeyDown/onKeyPress
186 187 * handlers and is used to provide custom key handling. Its return
187 188 * value is used to determine if CodeMirror should ignore the event:
188 189 * true = ignore, false = don't ignore.
189 190 * @method handle_codemirror_keyevent
190 191 */
191 192 CodeCell.prototype.handle_codemirror_keyevent = function (editor, event) {
192 193
193 194 var that = this;
194 195 // whatever key is pressed, first, cancel the tooltip request before
195 196 // they are sent, and remove tooltip if any, except for tab again
196 197 var tooltip_closed = null;
197 198 if (event.type === 'keydown' && event.which != key.TAB ) {
198 199 tooltip_closed = IPython.tooltip.remove_and_cancel_tooltip();
199 200 }
200 201
201 202 var cur = editor.getCursor();
202 203 if (event.keyCode === key.ENTER){
203 204 this.auto_highlight();
204 205 }
205 206
206 207 if (event.keyCode === key.ENTER && (event.shiftKey || event.ctrlKey || event.altKey)) {
207 208 // Always ignore shift-enter in CodeMirror as we handle it.
208 209 return true;
209 210 } else if (event.which === 40 && event.type === 'keypress' && IPython.tooltip.time_before_tooltip >= 0) {
210 211 // triger on keypress (!) otherwise inconsistent event.which depending on plateform
211 212 // browser and keyboard layout !
212 213 // Pressing '(' , request tooltip, don't forget to reappend it
213 214 // The second argument says to hide the tooltip if the docstring
214 215 // is actually empty
215 216 IPython.tooltip.pending(that, true);
216 217 } else if (event.which === key.UPARROW && event.type === 'keydown') {
217 218 // If we are not at the top, let CM handle the up arrow and
218 219 // prevent the global keydown handler from handling it.
219 220 if (!that.at_top()) {
220 221 event.stop();
221 222 return false;
222 223 } else {
223 224 return true;
224 225 }
225 226 } else if (event.which === key.ESC && event.type === 'keydown') {
226 227 // First see if the tooltip is active and if so cancel it.
227 228 if (tooltip_closed) {
228 229 // The call to remove_and_cancel_tooltip above in L177 doesn't pass
229 230 // force=true. Because of this it won't actually close the tooltip
230 231 // if it is in sticky mode. Thus, we have to check again if it is open
231 232 // and close it with force=true.
232 233 if (!IPython.tooltip._hidden) {
233 234 IPython.tooltip.remove_and_cancel_tooltip(true);
234 235 }
235 236 // If we closed the tooltip, don't let CM or the global handlers
236 237 // handle this event.
237 238 event.stop();
238 239 return true;
239 240 }
240 241 if (that.code_mirror.options.keyMap === "vim-insert") {
241 242 // vim keyMap is active and in insert mode. In this case we leave vim
242 243 // insert mode, but remain in notebook edit mode.
243 244 // Let' CM handle this event and prevent global handling.
244 245 event.stop();
245 246 return false;
246 247 } else {
247 248 // vim keyMap is not active. Leave notebook edit mode.
248 249 // Don't let CM handle the event, defer to global handling.
249 250 return true;
250 251 }
251 252 } else if (event.which === key.DOWNARROW && event.type === 'keydown') {
252 253 // If we are not at the bottom, let CM handle the down arrow and
253 254 // prevent the global keydown handler from handling it.
254 255 if (!that.at_bottom()) {
255 256 event.stop();
256 257 return false;
257 258 } else {
258 259 return true;
259 260 }
260 261 } else if (event.keyCode === key.TAB && event.type === 'keydown' && event.shiftKey) {
261 262 if (editor.somethingSelected()){
262 263 var anchor = editor.getCursor("anchor");
263 264 var head = editor.getCursor("head");
264 265 if( anchor.line != head.line){
265 266 return false;
266 267 }
267 268 }
268 269 IPython.tooltip.request(that);
269 270 event.stop();
270 271 return true;
271 272 } else if (event.keyCode === key.TAB && event.type == 'keydown') {
272 273 // Tab completion.
273 274 IPython.tooltip.remove_and_cancel_tooltip();
274 275 if (editor.somethingSelected()) {
275 276 return false;
276 277 }
277 278 var pre_cursor = editor.getRange({line:cur.line,ch:0},cur);
278 279 if (pre_cursor.trim() === "") {
279 280 // Don't autocomplete if the part of the line before the cursor
280 281 // is empty. In this case, let CodeMirror handle indentation.
281 282 return false;
282 283 } else {
283 284 event.stop();
284 285 this.completer.startCompletion();
285 286 return true;
286 287 }
287 288 } else {
288 289 // keypress/keyup also trigger on TAB press, and we don't want to
289 290 // use those to disable tab completion.
290 291 return false;
291 292 }
292 293 return false;
293 294 };
294 295
295 296 // Kernel related calls.
296 297
297 298 CodeCell.prototype.set_kernel = function (kernel) {
298 299 this.kernel = kernel;
299 300 };
300 301
301 302 /**
302 303 * Execute current code cell to the kernel
303 304 * @method execute
304 305 */
305 306 CodeCell.prototype.execute = function () {
306 307 this.output_area.clear_output();
307 308
308 309 // Clear widget area
309 310 this.widget_subarea.html('');
310 311 this.widget_subarea.height('');
311 312 this.widget_area.height('');
312 313 this.widget_area.hide();
313 314
314 315 this.set_input_prompt('*');
315 316 this.element.addClass("running");
316 317 if (this.last_msg_id) {
317 318 this.kernel.clear_callbacks_for_msg(this.last_msg_id);
318 319 }
319 320 var callbacks = this.get_callbacks();
320 321
321 322 var old_msg_id = this.last_msg_id;
322 323 this.last_msg_id = this.kernel.execute(this.get_text(), callbacks, {silent: false, store_history: true});
323 324 if (old_msg_id) {
324 325 delete CodeCell.msg_cells[old_msg_id];
325 326 }
326 327 CodeCell.msg_cells[this.last_msg_id] = this;
327 328 };
328 329
329 330 /**
330 331 * Construct the default callbacks for
331 332 * @method get_callbacks
332 333 */
333 334 CodeCell.prototype.get_callbacks = function () {
334 335 return {
335 336 shell : {
336 337 reply : $.proxy(this._handle_execute_reply, this),
337 338 payload : {
338 339 set_next_input : $.proxy(this._handle_set_next_input, this),
339 340 page : $.proxy(this._open_with_pager, this)
340 341 }
341 342 },
342 343 iopub : {
343 344 output : $.proxy(this.output_area.handle_output, this.output_area),
344 345 clear_output : $.proxy(this.output_area.handle_clear_output, this.output_area),
345 346 },
346 347 input : $.proxy(this._handle_input_request, this)
347 348 };
348 349 };
349 350
350 351 CodeCell.prototype._open_with_pager = function (payload) {
351 352 $([IPython.events]).trigger('open_with_text.Pager', payload);
352 353 };
353 354
354 355 /**
355 356 * @method _handle_execute_reply
356 357 * @private
357 358 */
358 359 CodeCell.prototype._handle_execute_reply = function (msg) {
359 360 this.set_input_prompt(msg.content.execution_count);
360 361 this.element.removeClass("running");
361 362 $([IPython.events]).trigger('set_dirty.Notebook', {value: true});
362 363 };
363 364
364 365 /**
365 366 * @method _handle_set_next_input
366 367 * @private
367 368 */
368 369 CodeCell.prototype._handle_set_next_input = function (payload) {
369 370 var data = {'cell': this, 'text': payload.text};
370 371 $([IPython.events]).trigger('set_next_input.Notebook', data);
371 372 };
372 373
373 374 /**
374 375 * @method _handle_input_request
375 376 * @private
376 377 */
377 378 CodeCell.prototype._handle_input_request = function (msg) {
378 379 this.output_area.append_raw_input(msg);
379 380 };
380 381
381 382
382 383 // Basic cell manipulation.
383 384
384 385 CodeCell.prototype.select = function () {
385 386 var cont = IPython.Cell.prototype.select.apply(this);
386 387 if (cont) {
387 388 this.code_mirror.refresh();
388 389 this.auto_highlight();
389 390 }
390 391 return cont;
391 392 };
392 393
393 394 CodeCell.prototype.render = function () {
394 395 var cont = IPython.Cell.prototype.render.apply(this);
395 396 // Always execute, even if we are already in the rendered state
396 397 return cont;
397 398 };
398 399
399 400 CodeCell.prototype.unrender = function () {
400 401 // CodeCell is always rendered
401 402 return false;
402 403 };
403 404
404 405 CodeCell.prototype.edit_mode = function () {
405 406 var cont = IPython.Cell.prototype.edit_mode.apply(this);
406 407 if (cont) {
407 408 this.focus_editor();
408 409 }
409 410 return cont;
410 411 }
411 412
412 413 CodeCell.prototype.select_all = function () {
413 414 var start = {line: 0, ch: 0};
414 415 var nlines = this.code_mirror.lineCount();
415 416 var last_line = this.code_mirror.getLine(nlines-1);
416 417 var end = {line: nlines-1, ch: last_line.length};
417 418 this.code_mirror.setSelection(start, end);
418 419 };
419 420
420 421
421 422 CodeCell.prototype.collapse_output = function () {
422 423 this.collapsed = true;
423 424 this.output_area.collapse();
424 425 };
425 426
426 427
427 428 CodeCell.prototype.expand_output = function () {
428 429 this.collapsed = false;
429 430 this.output_area.expand();
430 431 this.output_area.unscroll_area();
431 432 };
432 433
433 434 CodeCell.prototype.scroll_output = function () {
434 435 this.output_area.expand();
435 436 this.output_area.scroll_if_long();
436 437 };
437 438
438 439 CodeCell.prototype.toggle_output = function () {
439 440 this.collapsed = Boolean(1 - this.collapsed);
440 441 this.output_area.toggle_output();
441 442 };
442 443
443 444 CodeCell.prototype.toggle_output_scroll = function () {
444 445 this.output_area.toggle_scroll();
445 446 };
446 447
447 448
448 449 CodeCell.input_prompt_classical = function (prompt_value, lines_number) {
449 450 var ns;
450 451 if (prompt_value == undefined) {
451 452 ns = "&nbsp;";
452 453 } else {
453 454 ns = encodeURIComponent(prompt_value);
454 455 }
455 456 return 'In&nbsp;[' + ns + ']:';
456 457 };
457 458
458 459 CodeCell.input_prompt_continuation = function (prompt_value, lines_number) {
459 460 var html = [CodeCell.input_prompt_classical(prompt_value, lines_number)];
460 461 for(var i=1; i < lines_number; i++) {
461 462 html.push(['...:']);
462 463 }
463 464 return html.join('<br/>');
464 465 };
465 466
466 467 CodeCell.input_prompt_function = CodeCell.input_prompt_classical;
467 468
468 469
469 470 CodeCell.prototype.set_input_prompt = function (number) {
470 471 var nline = 1;
471 472 if (this.code_mirror !== undefined) {
472 473 nline = this.code_mirror.lineCount();
473 474 }
474 475 this.input_prompt_number = number;
475 476 var prompt_html = CodeCell.input_prompt_function(this.input_prompt_number, nline);
476 477 this.element.find('div.input_prompt').html(prompt_html);
477 478 };
478 479
479 480
480 481 CodeCell.prototype.clear_input = function () {
481 482 this.code_mirror.setValue('');
482 483 };
483 484
484 485
485 486 CodeCell.prototype.get_text = function () {
486 487 return this.code_mirror.getValue();
487 488 };
488 489
489 490
490 491 CodeCell.prototype.set_text = function (code) {
491 492 return this.code_mirror.setValue(code);
492 493 };
493 494
494 495
495 496 CodeCell.prototype.at_top = function () {
496 497 var cursor = this.code_mirror.getCursor();
497 498 if (cursor.line === 0 && cursor.ch === 0) {
498 499 return true;
499 500 } else {
500 501 return false;
501 502 }
502 503 };
503 504
504 505
505 506 CodeCell.prototype.at_bottom = function () {
506 507 var cursor = this.code_mirror.getCursor();
507 508 if (cursor.line === (this.code_mirror.lineCount()-1) && cursor.ch === this.code_mirror.getLine(cursor.line).length) {
508 509 return true;
509 510 } else {
510 511 return false;
511 512 }
512 513 };
513 514
514 515
515 516 CodeCell.prototype.clear_output = function (wait) {
516 517 this.output_area.clear_output(wait);
517 518 this.set_input_prompt();
518 519 };
519 520
520 521
521 522 // JSON serialization
522 523
523 524 CodeCell.prototype.fromJSON = function (data) {
524 525 IPython.Cell.prototype.fromJSON.apply(this, arguments);
525 526 if (data.cell_type === 'code') {
526 527 if (data.input !== undefined) {
527 528 this.set_text(data.input);
528 529 // make this value the starting point, so that we can only undo
529 530 // to this state, instead of a blank cell
530 531 this.code_mirror.clearHistory();
531 532 this.auto_highlight();
532 533 }
533 534 if (data.prompt_number !== undefined) {
534 535 this.set_input_prompt(data.prompt_number);
535 536 } else {
536 537 this.set_input_prompt();
537 538 }
538 539 this.output_area.trusted = data.trusted || false;
539 540 this.output_area.fromJSON(data.outputs);
540 541 if (data.collapsed !== undefined) {
541 542 if (data.collapsed) {
542 543 this.collapse_output();
543 544 } else {
544 545 this.expand_output();
545 546 }
546 547 }
547 548 }
548 549 };
549 550
550 551
551 552 CodeCell.prototype.toJSON = function () {
552 553 var data = IPython.Cell.prototype.toJSON.apply(this);
553 554 data.input = this.get_text();
554 555 // is finite protect against undefined and '*' value
555 556 if (isFinite(this.input_prompt_number)) {
556 557 data.prompt_number = this.input_prompt_number;
557 558 }
558 559 var outputs = this.output_area.toJSON();
559 560 data.outputs = outputs;
560 561 data.language = 'python';
561 562 data.trusted = this.output_area.trusted;
562 563 data.collapsed = this.collapsed;
563 564 return data;
564 565 };
565 566
566 567
567 568 IPython.CodeCell = CodeCell;
568 569
569 570 return IPython;
570 571 }(IPython));
General Comments 0
You need to be logged in to leave comments. Login now