Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Avatar for KuCalc : devops.
Download
50654 views
1
// Copyright (c) IPython Development Team.
2
// Distributed under the terms of the Modified BSD License.
3
4
/**
5
* @module notebook
6
*/
7
define(function (require) {
8
"use strict";
9
var IPython = require('base/js/namespace');
10
var $ = require('jquery');
11
var utils = require('base/js/utils');
12
var dialog = require('base/js/dialog');
13
var cellmod = require('notebook/js/cell');
14
var textcell = require('notebook/js/textcell');
15
var codecell = require('notebook/js/codecell');
16
var moment = require('moment');
17
var configmod = require('services/config');
18
var session = require('services/sessions/session');
19
var celltoolbar = require('notebook/js/celltoolbar');
20
var marked = require('components/marked/lib/marked');
21
var CodeMirror = require('codemirror/lib/codemirror');
22
var runMode = require('codemirror/addon/runmode/runmode');
23
var mathjaxutils = require('notebook/js/mathjaxutils');
24
var keyboard = require('base/js/keyboard');
25
var tooltip = require('notebook/js/tooltip');
26
var default_celltoolbar = require('notebook/js/celltoolbarpresets/default');
27
var rawcell_celltoolbar = require('notebook/js/celltoolbarpresets/rawcell');
28
var slideshow_celltoolbar = require('notebook/js/celltoolbarpresets/slideshow');
29
var scrollmanager = require('notebook/js/scrollmanager');
30
31
/**
32
* Contains and manages cells.
33
*
34
* @class Notebook
35
* @param {string} selector
36
* @param {object} options - Dictionary of keyword arguments.
37
* @param {jQuery} options.events - selector of Events
38
* @param {KeyboardManager} options.keyboard_manager
39
* @param {Contents} options.contents
40
* @param {SaveWidget} options.save_widget
41
* @param {object} options.config
42
* @param {string} options.base_url
43
* @param {string} options.notebook_path
44
* @param {string} options.notebook_name
45
*/
46
var Notebook = function (selector, options) {
47
this.config = options.config;
48
this.class_config = new configmod.ConfigWithDefaults(this.config,
49
Notebook.options_default, 'Notebook');
50
this.base_url = options.base_url;
51
this.notebook_path = options.notebook_path;
52
this.notebook_name = options.notebook_name;
53
this.events = options.events;
54
this.keyboard_manager = options.keyboard_manager;
55
this.contents = options.contents;
56
this.save_widget = options.save_widget;
57
this.tooltip = new tooltip.Tooltip(this.events);
58
this.ws_url = options.ws_url;
59
this._session_starting = false;
60
this.last_modified = null;
61
62
// Create default scroll manager.
63
this.scroll_manager = new scrollmanager.ScrollManager(this);
64
65
// TODO: This code smells (and the other `= this` line a couple lines down)
66
// We need a better way to deal with circular instance references.
67
this.keyboard_manager.notebook = this;
68
this.save_widget.notebook = this;
69
70
mathjaxutils.init();
71
72
if (marked) {
73
marked.setOptions({
74
gfm : true,
75
tables: true,
76
// FIXME: probably want central config for CodeMirror theme when we have js config
77
langPrefix: "cm-s-ipython language-",
78
highlight: function(code, lang, callback) {
79
if (!lang) {
80
// no language, no highlight
81
if (callback) {
82
callback(null, code);
83
return;
84
} else {
85
return code;
86
}
87
}
88
utils.requireCodeMirrorMode(lang, function (spec) {
89
var el = document.createElement("div");
90
var mode = CodeMirror.getMode({}, spec);
91
if (!mode) {
92
console.log("No CodeMirror mode: " + lang);
93
callback(null, code);
94
return;
95
}
96
try {
97
CodeMirror.runMode(code, spec, el);
98
callback(null, el.innerHTML);
99
} catch (err) {
100
console.log("Failed to highlight " + lang + " code", err);
101
callback(err, code);
102
}
103
}, function (err) {
104
console.log("No CodeMirror mode: " + lang);
105
callback(err, code);
106
});
107
}
108
});
109
}
110
111
this.element = $(selector);
112
this.element.scroll();
113
this.element.data("notebook", this);
114
this.next_prompt_number = 1;
115
this.session = null;
116
this.kernel = null;
117
this.clipboard = null;
118
this.undelete_backup = null;
119
this.undelete_index = null;
120
this.undelete_below = false;
121
this.paste_enabled = false;
122
this.writable = false;
123
// It is important to start out in command mode to match the intial mode
124
// of the KeyboardManager.
125
this.mode = 'command';
126
this.set_dirty(false);
127
this.metadata = {};
128
this._checkpoint_after_save = false;
129
this.last_checkpoint = null;
130
this.checkpoints = [];
131
this.autosave_interval = 0;
132
this.autosave_timer = null;
133
// autosave *at most* every two minutes
134
this.minimum_autosave_interval = 120000;
135
this.notebook_name_blacklist_re = /[\/\\:]/;
136
this.nbformat = 4; // Increment this when changing the nbformat
137
this.nbformat_minor = this.current_nbformat_minor = 0; // Increment this when changing the nbformat
138
this.codemirror_mode = 'ipython';
139
this.create_elements();
140
this.bind_events();
141
this.kernel_selector = null;
142
this.dirty = null;
143
this.trusted = null;
144
this._fully_loaded = false;
145
146
// Trigger cell toolbar registration.
147
default_celltoolbar.register(this);
148
rawcell_celltoolbar.register(this);
149
slideshow_celltoolbar.register(this);
150
151
// prevent assign to miss-typed properties.
152
Object.seal(this);
153
};
154
155
Notebook.options_default = {
156
// can be any cell type, or the special values of
157
// 'above', 'below', or 'selected' to get the value from another cell.
158
default_cell_type: 'code'
159
};
160
161
/**
162
* Create an HTML and CSS representation of the notebook.
163
*/
164
Notebook.prototype.create_elements = function () {
165
var that = this;
166
this.element.attr('tabindex','-1');
167
this.container = $("<div/>").addClass("container").attr("id", "notebook-container");
168
// We add this end_space div to the end of the notebook div to:
169
// i) provide a margin between the last cell and the end of the notebook
170
// ii) to prevent the div from scrolling up when the last cell is being
171
// edited, but is too low on the page, which browsers will do automatically.
172
var end_space = $('<div/>').addClass('end_space');
173
end_space.dblclick(function (e) {
174
var ncells = that.ncells();
175
that.insert_cell_below('code',ncells-1);
176
});
177
this.element.append(this.container);
178
this.container.after(end_space);
179
};
180
181
/**
182
* Bind JavaScript events: key presses and custom IPython events.
183
*/
184
Notebook.prototype.bind_events = function () {
185
var that = this;
186
187
this.events.on('set_next_input.Notebook', function (event, data) {
188
if (data.replace) {
189
data.cell.set_text(data.text);
190
data.cell.clear_output();
191
} else {
192
var index = that.find_cell_index(data.cell);
193
var new_cell = that.insert_cell_below('code',index);
194
new_cell.set_text(data.text);
195
}
196
that.dirty = true;
197
});
198
199
this.events.on('unrecognized_cell.Cell', function () {
200
that.warn_nbformat_minor();
201
});
202
203
this.events.on('unrecognized_output.OutputArea', function () {
204
that.warn_nbformat_minor();
205
});
206
207
this.events.on('set_dirty.Notebook', function (event, data) {
208
that.dirty = data.value;
209
});
210
211
this.events.on('trust_changed.Notebook', function (event, trusted) {
212
that.trusted = trusted;
213
});
214
215
this.events.on('select.Cell', function (event, data) {
216
var index = that.find_cell_index(data.cell);
217
that.select(index);
218
});
219
220
this.events.on('edit_mode.Cell', function (event, data) {
221
that.handle_edit_mode(data.cell);
222
});
223
224
this.events.on('command_mode.Cell', function (event, data) {
225
that.handle_command_mode(data.cell);
226
});
227
228
this.events.on('spec_changed.Kernel', function(event, data) {
229
that.metadata.kernelspec = {
230
name: data.name,
231
display_name: data.spec.display_name,
232
language: data.spec.language,
233
};
234
// start session if the current session isn't already correct
235
if (!(that.session && that.session.kernel && that.session.kernel.name === data.name)) {
236
that.start_session(data.name);
237
}
238
});
239
240
this.events.on('kernel_ready.Kernel', function(event, data) {
241
var kinfo = data.kernel.info_reply;
242
if (!kinfo.language_info) {
243
delete that.metadata.language_info;
244
return;
245
}
246
var langinfo = kinfo.language_info;
247
that.metadata.language_info = langinfo;
248
// Mode 'null' should be plain, unhighlighted text.
249
var cm_mode = langinfo.codemirror_mode || langinfo.name || 'null';
250
that.set_codemirror_mode(cm_mode);
251
});
252
253
var collapse_time = function (time) {
254
var app_height = $('#ipython-main-app').height(); // content height
255
var splitter_height = $('div#pager_splitter').outerHeight(true);
256
var new_height = app_height - splitter_height;
257
that.element.animate({height : new_height + 'px'}, time);
258
};
259
260
this.element.bind('collapse_pager', function (event, extrap) {
261
var time = (extrap !== undefined) ? ((extrap.duration !== undefined ) ? extrap.duration : 'fast') : 'fast';
262
collapse_time(time);
263
});
264
265
var expand_time = function (time) {
266
var app_height = $('#ipython-main-app').height(); // content height
267
var splitter_height = $('div#pager_splitter').outerHeight(true);
268
var pager_height = $('div#pager').outerHeight(true);
269
var new_height = app_height - pager_height - splitter_height;
270
that.element.animate({height : new_height + 'px'}, time);
271
};
272
273
this.element.bind('expand_pager', function (event, extrap) {
274
var time = (extrap !== undefined) ? ((extrap.duration !== undefined ) ? extrap.duration : 'fast') : 'fast';
275
expand_time(time);
276
});
277
278
// Firefox 22 broke $(window).on("beforeunload")
279
// I'm not sure why or how.
280
window.onbeforeunload = function (e) {
281
// TODO: Make killing the kernel configurable.
282
var kill_kernel = false;
283
if (kill_kernel) {
284
that.session.delete();
285
}
286
// if we are autosaving, trigger an autosave on nav-away.
287
// still warn, because if we don't the autosave may fail.
288
if (that.dirty) {
289
if ( that.autosave_interval ) {
290
// schedule autosave in a timeout
291
// this gives you a chance to forcefully discard changes
292
// by reloading the page if you *really* want to.
293
// the timer doesn't start until you *dismiss* the dialog.
294
setTimeout(function () {
295
if (that.dirty) {
296
that.save_notebook();
297
}
298
}, 1000);
299
return "Autosave in progress, latest changes may be lost.";
300
} else {
301
return "Unsaved changes will be lost.";
302
}
303
}
304
// IE treats null as a string. Instead just return which will avoid the dialog.
305
return;
306
};
307
};
308
309
/**
310
* Trigger a warning dialog about missing functionality from newer minor versions
311
*/
312
Notebook.prototype.warn_nbformat_minor = function (event) {
313
var v = 'v' + this.nbformat + '.';
314
var orig_vs = v + this.nbformat_minor;
315
var this_vs = v + this.current_nbformat_minor;
316
var msg = "This notebook is version " + orig_vs + ", but we only fully support up to " +
317
this_vs + ". You can still work with this notebook, but cell and output types " +
318
"introduced in later notebook versions will not be available.";
319
320
dialog.modal({
321
notebook: this,
322
keyboard_manager: this.keyboard_manager,
323
title : "Newer Notebook",
324
body : msg,
325
buttons : {
326
OK : {
327
"class" : "btn-danger"
328
}
329
}
330
});
331
};
332
333
/**
334
* Set the dirty flag, and trigger the set_dirty.Notebook event
335
*/
336
Notebook.prototype.set_dirty = function (value) {
337
if (value === undefined) {
338
value = true;
339
}
340
if (this.dirty === value) {
341
return;
342
}
343
this.events.trigger('set_dirty.Notebook', {value: value});
344
};
345
346
/**
347
* Scroll the top of the page to a given cell.
348
*
349
* @param {integer} index - An index of the cell to view
350
* @param {integer} time - Animation time in milliseconds
351
* @return {integer} Pixel offset from the top of the container
352
*/
353
Notebook.prototype.scroll_to_cell = function (index, time) {
354
var cells = this.get_cells();
355
time = time || 0;
356
index = Math.min(cells.length-1,index);
357
index = Math.max(0 ,index);
358
var scroll_value = cells[index].element.position().top-cells[0].element.position().top ;
359
this.scroll_manager.element.animate({scrollTop:scroll_value}, time);
360
return scroll_value;
361
};
362
363
/**
364
* Scroll to the bottom of the page.
365
*/
366
Notebook.prototype.scroll_to_bottom = function () {
367
this.scroll_manager.element.animate({scrollTop:this.element.get(0).scrollHeight}, 0);
368
};
369
370
/**
371
* Scroll to the top of the page.
372
*/
373
Notebook.prototype.scroll_to_top = function () {
374
this.scroll_manager.element.animate({scrollTop:0}, 0);
375
};
376
377
// Edit Notebook metadata
378
379
/**
380
* Display a dialog that allows the user to edit the Notebook's metadata.
381
*/
382
Notebook.prototype.edit_metadata = function () {
383
var that = this;
384
dialog.edit_metadata({
385
md: this.metadata,
386
callback: function (md) {
387
that.metadata = md;
388
},
389
name: 'Notebook',
390
notebook: this,
391
keyboard_manager: this.keyboard_manager});
392
};
393
394
// Cell indexing, retrieval, etc.
395
396
/**
397
* Get all cell elements in the notebook.
398
*
399
* @return {jQuery} A selector of all cell elements
400
*/
401
Notebook.prototype.get_cell_elements = function () {
402
return this.container.find(".cell").not('.cell .cell');
403
};
404
405
/**
406
* Get a particular cell element.
407
*
408
* @param {integer} index An index of a cell to select
409
* @return {jQuery} A selector of the given cell.
410
*/
411
Notebook.prototype.get_cell_element = function (index) {
412
var result = null;
413
var e = this.get_cell_elements().eq(index);
414
if (e.length !== 0) {
415
result = e;
416
}
417
return result;
418
};
419
420
/**
421
* Try to get a particular cell by msg_id.
422
*
423
* @param {string} msg_id A message UUID
424
* @return {Cell} Cell or null if no cell was found.
425
*/
426
Notebook.prototype.get_msg_cell = function (msg_id) {
427
return codecell.CodeCell.msg_cells[msg_id] || null;
428
};
429
430
/**
431
* Count the cells in this notebook.
432
*
433
* @return {integer} The number of cells in this notebook
434
*/
435
Notebook.prototype.ncells = function () {
436
return this.get_cell_elements().length;
437
};
438
439
/**
440
* Get all Cell objects in this notebook.
441
*
442
* @return {Array} This notebook's Cell objects
443
*/
444
Notebook.prototype.get_cells = function () {
445
// TODO: we are often calling cells as cells()[i], which we should optimize
446
// to cells(i) or a new method.
447
return this.get_cell_elements().toArray().map(function (e) {
448
return $(e).data("cell");
449
});
450
};
451
452
/**
453
* Get a Cell objects from this notebook.
454
*
455
* @param {integer} index - An index of a cell to retrieve
456
* @return {Cell} Cell or null if no cell was found.
457
*/
458
Notebook.prototype.get_cell = function (index) {
459
var result = null;
460
var ce = this.get_cell_element(index);
461
if (ce !== null) {
462
result = ce.data('cell');
463
}
464
return result;
465
};
466
467
/**
468
* Get the cell below a given cell.
469
*
470
* @param {Cell} cell
471
* @return {Cell} the next cell or null if no cell was found.
472
*/
473
Notebook.prototype.get_next_cell = function (cell) {
474
var result = null;
475
var index = this.find_cell_index(cell);
476
if (this.is_valid_cell_index(index+1)) {
477
result = this.get_cell(index+1);
478
}
479
return result;
480
};
481
482
/**
483
* Get the cell above a given cell.
484
*
485
* @param {Cell} cell
486
* @return {Cell} The previous cell or null if no cell was found.
487
*/
488
Notebook.prototype.get_prev_cell = function (cell) {
489
var result = null;
490
var index = this.find_cell_index(cell);
491
if (index !== null && index > 0) {
492
result = this.get_cell(index-1);
493
}
494
return result;
495
};
496
497
/**
498
* Get the numeric index of a given cell.
499
*
500
* @param {Cell} cell
501
* @return {integer} The cell's numeric index or null if no cell was found.
502
*/
503
Notebook.prototype.find_cell_index = function (cell) {
504
var result = null;
505
this.get_cell_elements().filter(function (index) {
506
if ($(this).data("cell") === cell) {
507
result = index;
508
}
509
});
510
return result;
511
};
512
513
/**
514
* Return given index if defined, or the selected index if not.
515
*
516
* @param {integer} [index] - A cell's index
517
* @return {integer} cell index
518
*/
519
Notebook.prototype.index_or_selected = function (index) {
520
var i;
521
if (index === undefined || index === null) {
522
i = this.get_selected_index();
523
if (i === null) {
524
i = 0;
525
}
526
} else {
527
i = index;
528
}
529
return i;
530
};
531
532
/**
533
* Get the currently selected cell.
534
*
535
* @return {Cell} The selected cell
536
*/
537
Notebook.prototype.get_selected_cell = function () {
538
var index = this.get_selected_index();
539
return this.get_cell(index);
540
};
541
542
/**
543
* Check whether a cell index is valid.
544
*
545
* @param {integer} index - A cell index
546
* @return True if the index is valid, false otherwise
547
*/
548
Notebook.prototype.is_valid_cell_index = function (index) {
549
if (index !== null && index >= 0 && index < this.ncells()) {
550
return true;
551
} else {
552
return false;
553
}
554
};
555
556
/**
557
* Get the index of the currently selected cell.
558
*
559
* @return {integer} The selected cell's numeric index
560
*/
561
Notebook.prototype.get_selected_index = function () {
562
var result = null;
563
this.get_cell_elements().filter(function (index) {
564
if ($(this).data("cell").selected === true) {
565
result = index;
566
}
567
});
568
return result;
569
};
570
571
572
// Cell selection.
573
574
/**
575
* Programmatically select a cell.
576
*
577
* @param {integer} index - A cell's index
578
* @return {Notebook} This notebook
579
*/
580
Notebook.prototype.select = function (index) {
581
if (this.is_valid_cell_index(index)) {
582
var sindex = this.get_selected_index();
583
if (sindex !== null && index !== sindex) {
584
// If we are about to select a different cell, make sure we are
585
// first in command mode.
586
if (this.mode !== 'command') {
587
this.command_mode();
588
}
589
this.get_cell(sindex).unselect();
590
}
591
var cell = this.get_cell(index);
592
cell.select();
593
if (cell.cell_type === 'heading') {
594
this.events.trigger('selected_cell_type_changed.Notebook',
595
{'cell_type':cell.cell_type,level:cell.level}
596
);
597
} else {
598
this.events.trigger('selected_cell_type_changed.Notebook',
599
{'cell_type':cell.cell_type}
600
);
601
}
602
}
603
return this;
604
};
605
606
/**
607
* Programmatically select the next cell.
608
*
609
* @return {Notebook} This notebook
610
*/
611
Notebook.prototype.select_next = function () {
612
var index = this.get_selected_index();
613
this.select(index+1);
614
return this;
615
};
616
617
/**
618
* Programmatically select the previous cell.
619
*
620
* @return {Notebook} This notebook
621
*/
622
Notebook.prototype.select_prev = function () {
623
var index = this.get_selected_index();
624
this.select(index-1);
625
return this;
626
};
627
628
629
// Edit/Command mode
630
631
/**
632
* Gets the index of the cell that is in edit mode.
633
*
634
* @return {integer} index
635
*/
636
Notebook.prototype.get_edit_index = function () {
637
var result = null;
638
this.get_cell_elements().filter(function (index) {
639
if ($(this).data("cell").mode === 'edit') {
640
result = index;
641
}
642
});
643
return result;
644
};
645
646
/**
647
* Handle when a a cell blurs and the notebook should enter command mode.
648
*
649
* @param {Cell} [cell] - Cell to enter command mode on.
650
*/
651
Notebook.prototype.handle_command_mode = function (cell) {
652
if (this.mode !== 'command') {
653
cell.command_mode();
654
this.mode = 'command';
655
this.events.trigger('command_mode.Notebook');
656
this.keyboard_manager.command_mode();
657
}
658
};
659
660
/**
661
* Make the notebook enter command mode.
662
*/
663
Notebook.prototype.command_mode = function () {
664
var cell = this.get_cell(this.get_edit_index());
665
if (cell && this.mode !== 'command') {
666
// We don't call cell.command_mode, but rather blur the CM editor
667
// which will trigger the call to handle_command_mode.
668
cell.code_mirror.getInputField().blur();
669
}
670
};
671
672
/**
673
* Handle when a cell fires it's edit_mode event.
674
*
675
* @param {Cell} [cell] Cell to enter edit mode on.
676
*/
677
Notebook.prototype.handle_edit_mode = function (cell) {
678
if (cell && this.mode !== 'edit') {
679
cell.edit_mode();
680
this.mode = 'edit';
681
this.events.trigger('edit_mode.Notebook');
682
this.keyboard_manager.edit_mode();
683
}
684
};
685
686
/**
687
* Make a cell enter edit mode.
688
*/
689
Notebook.prototype.edit_mode = function () {
690
var cell = this.get_selected_cell();
691
if (cell && this.mode !== 'edit') {
692
cell.unrender();
693
cell.focus_editor();
694
}
695
};
696
697
/**
698
* Ensure either cell, or codemirror is focused. Is none
699
* is focused, focus the cell.
700
*/
701
Notebook.prototype.ensure_focused = function(){
702
var cell = this.get_selected_cell();
703
if (cell === null) {return;} // No cell is selected
704
cell.ensure_focused();
705
}
706
707
/**
708
* Focus the currently selected cell.
709
*/
710
Notebook.prototype.focus_cell = function () {
711
var cell = this.get_selected_cell();
712
if (cell === null) {return;} // No cell is selected
713
cell.focus_cell();
714
};
715
716
// Cell movement
717
718
/**
719
* Move given (or selected) cell up and select it.
720
*
721
* @param {integer} [index] - cell index
722
* @return {Notebook} This notebook
723
*/
724
Notebook.prototype.move_cell_up = function (index) {
725
var i = this.index_or_selected(index);
726
if (this.is_valid_cell_index(i) && i > 0) {
727
var pivot = this.get_cell_element(i-1);
728
var tomove = this.get_cell_element(i);
729
if (pivot !== null && tomove !== null) {
730
tomove.detach();
731
pivot.before(tomove);
732
this.select(i-1);
733
var cell = this.get_selected_cell();
734
cell.focus_cell();
735
}
736
this.set_dirty(true);
737
}
738
return this;
739
};
740
741
742
/**
743
* Move given (or selected) cell down and select it.
744
*
745
* @param {integer} [index] - cell index
746
* @return {Notebook} This notebook
747
*/
748
Notebook.prototype.move_cell_down = function (index) {
749
var i = this.index_or_selected(index);
750
if (this.is_valid_cell_index(i) && this.is_valid_cell_index(i+1)) {
751
var pivot = this.get_cell_element(i+1);
752
var tomove = this.get_cell_element(i);
753
if (pivot !== null && tomove !== null) {
754
tomove.detach();
755
pivot.after(tomove);
756
this.select(i+1);
757
var cell = this.get_selected_cell();
758
cell.focus_cell();
759
}
760
}
761
this.set_dirty();
762
return this;
763
};
764
765
766
// Insertion, deletion.
767
768
/**
769
* Delete a cell from the notebook without any precautions
770
* Needed to reload checkpoints and other things like that.
771
*
772
* @param {integer} [index] - cell's numeric index
773
* @return {Notebook} This notebook
774
*/
775
Notebook.prototype._unsafe_delete_cell = function (index) {
776
var i = this.index_or_selected(index);
777
var cell = this.get_cell(i);
778
779
$('#undelete_cell').addClass('disabled');
780
if (this.is_valid_cell_index(i)) {
781
var old_ncells = this.ncells();
782
var ce = this.get_cell_element(i);
783
ce.remove();
784
this.set_dirty(true);
785
}
786
return this;
787
};
788
789
/**
790
* Delete a cell from the notebook.
791
*
792
* @param {integer} [index] - cell's numeric index
793
* @return {Notebook} This notebook
794
*/
795
Notebook.prototype.delete_cell = function (index) {
796
var i = this.index_or_selected(index);
797
var cell = this.get_cell(i);
798
if (!cell.is_deletable()) {
799
return this;
800
}
801
802
this.undelete_backup = cell.toJSON();
803
$('#undelete_cell').removeClass('disabled');
804
if (this.is_valid_cell_index(i)) {
805
var old_ncells = this.ncells();
806
var ce = this.get_cell_element(i);
807
ce.remove();
808
if (i === 0) {
809
// Always make sure we have at least one cell.
810
if (old_ncells === 1) {
811
this.insert_cell_below('code');
812
}
813
this.select(0);
814
this.undelete_index = 0;
815
this.undelete_below = false;
816
} else if (i === old_ncells-1 && i !== 0) {
817
this.select(i-1);
818
this.undelete_index = i - 1;
819
this.undelete_below = true;
820
} else {
821
this.select(i);
822
this.undelete_index = i;
823
this.undelete_below = false;
824
}
825
this.events.trigger('delete.Cell', {'cell': cell, 'index': i});
826
this.set_dirty(true);
827
}
828
return this;
829
};
830
831
/**
832
* Restore the most recently deleted cell.
833
*/
834
Notebook.prototype.undelete_cell = function() {
835
if (this.undelete_backup !== null && this.undelete_index !== null) {
836
var current_index = this.get_selected_index();
837
if (this.undelete_index < current_index) {
838
current_index = current_index + 1;
839
}
840
if (this.undelete_index >= this.ncells()) {
841
this.select(this.ncells() - 1);
842
}
843
else {
844
this.select(this.undelete_index);
845
}
846
var cell_data = this.undelete_backup;
847
var new_cell = null;
848
if (this.undelete_below) {
849
new_cell = this.insert_cell_below(cell_data.cell_type);
850
} else {
851
new_cell = this.insert_cell_above(cell_data.cell_type);
852
}
853
new_cell.fromJSON(cell_data);
854
if (this.undelete_below) {
855
this.select(current_index+1);
856
} else {
857
this.select(current_index);
858
}
859
this.undelete_backup = null;
860
this.undelete_index = null;
861
}
862
$('#undelete_cell').addClass('disabled');
863
};
864
865
/**
866
* Insert a cell so that after insertion the cell is at given index.
867
*
868
* If cell type is not provided, it will default to the type of the
869
* currently active cell.
870
*
871
* Similar to insert_above, but index parameter is mandatory.
872
*
873
* Index will be brought back into the accessible range [0,n].
874
*
875
* @param {string} [type] - in ['code','markdown', 'raw'], defaults to 'code'
876
* @param {integer} [index] - a valid index where to insert cell
877
* @return {Cell|null} created cell or null
878
*/
879
Notebook.prototype.insert_cell_at_index = function(type, index){
880
881
var ncells = this.ncells();
882
index = Math.min(index, ncells);
883
index = Math.max(index, 0);
884
var cell = null;
885
type = type || this.class_config.get_sync('default_cell_type');
886
if (type === 'above') {
887
if (index > 0) {
888
type = this.get_cell(index-1).cell_type;
889
} else {
890
type = 'code';
891
}
892
} else if (type === 'below') {
893
if (index < ncells) {
894
type = this.get_cell(index).cell_type;
895
} else {
896
type = 'code';
897
}
898
} else if (type === 'selected') {
899
type = this.get_selected_cell().cell_type;
900
}
901
902
if (ncells === 0 || this.is_valid_cell_index(index) || index === ncells) {
903
var cell_options = {
904
events: this.events,
905
config: this.config,
906
keyboard_manager: this.keyboard_manager,
907
notebook: this,
908
tooltip: this.tooltip
909
};
910
switch(type) {
911
case 'code':
912
cell = new codecell.CodeCell(this.kernel, cell_options);
913
cell.set_input_prompt();
914
break;
915
case 'markdown':
916
cell = new textcell.MarkdownCell(cell_options);
917
break;
918
case 'raw':
919
cell = new textcell.RawCell(cell_options);
920
break;
921
default:
922
console.log("Unrecognized cell type: ", type, cellmod);
923
cell = new cellmod.UnrecognizedCell(cell_options);
924
}
925
926
if(this._insert_element_at_index(cell.element,index)) {
927
cell.render();
928
this.events.trigger('create.Cell', {'cell': cell, 'index': index});
929
cell.refresh();
930
// We used to select the cell after we refresh it, but there
931
// are now cases were this method is called where select is
932
// not appropriate. The selection logic should be handled by the
933
// caller of the the top level insert_cell methods.
934
this.set_dirty(true);
935
}
936
}
937
return cell;
938
939
};
940
941
/**
942
* Insert an element at given cell index.
943
*
944
* @param {HTMLElement} element - a cell element
945
* @param {integer} [index] - a valid index where to inser cell
946
* @returns {boolean} success
947
*/
948
Notebook.prototype._insert_element_at_index = function(element, index){
949
if (element === undefined){
950
return false;
951
}
952
953
var ncells = this.ncells();
954
955
if (ncells === 0) {
956
// special case append if empty
957
this.container.append(element);
958
} else if ( ncells === index ) {
959
// special case append it the end, but not empty
960
this.get_cell_element(index-1).after(element);
961
} else if (this.is_valid_cell_index(index)) {
962
// otherwise always somewhere to append to
963
this.get_cell_element(index).before(element);
964
} else {
965
return false;
966
}
967
968
if (this.undelete_index !== null && index <= this.undelete_index) {
969
this.undelete_index = this.undelete_index + 1;
970
this.set_dirty(true);
971
}
972
return true;
973
};
974
975
/**
976
* Insert a cell of given type above given index, or at top
977
* of notebook if index smaller than 0.
978
*
979
* @param {string} [type] - cell type
980
* @param {integer} [index] - defaults to the currently selected cell
981
* @return {Cell|null} handle to created cell or null
982
*/
983
Notebook.prototype.insert_cell_above = function (type, index) {
984
index = this.index_or_selected(index);
985
return this.insert_cell_at_index(type, index);
986
};
987
988
/**
989
* Insert a cell of given type below given index, or at bottom
990
* of notebook if index greater than number of cells
991
*
992
* @param {string} [type] - cell type
993
* @param {integer} [index] - defaults to the currently selected cell
994
* @return {Cell|null} handle to created cell or null
995
*/
996
Notebook.prototype.insert_cell_below = function (type, index) {
997
index = this.index_or_selected(index);
998
return this.insert_cell_at_index(type, index+1);
999
};
1000
1001
1002
/**
1003
* Insert cell at end of notebook
1004
*
1005
* @param {string} type - cell type
1006
* @return {Cell|null} handle to created cell or null
1007
*/
1008
Notebook.prototype.insert_cell_at_bottom = function (type){
1009
var len = this.ncells();
1010
return this.insert_cell_below(type,len-1);
1011
};
1012
1013
/**
1014
* Turn a cell into a code cell.
1015
*
1016
* @param {integer} [index] - cell index
1017
*/
1018
Notebook.prototype.to_code = function (index) {
1019
var i = this.index_or_selected(index);
1020
if (this.is_valid_cell_index(i)) {
1021
var source_cell = this.get_cell(i);
1022
if (!(source_cell instanceof codecell.CodeCell)) {
1023
var target_cell = this.insert_cell_below('code',i);
1024
var text = source_cell.get_text();
1025
if (text === source_cell.placeholder) {
1026
text = '';
1027
}
1028
//metadata
1029
target_cell.metadata = source_cell.metadata;
1030
1031
target_cell.set_text(text);
1032
// make this value the starting point, so that we can only undo
1033
// to this state, instead of a blank cell
1034
target_cell.code_mirror.clearHistory();
1035
source_cell.element.remove();
1036
this.select(i);
1037
var cursor = source_cell.code_mirror.getCursor();
1038
target_cell.code_mirror.setCursor(cursor);
1039
this.set_dirty(true);
1040
}
1041
}
1042
};
1043
1044
/**
1045
* Turn a cell into a Markdown cell.
1046
*
1047
* @param {integer} [index] - cell index
1048
*/
1049
Notebook.prototype.to_markdown = function (index) {
1050
var i = this.index_or_selected(index);
1051
if (this.is_valid_cell_index(i)) {
1052
var source_cell = this.get_cell(i);
1053
1054
if (!(source_cell instanceof textcell.MarkdownCell)) {
1055
var target_cell = this.insert_cell_below('markdown',i);
1056
var text = source_cell.get_text();
1057
1058
if (text === source_cell.placeholder) {
1059
text = '';
1060
}
1061
// metadata
1062
target_cell.metadata = source_cell.metadata;
1063
// We must show the editor before setting its contents
1064
target_cell.unrender();
1065
target_cell.set_text(text);
1066
// make this value the starting point, so that we can only undo
1067
// to this state, instead of a blank cell
1068
target_cell.code_mirror.clearHistory();
1069
source_cell.element.remove();
1070
this.select(i);
1071
if ((source_cell instanceof textcell.TextCell) && source_cell.rendered) {
1072
target_cell.render();
1073
}
1074
var cursor = source_cell.code_mirror.getCursor();
1075
target_cell.code_mirror.setCursor(cursor);
1076
this.set_dirty(true);
1077
}
1078
}
1079
};
1080
1081
/**
1082
* Turn a cell into a raw text cell.
1083
*
1084
* @param {integer} [index] - cell index
1085
*/
1086
Notebook.prototype.to_raw = function (index) {
1087
var i = this.index_or_selected(index);
1088
if (this.is_valid_cell_index(i)) {
1089
var target_cell = null;
1090
var source_cell = this.get_cell(i);
1091
1092
if (!(source_cell instanceof textcell.RawCell)) {
1093
target_cell = this.insert_cell_below('raw',i);
1094
var text = source_cell.get_text();
1095
if (text === source_cell.placeholder) {
1096
text = '';
1097
}
1098
//metadata
1099
target_cell.metadata = source_cell.metadata;
1100
// We must show the editor before setting its contents
1101
target_cell.unrender();
1102
target_cell.set_text(text);
1103
// make this value the starting point, so that we can only undo
1104
// to this state, instead of a blank cell
1105
target_cell.code_mirror.clearHistory();
1106
source_cell.element.remove();
1107
this.select(i);
1108
var cursor = source_cell.code_mirror.getCursor();
1109
target_cell.code_mirror.setCursor(cursor);
1110
this.set_dirty(true);
1111
}
1112
}
1113
};
1114
1115
/**
1116
* Warn about heading cell support removal.
1117
*/
1118
Notebook.prototype._warn_heading = function () {
1119
dialog.modal({
1120
notebook: this,
1121
keyboard_manager: this.keyboard_manager,
1122
title : "Use markdown headings",
1123
body : $("<p/>").text(
1124
'IPython no longer uses special heading cells. ' +
1125
'Instead, write your headings in Markdown cells using # characters:'
1126
).append($('<pre/>').text(
1127
'## This is a level 2 heading'
1128
)),
1129
buttons : {
1130
"OK" : {}
1131
}
1132
});
1133
};
1134
1135
/**
1136
* Turn a cell into a heading containing markdown cell.
1137
*
1138
* @param {integer} [index] - cell index
1139
* @param {integer} [level] - heading level (e.g., 1 for h1)
1140
*/
1141
Notebook.prototype.to_heading = function (index, level) {
1142
this.to_markdown(index);
1143
level = level || 1;
1144
var i = this.index_or_selected(index);
1145
if (this.is_valid_cell_index(i)) {
1146
var cell = this.get_cell(i);
1147
cell.set_heading_level(level);
1148
this.set_dirty(true);
1149
}
1150
};
1151
1152
1153
// Cut/Copy/Paste
1154
1155
/**
1156
* Enable the UI elements for pasting cells.
1157
*/
1158
Notebook.prototype.enable_paste = function () {
1159
var that = this;
1160
if (!this.paste_enabled) {
1161
$('#paste_cell_replace').removeClass('disabled')
1162
.on('click', function () {that.paste_cell_replace();});
1163
$('#paste_cell_above').removeClass('disabled')
1164
.on('click', function () {that.paste_cell_above();});
1165
$('#paste_cell_below').removeClass('disabled')
1166
.on('click', function () {that.paste_cell_below();});
1167
this.paste_enabled = true;
1168
}
1169
};
1170
1171
/**
1172
* Disable the UI elements for pasting cells.
1173
*/
1174
Notebook.prototype.disable_paste = function () {
1175
if (this.paste_enabled) {
1176
$('#paste_cell_replace').addClass('disabled').off('click');
1177
$('#paste_cell_above').addClass('disabled').off('click');
1178
$('#paste_cell_below').addClass('disabled').off('click');
1179
this.paste_enabled = false;
1180
}
1181
};
1182
1183
/**
1184
* Cut a cell.
1185
*/
1186
Notebook.prototype.cut_cell = function () {
1187
this.copy_cell();
1188
this.delete_cell();
1189
};
1190
1191
/**
1192
* Copy a cell.
1193
*/
1194
Notebook.prototype.copy_cell = function () {
1195
var cell = this.get_selected_cell();
1196
this.clipboard = cell.toJSON();
1197
// remove undeletable status from the copied cell
1198
if (this.clipboard.metadata.deletable !== undefined) {
1199
delete this.clipboard.metadata.deletable;
1200
}
1201
this.enable_paste();
1202
};
1203
1204
/**
1205
* Replace the selected cell with the cell in the clipboard.
1206
*/
1207
Notebook.prototype.paste_cell_replace = function () {
1208
if (this.clipboard !== null && this.paste_enabled) {
1209
var cell_data = this.clipboard;
1210
var new_cell = this.insert_cell_above(cell_data.cell_type);
1211
new_cell.fromJSON(cell_data);
1212
var old_cell = this.get_next_cell(new_cell);
1213
this.delete_cell(this.find_cell_index(old_cell));
1214
this.select(this.find_cell_index(new_cell));
1215
}
1216
};
1217
1218
/**
1219
* Paste a cell from the clipboard above the selected cell.
1220
*/
1221
Notebook.prototype.paste_cell_above = function () {
1222
if (this.clipboard !== null && this.paste_enabled) {
1223
var cell_data = this.clipboard;
1224
var new_cell = this.insert_cell_above(cell_data.cell_type);
1225
new_cell.fromJSON(cell_data);
1226
new_cell.focus_cell();
1227
}
1228
};
1229
1230
/**
1231
* Paste a cell from the clipboard below the selected cell.
1232
*/
1233
Notebook.prototype.paste_cell_below = function () {
1234
if (this.clipboard !== null && this.paste_enabled) {
1235
var cell_data = this.clipboard;
1236
var new_cell = this.insert_cell_below(cell_data.cell_type);
1237
new_cell.fromJSON(cell_data);
1238
new_cell.focus_cell();
1239
}
1240
};
1241
1242
// Split/merge
1243
1244
/**
1245
* Split the selected cell into two cells.
1246
*/
1247
Notebook.prototype.split_cell = function () {
1248
var cell = this.get_selected_cell();
1249
if (cell.is_splittable()) {
1250
var texta = cell.get_pre_cursor();
1251
var textb = cell.get_post_cursor();
1252
cell.set_text(textb);
1253
var new_cell = this.insert_cell_above(cell.cell_type);
1254
// Unrender the new cell so we can call set_text.
1255
new_cell.unrender();
1256
new_cell.set_text(texta);
1257
}
1258
};
1259
1260
/**
1261
* Merge the selected cell into the cell above it.
1262
*/
1263
Notebook.prototype.merge_cell_above = function () {
1264
var index = this.get_selected_index();
1265
var cell = this.get_cell(index);
1266
var render = cell.rendered;
1267
if (!cell.is_mergeable()) {
1268
return;
1269
}
1270
if (index > 0) {
1271
var upper_cell = this.get_cell(index-1);
1272
if (!upper_cell.is_mergeable()) {
1273
return;
1274
}
1275
var upper_text = upper_cell.get_text();
1276
var text = cell.get_text();
1277
if (cell instanceof codecell.CodeCell) {
1278
cell.set_text(upper_text+'\n'+text);
1279
} else {
1280
cell.unrender(); // Must unrender before we set_text.
1281
cell.set_text(upper_text+'\n\n'+text);
1282
if (render) {
1283
// The rendered state of the final cell should match
1284
// that of the original selected cell;
1285
cell.render();
1286
}
1287
}
1288
this.delete_cell(index-1);
1289
this.select(this.find_cell_index(cell));
1290
}
1291
};
1292
1293
/**
1294
* Merge the selected cell into the cell below it.
1295
*/
1296
Notebook.prototype.merge_cell_below = function () {
1297
var index = this.get_selected_index();
1298
var cell = this.get_cell(index);
1299
var render = cell.rendered;
1300
if (!cell.is_mergeable()) {
1301
return;
1302
}
1303
if (index < this.ncells()-1) {
1304
var lower_cell = this.get_cell(index+1);
1305
if (!lower_cell.is_mergeable()) {
1306
return;
1307
}
1308
var lower_text = lower_cell.get_text();
1309
var text = cell.get_text();
1310
if (cell instanceof codecell.CodeCell) {
1311
cell.set_text(text+'\n'+lower_text);
1312
} else {
1313
cell.unrender(); // Must unrender before we set_text.
1314
cell.set_text(text+'\n\n'+lower_text);
1315
if (render) {
1316
// The rendered state of the final cell should match
1317
// that of the original selected cell;
1318
cell.render();
1319
}
1320
}
1321
this.delete_cell(index+1);
1322
this.select(this.find_cell_index(cell));
1323
}
1324
};
1325
1326
1327
// Cell collapsing and output clearing
1328
1329
/**
1330
* Hide a cell's output.
1331
*
1332
* @param {integer} index - cell index
1333
*/
1334
Notebook.prototype.collapse_output = function (index) {
1335
var i = this.index_or_selected(index);
1336
var cell = this.get_cell(i);
1337
if (cell !== null && (cell instanceof codecell.CodeCell)) {
1338
cell.collapse_output();
1339
this.set_dirty(true);
1340
}
1341
};
1342
1343
/**
1344
* Hide each code cell's output area.
1345
*/
1346
Notebook.prototype.collapse_all_output = function () {
1347
this.get_cells().map(function (cell, i) {
1348
if (cell instanceof codecell.CodeCell) {
1349
cell.collapse_output();
1350
}
1351
});
1352
// this should not be set if the `collapse` key is removed from nbformat
1353
this.set_dirty(true);
1354
};
1355
1356
/**
1357
* Show a cell's output.
1358
*
1359
* @param {integer} index - cell index
1360
*/
1361
Notebook.prototype.expand_output = function (index) {
1362
var i = this.index_or_selected(index);
1363
var cell = this.get_cell(i);
1364
if (cell !== null && (cell instanceof codecell.CodeCell)) {
1365
cell.expand_output();
1366
this.set_dirty(true);
1367
}
1368
};
1369
1370
/**
1371
* Expand each code cell's output area, and remove scrollbars.
1372
*/
1373
Notebook.prototype.expand_all_output = function () {
1374
this.get_cells().map(function (cell, i) {
1375
if (cell instanceof codecell.CodeCell) {
1376
cell.expand_output();
1377
}
1378
});
1379
// this should not be set if the `collapse` key is removed from nbformat
1380
this.set_dirty(true);
1381
};
1382
1383
/**
1384
* Clear the selected CodeCell's output area.
1385
*
1386
* @param {integer} index - cell index
1387
*/
1388
Notebook.prototype.clear_output = function (index) {
1389
var i = this.index_or_selected(index);
1390
var cell = this.get_cell(i);
1391
if (cell !== null && (cell instanceof codecell.CodeCell)) {
1392
cell.clear_output();
1393
this.set_dirty(true);
1394
}
1395
};
1396
1397
/**
1398
* Clear each code cell's output area.
1399
*/
1400
Notebook.prototype.clear_all_output = function () {
1401
this.get_cells().map(function (cell, i) {
1402
if (cell instanceof codecell.CodeCell) {
1403
cell.clear_output();
1404
}
1405
});
1406
this.set_dirty(true);
1407
};
1408
1409
/**
1410
* Scroll the selected CodeCell's output area.
1411
*
1412
* @param {integer} index - cell index
1413
*/
1414
Notebook.prototype.scroll_output = function (index) {
1415
var i = this.index_or_selected(index);
1416
var cell = this.get_cell(i);
1417
if (cell !== null && (cell instanceof codecell.CodeCell)) {
1418
cell.scroll_output();
1419
this.set_dirty(true);
1420
}
1421
};
1422
1423
/**
1424
* Expand each code cell's output area and add a scrollbar for long output.
1425
*/
1426
Notebook.prototype.scroll_all_output = function () {
1427
this.get_cells().map(function (cell, i) {
1428
if (cell instanceof codecell.CodeCell) {
1429
cell.scroll_output();
1430
}
1431
});
1432
// this should not be set if the `collapse` key is removed from nbformat
1433
this.set_dirty(true);
1434
};
1435
1436
/**
1437
* Toggle whether a cell's output is collapsed or expanded.
1438
*
1439
* @param {integer} index - cell index
1440
*/
1441
Notebook.prototype.toggle_output = function (index) {
1442
var i = this.index_or_selected(index);
1443
var cell = this.get_cell(i);
1444
if (cell !== null && (cell instanceof codecell.CodeCell)) {
1445
cell.toggle_output();
1446
this.set_dirty(true);
1447
}
1448
};
1449
1450
/**
1451
* Toggle the output of all cells.
1452
*/
1453
Notebook.prototype.toggle_all_output = function () {
1454
this.get_cells().map(function (cell, i) {
1455
if (cell instanceof codecell.CodeCell) {
1456
cell.toggle_output();
1457
}
1458
});
1459
// this should not be set if the `collapse` key is removed from nbformat
1460
this.set_dirty(true);
1461
};
1462
1463
/**
1464
* Toggle a scrollbar for long cell outputs.
1465
*
1466
* @param {integer} index - cell index
1467
*/
1468
Notebook.prototype.toggle_output_scroll = function (index) {
1469
var i = this.index_or_selected(index);
1470
var cell = this.get_cell(i);
1471
if (cell !== null && (cell instanceof codecell.CodeCell)) {
1472
cell.toggle_output_scroll();
1473
this.set_dirty(true);
1474
}
1475
};
1476
1477
/**
1478
* Toggle the scrolling of long output on all cells.
1479
*/
1480
Notebook.prototype.toggle_all_output_scroll = function () {
1481
this.get_cells().map(function (cell, i) {
1482
if (cell instanceof codecell.CodeCell) {
1483
cell.toggle_output_scroll();
1484
}
1485
});
1486
// this should not be set if the `collapse` key is removed from nbformat
1487
this.set_dirty(true);
1488
};
1489
1490
// Other cell functions: line numbers, ...
1491
1492
/**
1493
* Toggle line numbers in the selected cell's input area.
1494
*/
1495
Notebook.prototype.cell_toggle_line_numbers = function() {
1496
this.get_selected_cell().toggle_line_numbers();
1497
};
1498
1499
/**
1500
* Set the codemirror mode for all code cells, including the default for
1501
* new code cells.
1502
*/
1503
Notebook.prototype.set_codemirror_mode = function(newmode){
1504
if (newmode === this.codemirror_mode) {
1505
return;
1506
}
1507
this.codemirror_mode = newmode;
1508
codecell.CodeCell.options_default.cm_config.mode = newmode;
1509
1510
var that = this;
1511
utils.requireCodeMirrorMode(newmode, function (spec) {
1512
that.get_cells().map(function(cell, i) {
1513
if (cell.cell_type === 'code'){
1514
cell.code_mirror.setOption('mode', spec);
1515
// This is currently redundant, because cm_config ends up as
1516
// codemirror's own .options object, but I don't want to
1517
// rely on that.
1518
cell.cm_config.mode = spec;
1519
}
1520
});
1521
});
1522
};
1523
1524
// Session related things
1525
1526
/**
1527
* Start a new session and set it on each code cell.
1528
*/
1529
Notebook.prototype.start_session = function (kernel_name) {
1530
if (this._session_starting) {
1531
throw new session.SessionAlreadyStarting();
1532
}
1533
this._session_starting = true;
1534
1535
var options = {
1536
base_url: this.base_url,
1537
ws_url: this.ws_url,
1538
notebook_path: this.notebook_path,
1539
notebook_name: this.notebook_name,
1540
kernel_name: kernel_name,
1541
notebook: this
1542
};
1543
1544
var success = $.proxy(this._session_started, this);
1545
var failure = $.proxy(this._session_start_failed, this);
1546
1547
if (this.session !== null) {
1548
this.session.restart(options, success, failure);
1549
} else {
1550
this.session = new session.Session(options);
1551
this.session.start(success, failure);
1552
}
1553
};
1554
1555
1556
/**
1557
* Once a session is started, link the code cells to the kernel and pass the
1558
* comm manager to the widget manager.
1559
*/
1560
Notebook.prototype._session_started = function (){
1561
this._session_starting = false;
1562
this.kernel = this.session.kernel;
1563
var ncells = this.ncells();
1564
for (var i=0; i<ncells; i++) {
1565
var cell = this.get_cell(i);
1566
if (cell instanceof codecell.CodeCell) {
1567
cell.set_kernel(this.session.kernel);
1568
}
1569
}
1570
};
1571
1572
/**
1573
* Called when the session fails to start.
1574
*/
1575
Notebook.prototype._session_start_failed = function(jqxhr, status, error){
1576
this._session_starting = false;
1577
utils.log_ajax_error(jqxhr, status, error);
1578
};
1579
1580
/**
1581
* Prompt the user to restart the IPython kernel.
1582
*/
1583
Notebook.prototype.restart_kernel = function () {
1584
var that = this;
1585
dialog.modal({
1586
notebook: this,
1587
keyboard_manager: this.keyboard_manager,
1588
title : "Restart kernel or continue running?",
1589
body : $("<p/>").text(
1590
'Do you want to restart the current kernel? You will lose all variables defined in it.'
1591
),
1592
buttons : {
1593
"Continue running" : {},
1594
"Restart" : {
1595
"class" : "btn-danger",
1596
"click" : function() {
1597
that.kernel.restart();
1598
}
1599
}
1600
}
1601
});
1602
};
1603
1604
/**
1605
* Execute or render cell outputs and go into command mode.
1606
*/
1607
Notebook.prototype.execute_cell = function () {
1608
// mode = shift, ctrl, alt
1609
var cell = this.get_selected_cell();
1610
1611
cell.execute();
1612
this.command_mode();
1613
this.set_dirty(true);
1614
};
1615
1616
/**
1617
* Execute or render cell outputs and insert a new cell below.
1618
*/
1619
Notebook.prototype.execute_cell_and_insert_below = function () {
1620
var cell = this.get_selected_cell();
1621
var cell_index = this.find_cell_index(cell);
1622
1623
cell.execute();
1624
1625
// If we are at the end always insert a new cell and return
1626
if (cell_index === (this.ncells()-1)) {
1627
this.command_mode();
1628
this.insert_cell_below();
1629
this.select(cell_index+1);
1630
this.edit_mode();
1631
this.scroll_to_bottom();
1632
this.set_dirty(true);
1633
return;
1634
}
1635
1636
this.command_mode();
1637
this.insert_cell_below();
1638
this.select(cell_index+1);
1639
this.edit_mode();
1640
this.set_dirty(true);
1641
};
1642
1643
/**
1644
* Execute or render cell outputs and select the next cell.
1645
*/
1646
Notebook.prototype.execute_cell_and_select_below = function () {
1647
1648
var cell = this.get_selected_cell();
1649
var cell_index = this.find_cell_index(cell);
1650
1651
cell.execute();
1652
1653
// If we are at the end always insert a new cell and return
1654
if (cell_index === (this.ncells()-1)) {
1655
this.command_mode();
1656
this.insert_cell_below();
1657
this.select(cell_index+1);
1658
this.edit_mode();
1659
this.scroll_to_bottom();
1660
this.set_dirty(true);
1661
return;
1662
}
1663
1664
this.command_mode();
1665
this.select(cell_index+1);
1666
this.focus_cell();
1667
this.set_dirty(true);
1668
};
1669
1670
/**
1671
* Execute all cells below the selected cell.
1672
*/
1673
Notebook.prototype.execute_cells_below = function () {
1674
this.execute_cell_range(this.get_selected_index(), this.ncells());
1675
this.scroll_to_bottom();
1676
};
1677
1678
/**
1679
* Execute all cells above the selected cell.
1680
*/
1681
Notebook.prototype.execute_cells_above = function () {
1682
this.execute_cell_range(0, this.get_selected_index());
1683
};
1684
1685
/**
1686
* Execute all cells.
1687
*/
1688
Notebook.prototype.execute_all_cells = function () {
1689
this.execute_cell_range(0, this.ncells());
1690
this.scroll_to_bottom();
1691
};
1692
1693
/**
1694
* Execute a contiguous range of cells.
1695
*
1696
* @param {integer} start - index of the first cell to execute (inclusive)
1697
* @param {integer} end - index of the last cell to execute (exclusive)
1698
*/
1699
Notebook.prototype.execute_cell_range = function (start, end) {
1700
this.command_mode();
1701
for (var i=start; i<end; i++) {
1702
this.select(i);
1703
this.execute_cell();
1704
}
1705
};
1706
1707
// Persistance and loading
1708
1709
/**
1710
* Getter method for this notebook's name.
1711
*
1712
* @return {string} This notebook's name (excluding file extension)
1713
*/
1714
Notebook.prototype.get_notebook_name = function () {
1715
var nbname = this.notebook_name.substring(0,this.notebook_name.length-6);
1716
return nbname;
1717
};
1718
1719
/**
1720
* Setter method for this notebook's name.
1721
*
1722
* @param {string} name
1723
*/
1724
Notebook.prototype.set_notebook_name = function (name) {
1725
var parent = utils.url_path_split(this.notebook_path)[0];
1726
this.notebook_name = name;
1727
this.notebook_path = utils.url_path_join(parent, name);
1728
};
1729
1730
/**
1731
* Check that a notebook's name is valid.
1732
*
1733
* @param {string} nbname - A name for this notebook
1734
* @return {boolean} True if the name is valid, false if invalid
1735
*/
1736
Notebook.prototype.test_notebook_name = function (nbname) {
1737
nbname = nbname || '';
1738
if (nbname.length>0 && !this.notebook_name_blacklist_re.test(nbname)) {
1739
return true;
1740
} else {
1741
return false;
1742
}
1743
};
1744
1745
/**
1746
* Load a notebook from JSON (.ipynb).
1747
*
1748
* @param {object} data - JSON representation of a notebook
1749
*/
1750
Notebook.prototype.fromJSON = function (data) {
1751
1752
var content = data.content;
1753
var ncells = this.ncells();
1754
var i;
1755
for (i=0; i<ncells; i++) {
1756
// Always delete cell 0 as they get renumbered as they are deleted.
1757
this._unsafe_delete_cell(0);
1758
}
1759
// Save the metadata and name.
1760
this.metadata = content.metadata;
1761
this.notebook_name = data.name;
1762
this.notebook_path = data.path;
1763
var trusted = true;
1764
1765
// Set the codemirror mode from language_info metadata
1766
if (this.metadata.language_info !== undefined) {
1767
var langinfo = this.metadata.language_info;
1768
// Mode 'null' should be plain, unhighlighted text.
1769
var cm_mode = langinfo.codemirror_mode || langinfo.name || 'null';
1770
this.set_codemirror_mode(cm_mode);
1771
}
1772
1773
var new_cells = content.cells;
1774
ncells = new_cells.length;
1775
var cell_data = null;
1776
var new_cell = null;
1777
for (i=0; i<ncells; i++) {
1778
cell_data = new_cells[i];
1779
new_cell = this.insert_cell_at_index(cell_data.cell_type, i);
1780
new_cell.fromJSON(cell_data);
1781
if (new_cell.cell_type === 'code' && !new_cell.output_area.trusted) {
1782
trusted = false;
1783
}
1784
}
1785
if (trusted !== this.trusted) {
1786
this.trusted = trusted;
1787
this.events.trigger("trust_changed.Notebook", trusted);
1788
}
1789
};
1790
1791
/**
1792
* Dump this notebook into a JSON-friendly object.
1793
*
1794
* @return {object} A JSON-friendly representation of this notebook.
1795
*/
1796
Notebook.prototype.toJSON = function () {
1797
// remove the conversion indicator, which only belongs in-memory
1798
delete this.metadata.orig_nbformat;
1799
delete this.metadata.orig_nbformat_minor;
1800
1801
var cells = this.get_cells();
1802
var ncells = cells.length;
1803
var cell_array = new Array(ncells);
1804
var trusted = true;
1805
for (var i=0; i<ncells; i++) {
1806
var cell = cells[i];
1807
if (cell.cell_type === 'code' && !cell.output_area.trusted) {
1808
trusted = false;
1809
}
1810
cell_array[i] = cell.toJSON();
1811
}
1812
var data = {
1813
cells: cell_array,
1814
metadata: this.metadata,
1815
nbformat: this.nbformat,
1816
nbformat_minor: this.nbformat_minor
1817
};
1818
if (trusted !== this.trusted) {
1819
this.trusted = trusted;
1820
this.events.trigger("trust_changed.Notebook", trusted);
1821
}
1822
return data;
1823
};
1824
1825
/**
1826
* Start an autosave timer which periodically saves the notebook.
1827
*
1828
* @param {integer} interval - the autosave interval in milliseconds
1829
*/
1830
Notebook.prototype.set_autosave_interval = function (interval) {
1831
var that = this;
1832
// clear previous interval, so we don't get simultaneous timers
1833
if (this.autosave_timer) {
1834
clearInterval(this.autosave_timer);
1835
}
1836
if (!this.writable) {
1837
// disable autosave if not writable
1838
interval = 0;
1839
}
1840
1841
this.autosave_interval = this.minimum_autosave_interval = interval;
1842
if (interval) {
1843
this.autosave_timer = setInterval(function() {
1844
if (that.dirty) {
1845
that.save_notebook();
1846
}
1847
}, interval);
1848
this.events.trigger("autosave_enabled.Notebook", interval);
1849
} else {
1850
this.autosave_timer = null;
1851
this.events.trigger("autosave_disabled.Notebook");
1852
}
1853
};
1854
1855
/**
1856
* Save this notebook on the server. This becomes a notebook instance's
1857
* .save_notebook method *after* the entire notebook has been loaded.
1858
*/
1859
Notebook.prototype.save_notebook = function (check_last_modified) {
1860
if (check_last_modified === undefined) {
1861
check_last_modified = true;
1862
}
1863
if (!this._fully_loaded) {
1864
this.events.trigger('notebook_save_failed.Notebook',
1865
new Error("Load failed, save is disabled")
1866
);
1867
return;
1868
} else if (!this.writable) {
1869
this.events.trigger('notebook_save_failed.Notebook',
1870
new Error("Notebook is read-only")
1871
);
1872
return;
1873
}
1874
1875
// Trigger an event before save, which allows listeners to modify
1876
// the notebook as needed.
1877
this.events.trigger('before_save.Notebook');
1878
1879
// Create a JSON model to be sent to the server.
1880
var model = {
1881
type : "notebook",
1882
content : this.toJSON()
1883
};
1884
// time the ajax call for autosave tuning purposes.
1885
var start = new Date().getTime();
1886
1887
var that = this;
1888
var _save = function () {
1889
return that.contents.save(that.notebook_path, model).then(
1890
$.proxy(that.save_notebook_success, that, start),
1891
function (error) {
1892
that.events.trigger('notebook_save_failed.Notebook', error);
1893
}
1894
);
1895
};
1896
1897
if (check_last_modified) {
1898
return this.contents.get(this.notebook_path, {content: false}).then(
1899
function (data) {
1900
var last_modified = new Date(data.last_modified);
1901
if (last_modified > that.last_modified) {
1902
dialog.modal({
1903
notebook: that,
1904
keyboard_manager: that.keyboard_manager,
1905
title: "Notebook changed",
1906
body: "Notebook has changed since we opened it. Overwrite the changed file?",
1907
buttons: {
1908
Cancel: {},
1909
Overwrite: {
1910
class: 'btn-danger',
1911
click: function () {
1912
_save();
1913
}
1914
},
1915
}
1916
});
1917
} else {
1918
return _save();
1919
}
1920
}, function (error) {
1921
// maybe it has been deleted or renamed? Go ahead and save.
1922
return _save();
1923
}
1924
);
1925
} else {
1926
return _save();
1927
}
1928
};
1929
1930
/**
1931
* Success callback for saving a notebook.
1932
*
1933
* @param {integer} start - Time when the save request start
1934
* @param {object} data - JSON representation of a notebook
1935
*/
1936
Notebook.prototype.save_notebook_success = function (start, data) {
1937
this.set_dirty(false);
1938
this.last_modified = new Date(data.last_modified);
1939
if (data.message) {
1940
// save succeeded, but validation failed.
1941
var body = $("<div>");
1942
var title = "Notebook validation failed";
1943
1944
body.append($("<p>").text(
1945
"The save operation succeeded," +
1946
" but the notebook does not appear to be valid." +
1947
" The validation error was:"
1948
)).append($("<div>").addClass("validation-error").append(
1949
$("<pre>").text(data.message)
1950
));
1951
dialog.modal({
1952
notebook: this,
1953
keyboard_manager: this.keyboard_manager,
1954
title: title,
1955
body: body,
1956
buttons : {
1957
OK : {
1958
"class" : "btn-primary"
1959
}
1960
}
1961
});
1962
}
1963
this.events.trigger('notebook_saved.Notebook');
1964
this._update_autosave_interval(start);
1965
if (this._checkpoint_after_save) {
1966
this.create_checkpoint();
1967
this._checkpoint_after_save = false;
1968
}
1969
};
1970
1971
/**
1972
* Update the autosave interval based on the duration of the last save.
1973
*
1974
* @param {integer} timestamp - when the save request started
1975
*/
1976
Notebook.prototype._update_autosave_interval = function (start) {
1977
var duration = (new Date().getTime() - start);
1978
if (this.autosave_interval) {
1979
// new save interval: higher of 10x save duration or parameter (default 30 seconds)
1980
var interval = Math.max(10 * duration, this.minimum_autosave_interval);
1981
// round to 10 seconds, otherwise we will be setting a new interval too often
1982
interval = 10000 * Math.round(interval / 10000);
1983
// set new interval, if it's changed
1984
if (interval !== this.autosave_interval) {
1985
this.set_autosave_interval(interval);
1986
}
1987
}
1988
};
1989
1990
/**
1991
* Explicitly trust the output of this notebook.
1992
*/
1993
Notebook.prototype.trust_notebook = function () {
1994
var body = $("<div>").append($("<p>")
1995
.text("A trusted IPython notebook may execute hidden malicious code ")
1996
.append($("<strong>")
1997
.append(
1998
$("<em>").text("when you open it")
1999
)
2000
).append(".").append(
2001
" Selecting trust will immediately reload this notebook in a trusted state."
2002
).append(
2003
" For more information, see the "
2004
).append($("<a>").attr("href", "http://ipython.org/ipython-doc/2/notebook/security.html")
2005
.text("IPython security documentation")
2006
).append(".")
2007
);
2008
2009
var nb = this;
2010
dialog.modal({
2011
notebook: this,
2012
keyboard_manager: this.keyboard_manager,
2013
title: "Trust this notebook?",
2014
body: body,
2015
2016
buttons: {
2017
Cancel : {},
2018
Trust : {
2019
class : "btn-danger",
2020
click : function () {
2021
var cells = nb.get_cells();
2022
for (var i = 0; i < cells.length; i++) {
2023
var cell = cells[i];
2024
if (cell.cell_type === 'code') {
2025
cell.output_area.trusted = true;
2026
}
2027
}
2028
nb.events.on('notebook_saved.Notebook', function () {
2029
window.location.reload();
2030
});
2031
nb.save_notebook();
2032
}
2033
}
2034
}
2035
});
2036
};
2037
2038
/**
2039
* Make a copy of the current notebook.
2040
*/
2041
Notebook.prototype.copy_notebook = function () {
2042
var that = this;
2043
var base_url = this.base_url;
2044
var w = window.open('', IPython._target);
2045
var parent = utils.url_path_split(this.notebook_path)[0];
2046
this.contents.copy(this.notebook_path, parent).then(
2047
function (data) {
2048
w.location = utils.url_join_encode(
2049
base_url, 'notebooks', data.path
2050
);
2051
},
2052
function(error) {
2053
w.close();
2054
that.events.trigger('notebook_copy_failed', error);
2055
}
2056
);
2057
};
2058
2059
/**
2060
* Ensure a filename has the right extension
2061
* Returns the filename with the appropriate extension, appending if necessary.
2062
*/
2063
Notebook.prototype.ensure_extension = function (name) {
2064
if (!name.match(/\.ipynb$/)) {
2065
name = name + ".ipynb";
2066
}
2067
return name;
2068
};
2069
2070
/**
2071
* Rename the notebook.
2072
* @param {string} new_name
2073
* @return {Promise} promise that resolves when the notebook is renamed.
2074
*/
2075
Notebook.prototype.rename = function (new_name) {
2076
new_name = this.ensure_extension(new_name);
2077
2078
var that = this;
2079
var parent = utils.url_path_split(this.notebook_path)[0];
2080
var new_path = utils.url_path_join(parent, new_name);
2081
return this.contents.rename(this.notebook_path, new_path).then(
2082
function (json) {
2083
that.notebook_name = json.name;
2084
that.notebook_path = json.path;
2085
that.last_modified = new Date(json.last_modified);
2086
that.session.rename_notebook(json.path);
2087
that.events.trigger('notebook_renamed.Notebook', json);
2088
}
2089
);
2090
};
2091
2092
/**
2093
* Delete this notebook
2094
*/
2095
Notebook.prototype.delete = function () {
2096
this.contents.delete(this.notebook_path);
2097
};
2098
2099
/**
2100
* Request a notebook's data from the server.
2101
*
2102
* @param {string} notebook_path - A notebook to load
2103
*/
2104
Notebook.prototype.load_notebook = function (notebook_path) {
2105
var that = this;
2106
this.notebook_path = notebook_path;
2107
this.notebook_name = utils.url_path_split(this.notebook_path)[1];
2108
this.events.trigger('notebook_loading.Notebook');
2109
this.contents.get(notebook_path, {type: 'notebook'}).then(
2110
$.proxy(this.load_notebook_success, this),
2111
$.proxy(this.load_notebook_error, this)
2112
);
2113
};
2114
2115
/**
2116
* Success callback for loading a notebook from the server.
2117
*
2118
* Load notebook data from the JSON response.
2119
*
2120
* @param {object} data JSON representation of a notebook
2121
*/
2122
Notebook.prototype.load_notebook_success = function (data) {
2123
var failed, msg;
2124
try {
2125
this.fromJSON(data);
2126
} catch (e) {
2127
failed = e;
2128
console.log("Notebook failed to load from JSON:", e);
2129
}
2130
if (failed || data.message) {
2131
// *either* fromJSON failed or validation failed
2132
var body = $("<div>");
2133
var title;
2134
if (failed) {
2135
title = "Notebook failed to load";
2136
body.append($("<p>").text(
2137
"The error was: "
2138
)).append($("<div>").addClass("js-error").text(
2139
failed.toString()
2140
)).append($("<p>").text(
2141
"See the error console for details."
2142
));
2143
} else {
2144
title = "Notebook validation failed";
2145
}
2146
2147
if (data.message) {
2148
if (failed) {
2149
msg = "The notebook also failed validation:";
2150
} else {
2151
msg = "An invalid notebook may not function properly." +
2152
" The validation error was:";
2153
}
2154
body.append($("<p>").text(
2155
msg
2156
)).append($("<div>").addClass("validation-error").append(
2157
$("<pre>").text(data.message)
2158
));
2159
}
2160
2161
dialog.modal({
2162
notebook: this,
2163
keyboard_manager: this.keyboard_manager,
2164
title: title,
2165
body: body,
2166
buttons : {
2167
OK : {
2168
"class" : "btn-primary"
2169
}
2170
}
2171
});
2172
}
2173
if (this.ncells() === 0) {
2174
this.insert_cell_below('code');
2175
this.edit_mode(0);
2176
} else {
2177
this.select(0);
2178
this.handle_command_mode(this.get_cell(0));
2179
}
2180
this.set_dirty(false);
2181
this.scroll_to_top();
2182
this.writable = data.writable || false;
2183
this.last_modified = new Date(data.last_modified);
2184
var nbmodel = data.content;
2185
var orig_nbformat = nbmodel.metadata.orig_nbformat;
2186
var orig_nbformat_minor = nbmodel.metadata.orig_nbformat_minor;
2187
if (orig_nbformat !== undefined && nbmodel.nbformat !== orig_nbformat) {
2188
var src;
2189
if (nbmodel.nbformat > orig_nbformat) {
2190
src = " an older notebook format ";
2191
} else {
2192
src = " a newer notebook format ";
2193
}
2194
2195
msg = "This notebook has been converted from" + src +
2196
"(v"+orig_nbformat+") to the current notebook " +
2197
"format (v"+nbmodel.nbformat+"). The next time you save this notebook, the " +
2198
"current notebook format will be used.";
2199
2200
if (nbmodel.nbformat > orig_nbformat) {
2201
msg += " Older versions of IPython may not be able to read the new format.";
2202
} else {
2203
msg += " Some features of the original notebook may not be available.";
2204
}
2205
msg += " To preserve the original version, close the " +
2206
"notebook without saving it.";
2207
dialog.modal({
2208
notebook: this,
2209
keyboard_manager: this.keyboard_manager,
2210
title : "Notebook converted",
2211
body : msg,
2212
buttons : {
2213
OK : {
2214
class : "btn-primary"
2215
}
2216
}
2217
});
2218
} else if (this.nbformat_minor < nbmodel.nbformat_minor) {
2219
this.nbformat_minor = nbmodel.nbformat_minor;
2220
}
2221
2222
if (this.session === null) {
2223
var kernel_name = utils.get_url_param('kernel_name');
2224
if (kernel_name) {
2225
this.kernel_selector.set_kernel(kernel_name);
2226
} else if (this.metadata.kernelspec) {
2227
this.kernel_selector.set_kernel(this.metadata.kernelspec);
2228
} else if (this.metadata.language) {
2229
// compat with IJulia, IHaskell, and other early kernels
2230
// adopters that where setting a language metadata.
2231
this.kernel_selector.set_kernel({
2232
name: "(No name)",
2233
language: this.metadata.language
2234
});
2235
// this should be stored in kspec now, delete it.
2236
// remove once we do not support notebook v3 anymore.
2237
delete this.metadata.language;
2238
} else {
2239
// setting kernel via set_kernel above triggers start_session,
2240
// otherwise start a new session with the server's default kernel
2241
// spec_changed events will fire after kernel is loaded
2242
this.start_session();
2243
}
2244
}
2245
// load our checkpoint list
2246
this.list_checkpoints();
2247
2248
// load toolbar state
2249
if (this.metadata.celltoolbar) {
2250
celltoolbar.CellToolbar.global_show();
2251
celltoolbar.CellToolbar.activate_preset(this.metadata.celltoolbar);
2252
} else {
2253
celltoolbar.CellToolbar.global_hide();
2254
}
2255
2256
if (!this.writable) {
2257
this.set_autosave_interval(0);
2258
this.events.trigger('notebook_read_only.Notebook');
2259
}
2260
2261
// now that we're fully loaded, it is safe to restore save functionality
2262
this._fully_loaded = true;
2263
this.events.trigger('notebook_loaded.Notebook');
2264
};
2265
2266
Notebook.prototype.set_kernelselector = function(k_selector){
2267
this.kernel_selector = k_selector;
2268
};
2269
2270
/**
2271
* Failure callback for loading a notebook from the server.
2272
*
2273
* @param {Error} error
2274
*/
2275
Notebook.prototype.load_notebook_error = function (error) {
2276
this.events.trigger('notebook_load_failed.Notebook', error);
2277
var msg;
2278
if (error.name === utils.XHR_ERROR && error.xhr.status === 500) {
2279
utils.log_ajax_error(error.xhr, error.xhr_status, error.xhr_error);
2280
msg = "An unknown error occurred while loading this notebook. " +
2281
"This version can load notebook formats " +
2282
"v" + this.nbformat + " or earlier. See the server log for details.";
2283
} else {
2284
msg = error.message;
2285
console.warn('Error stack trace while loading notebook was:');
2286
console.warn(error.stack);
2287
}
2288
dialog.modal({
2289
notebook: this,
2290
keyboard_manager: this.keyboard_manager,
2291
title: "Error loading notebook",
2292
body : msg,
2293
buttons : {
2294
"OK": {}
2295
}
2296
});
2297
};
2298
2299
/********************* checkpoint-related ********************/
2300
2301
/**
2302
* Save the notebook then immediately create a checkpoint.
2303
*/
2304
Notebook.prototype.save_checkpoint = function () {
2305
this._checkpoint_after_save = true;
2306
this.save_notebook();
2307
};
2308
2309
/**
2310
* Add a checkpoint for this notebook.
2311
*/
2312
Notebook.prototype.add_checkpoint = function (checkpoint) {
2313
var found = false;
2314
for (var i = 0; i < this.checkpoints.length; i++) {
2315
var existing = this.checkpoints[i];
2316
if (existing.id === checkpoint.id) {
2317
found = true;
2318
this.checkpoints[i] = checkpoint;
2319
break;
2320
}
2321
}
2322
if (!found) {
2323
this.checkpoints.push(checkpoint);
2324
}
2325
this.last_checkpoint = this.checkpoints[this.checkpoints.length - 1];
2326
};
2327
2328
/**
2329
* List checkpoints for this notebook.
2330
*/
2331
Notebook.prototype.list_checkpoints = function () {
2332
var that = this;
2333
this.contents.list_checkpoints(this.notebook_path).then(
2334
$.proxy(this.list_checkpoints_success, this),
2335
function(error) {
2336
that.events.trigger('list_checkpoints_failed.Notebook', error);
2337
}
2338
);
2339
};
2340
2341
/**
2342
* Success callback for listing checkpoints.
2343
*
2344
* @param {object} data - JSON representation of a checkpoint
2345
*/
2346
Notebook.prototype.list_checkpoints_success = function (data) {
2347
this.checkpoints = data;
2348
if (data.length) {
2349
this.last_checkpoint = data[data.length - 1];
2350
} else {
2351
this.last_checkpoint = null;
2352
}
2353
this.events.trigger('checkpoints_listed.Notebook', [data]);
2354
};
2355
2356
/**
2357
* Create a checkpoint of this notebook on the server from the most recent save.
2358
*/
2359
Notebook.prototype.create_checkpoint = function () {
2360
var that = this;
2361
this.contents.create_checkpoint(this.notebook_path).then(
2362
$.proxy(this.create_checkpoint_success, this),
2363
function (error) {
2364
that.events.trigger('checkpoint_failed.Notebook', error);
2365
}
2366
);
2367
};
2368
2369
/**
2370
* Success callback for creating a checkpoint.
2371
*
2372
* @param {object} data - JSON representation of a checkpoint
2373
*/
2374
Notebook.prototype.create_checkpoint_success = function (data) {
2375
this.add_checkpoint(data);
2376
this.events.trigger('checkpoint_created.Notebook', data);
2377
};
2378
2379
/**
2380
* Display the restore checkpoint dialog
2381
* @param {string} checkpoint ID
2382
*/
2383
Notebook.prototype.restore_checkpoint_dialog = function (checkpoint) {
2384
var that = this;
2385
checkpoint = checkpoint || this.last_checkpoint;
2386
if ( ! checkpoint ) {
2387
console.log("restore dialog, but no checkpoint to restore to!");
2388
return;
2389
}
2390
var body = $('<div/>').append(
2391
$('<p/>').addClass("p-space").text(
2392
"Are you sure you want to revert the notebook to " +
2393
"the latest checkpoint?"
2394
).append(
2395
$("<strong/>").text(
2396
" This cannot be undone."
2397
)
2398
)
2399
).append(
2400
$('<p/>').addClass("p-space").text("The checkpoint was last updated at:")
2401
).append(
2402
$('<p/>').addClass("p-space").text(
2403
moment(checkpoint.last_modified).format('LLLL') +
2404
' ('+moment(checkpoint.last_modified).fromNow()+')'// Long form: Tuesday, January 27, 2015 12:15 PM
2405
).css("text-align", "center")
2406
);
2407
2408
dialog.modal({
2409
notebook: this,
2410
keyboard_manager: this.keyboard_manager,
2411
title : "Revert notebook to checkpoint",
2412
body : body,
2413
buttons : {
2414
Revert : {
2415
class : "btn-danger",
2416
click : function () {
2417
that.restore_checkpoint(checkpoint.id);
2418
}
2419
},
2420
Cancel : {}
2421
}
2422
});
2423
};
2424
2425
/**
2426
* Restore the notebook to a checkpoint state.
2427
*
2428
* @param {string} checkpoint ID
2429
*/
2430
Notebook.prototype.restore_checkpoint = function (checkpoint) {
2431
this.events.trigger('notebook_restoring.Notebook', checkpoint);
2432
var that = this;
2433
this.contents.restore_checkpoint(this.notebook_path, checkpoint).then(
2434
$.proxy(this.restore_checkpoint_success, this),
2435
function (error) {
2436
that.events.trigger('checkpoint_restore_failed.Notebook', error);
2437
}
2438
);
2439
};
2440
2441
/**
2442
* Success callback for restoring a notebook to a checkpoint.
2443
*/
2444
Notebook.prototype.restore_checkpoint_success = function () {
2445
this.events.trigger('checkpoint_restored.Notebook');
2446
this.load_notebook(this.notebook_path);
2447
};
2448
2449
/**
2450
* Delete a notebook checkpoint.
2451
*
2452
* @param {string} checkpoint ID
2453
*/
2454
Notebook.prototype.delete_checkpoint = function (checkpoint) {
2455
this.events.trigger('notebook_restoring.Notebook', checkpoint);
2456
var that = this;
2457
this.contents.delete_checkpoint(this.notebook_path, checkpoint).then(
2458
$.proxy(this.delete_checkpoint_success, this),
2459
function (error) {
2460
that.events.trigger('checkpoint_delete_failed.Notebook', error);
2461
}
2462
);
2463
};
2464
2465
/**
2466
* Success callback for deleting a notebook checkpoint.
2467
*/
2468
Notebook.prototype.delete_checkpoint_success = function () {
2469
this.events.trigger('checkpoint_deleted.Notebook');
2470
this.load_notebook(this.notebook_path);
2471
};
2472
2473
2474
// For backwards compatability.
2475
IPython.Notebook = Notebook;
2476
2477
return {'Notebook': Notebook};
2478
});
2479
2480