##// END OF EJS Templates
Notebook id removed from the notebook format.
Brian E. Granger -
Show More
@@ -1,813 +1,812
1
1
2 //============================================================================
2 //============================================================================
3 // Notebook
3 // Notebook
4 //============================================================================
4 //============================================================================
5
5
6 var IPython = (function (IPython) {
6 var IPython = (function (IPython) {
7
7
8 var utils = IPython.utils;
8 var utils = IPython.utils;
9
9
10 var Notebook = function (selector) {
10 var Notebook = function (selector) {
11 this.element = $(selector);
11 this.element = $(selector);
12 this.element.scroll();
12 this.element.scroll();
13 this.element.data("notebook", this);
13 this.element.data("notebook", this);
14 this.next_prompt_number = 1;
14 this.next_prompt_number = 1;
15 this.kernel = null;
15 this.kernel = null;
16 this.dirty = false;
16 this.dirty = false;
17 this.msg_cell_map = {};
17 this.msg_cell_map = {};
18 this.style();
18 this.style();
19 this.create_elements();
19 this.create_elements();
20 this.bind_events();
20 this.bind_events();
21 };
21 };
22
22
23
23
24 Notebook.prototype.style = function () {
24 Notebook.prototype.style = function () {
25 $('div#notebook').addClass('border-box-sizing');
25 $('div#notebook').addClass('border-box-sizing');
26 };
26 };
27
27
28
28
29 Notebook.prototype.create_elements = function () {
29 Notebook.prototype.create_elements = function () {
30 // We add this end_space div to the end of the notebook div to:
30 // We add this end_space div to the end of the notebook div to:
31 // i) provide a margin between the last cell and the end of the notebook
31 // i) provide a margin between the last cell and the end of the notebook
32 // ii) to prevent the div from scrolling up when the last cell is being
32 // ii) to prevent the div from scrolling up when the last cell is being
33 // edited, but is too low on the page, which browsers will do automatically.
33 // edited, but is too low on the page, which browsers will do automatically.
34 this.element.append($('<div class="end_space"></div>').height(50));
34 this.element.append($('<div class="end_space"></div>').height(50));
35 $('div#notebook').addClass('border-box-sizing');
35 $('div#notebook').addClass('border-box-sizing');
36 };
36 };
37
37
38
38
39 Notebook.prototype.bind_events = function () {
39 Notebook.prototype.bind_events = function () {
40 var that = this;
40 var that = this;
41 $(document).keydown(function (event) {
41 $(document).keydown(function (event) {
42 // console.log(event);
42 // console.log(event);
43 if (event.which === 38) {
43 if (event.which === 38) {
44 var cell = that.selected_cell();
44 var cell = that.selected_cell();
45 if (cell.at_top()) {
45 if (cell.at_top()) {
46 event.preventDefault();
46 event.preventDefault();
47 that.select_prev();
47 that.select_prev();
48 };
48 };
49 } else if (event.which === 40) {
49 } else if (event.which === 40) {
50 var cell = that.selected_cell();
50 var cell = that.selected_cell();
51 if (cell.at_bottom()) {
51 if (cell.at_bottom()) {
52 event.preventDefault();
52 event.preventDefault();
53 that.select_next();
53 that.select_next();
54 };
54 };
55 } else if (event.which === 13 && event.shiftKey) {
55 } else if (event.which === 13 && event.shiftKey) {
56 that.execute_selected_cell();
56 that.execute_selected_cell();
57 return false;
57 return false;
58 } else if (event.which === 13 && event.ctrlKey) {
58 } else if (event.which === 13 && event.ctrlKey) {
59 that.execute_selected_cell({terminal:true});
59 that.execute_selected_cell({terminal:true});
60 return false;
60 return false;
61 };
61 };
62 });
62 });
63
63
64 this.element.bind('collapse_pager', function () {
64 this.element.bind('collapse_pager', function () {
65 var app_height = $('div#main_app').height(); // content height
65 var app_height = $('div#main_app').height(); // content height
66 var splitter_height = $('div#pager_splitter').outerHeight(true);
66 var splitter_height = $('div#pager_splitter').outerHeight(true);
67 var new_height = app_height - splitter_height;
67 var new_height = app_height - splitter_height;
68 that.element.animate({height : new_height + 'px'}, 'fast');
68 that.element.animate({height : new_height + 'px'}, 'fast');
69 });
69 });
70
70
71 this.element.bind('expand_pager', function () {
71 this.element.bind('expand_pager', function () {
72 var app_height = $('div#main_app').height(); // content height
72 var app_height = $('div#main_app').height(); // content height
73 var splitter_height = $('div#pager_splitter').outerHeight(true);
73 var splitter_height = $('div#pager_splitter').outerHeight(true);
74 var pager_height = $('div#pager').outerHeight(true);
74 var pager_height = $('div#pager').outerHeight(true);
75 var new_height = app_height - pager_height - splitter_height;
75 var new_height = app_height - pager_height - splitter_height;
76 that.element.animate({height : new_height + 'px'}, 'fast');
76 that.element.animate({height : new_height + 'px'}, 'fast');
77 });
77 });
78
78
79 this.element.bind('collapse_left_panel', function () {
79 this.element.bind('collapse_left_panel', function () {
80 var splitter_width = $('div#left_panel_splitter').outerWidth(true);
80 var splitter_width = $('div#left_panel_splitter').outerWidth(true);
81 var new_margin = splitter_width;
81 var new_margin = splitter_width;
82 $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast');
82 $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast');
83 });
83 });
84
84
85 this.element.bind('expand_left_panel', function () {
85 this.element.bind('expand_left_panel', function () {
86 var splitter_width = $('div#left_panel_splitter').outerWidth(true);
86 var splitter_width = $('div#left_panel_splitter').outerWidth(true);
87 var left_panel_width = IPython.left_panel.width;
87 var left_panel_width = IPython.left_panel.width;
88 var new_margin = splitter_width + left_panel_width;
88 var new_margin = splitter_width + left_panel_width;
89 $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast');
89 $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast');
90 });
90 });
91
91
92 $(window).bind('beforeunload', function () {
92 $(window).bind('beforeunload', function () {
93 var kill_kernel = $('#kill_kernel').prop('checked');
93 var kill_kernel = $('#kill_kernel').prop('checked');
94 if (kill_kernel) {
94 if (kill_kernel) {
95 that.kernel.kill();
95 that.kernel.kill();
96 }
96 }
97 if (that.dirty) {
97 if (that.dirty) {
98 return "You have unsaved changes that will be lost if you leave this page.";
98 return "You have unsaved changes that will be lost if you leave this page.";
99 };
99 };
100 });
100 });
101 };
101 };
102
102
103
103
104 Notebook.prototype.scroll_to_bottom = function () {
104 Notebook.prototype.scroll_to_bottom = function () {
105 this.element.animate({scrollTop:this.element.get(0).scrollHeight}, 0);
105 this.element.animate({scrollTop:this.element.get(0).scrollHeight}, 0);
106 };
106 };
107
107
108
108
109 Notebook.prototype.scroll_to_top = function () {
109 Notebook.prototype.scroll_to_top = function () {
110 this.element.animate({scrollTop:0}, 0);
110 this.element.animate({scrollTop:0}, 0);
111 };
111 };
112
112
113
113
114 // Cell indexing, retrieval, etc.
114 // Cell indexing, retrieval, etc.
115
115
116
116
117 Notebook.prototype.cell_elements = function () {
117 Notebook.prototype.cell_elements = function () {
118 return this.element.children("div.cell");
118 return this.element.children("div.cell");
119 }
119 }
120
120
121
121
122 Notebook.prototype.ncells = function (cell) {
122 Notebook.prototype.ncells = function (cell) {
123 return this.cell_elements().length;
123 return this.cell_elements().length;
124 }
124 }
125
125
126
126
127 // TODO: we are often calling cells as cells()[i], which we should optimize
127 // TODO: we are often calling cells as cells()[i], which we should optimize
128 // to cells(i) or a new method.
128 // to cells(i) or a new method.
129 Notebook.prototype.cells = function () {
129 Notebook.prototype.cells = function () {
130 return this.cell_elements().toArray().map(function (e) {
130 return this.cell_elements().toArray().map(function (e) {
131 return $(e).data("cell");
131 return $(e).data("cell");
132 });
132 });
133 }
133 }
134
134
135
135
136 Notebook.prototype.find_cell_index = function (cell) {
136 Notebook.prototype.find_cell_index = function (cell) {
137 var result = null;
137 var result = null;
138 this.cell_elements().filter(function (index) {
138 this.cell_elements().filter(function (index) {
139 if ($(this).data("cell") === cell) {
139 if ($(this).data("cell") === cell) {
140 result = index;
140 result = index;
141 };
141 };
142 });
142 });
143 return result;
143 return result;
144 };
144 };
145
145
146
146
147 Notebook.prototype.index_or_selected = function (index) {
147 Notebook.prototype.index_or_selected = function (index) {
148 return index || this.selected_index() || 0;
148 return index || this.selected_index() || 0;
149 }
149 }
150
150
151
151
152 Notebook.prototype.select = function (index) {
152 Notebook.prototype.select = function (index) {
153 if (index !== undefined && index >= 0 && index < this.ncells()) {
153 if (index !== undefined && index >= 0 && index < this.ncells()) {
154 if (this.selected_index() !== null) {
154 if (this.selected_index() !== null) {
155 this.selected_cell().unselect();
155 this.selected_cell().unselect();
156 };
156 };
157 this.cells()[index].select();
157 this.cells()[index].select();
158 if (index === (this.ncells()-1)) {
158 if (index === (this.ncells()-1)) {
159 this.scroll_to_bottom();
159 this.scroll_to_bottom();
160 };
160 };
161 };
161 };
162 return this;
162 return this;
163 };
163 };
164
164
165
165
166 Notebook.prototype.select_next = function () {
166 Notebook.prototype.select_next = function () {
167 var index = this.selected_index();
167 var index = this.selected_index();
168 if (index !== null && index >= 0 && (index+1) < this.ncells()) {
168 if (index !== null && index >= 0 && (index+1) < this.ncells()) {
169 this.select(index+1);
169 this.select(index+1);
170 };
170 };
171 return this;
171 return this;
172 };
172 };
173
173
174
174
175 Notebook.prototype.select_prev = function () {
175 Notebook.prototype.select_prev = function () {
176 var index = this.selected_index();
176 var index = this.selected_index();
177 if (index !== null && index >= 0 && (index-1) < this.ncells()) {
177 if (index !== null && index >= 0 && (index-1) < this.ncells()) {
178 this.select(index-1);
178 this.select(index-1);
179 };
179 };
180 return this;
180 return this;
181 };
181 };
182
182
183
183
184 Notebook.prototype.selected_index = function () {
184 Notebook.prototype.selected_index = function () {
185 var result = null;
185 var result = null;
186 this.cell_elements().filter(function (index) {
186 this.cell_elements().filter(function (index) {
187 if ($(this).data("cell").selected === true) {
187 if ($(this).data("cell").selected === true) {
188 result = index;
188 result = index;
189 };
189 };
190 });
190 });
191 return result;
191 return result;
192 };
192 };
193
193
194
194
195 Notebook.prototype.cell_for_msg = function (msg_id) {
195 Notebook.prototype.cell_for_msg = function (msg_id) {
196 var cell_id = this.msg_cell_map[msg_id];
196 var cell_id = this.msg_cell_map[msg_id];
197 var result = null;
197 var result = null;
198 this.cell_elements().filter(function (index) {
198 this.cell_elements().filter(function (index) {
199 cell = $(this).data("cell");
199 cell = $(this).data("cell");
200 if (cell.cell_id === cell_id) {
200 if (cell.cell_id === cell_id) {
201 result = cell;
201 result = cell;
202 };
202 };
203 });
203 });
204 return result;
204 return result;
205 };
205 };
206
206
207
207
208 Notebook.prototype.selected_cell = function () {
208 Notebook.prototype.selected_cell = function () {
209 return this.cell_elements().eq(this.selected_index()).data("cell");
209 return this.cell_elements().eq(this.selected_index()).data("cell");
210 }
210 }
211
211
212
212
213 // Cell insertion, deletion and moving.
213 // Cell insertion, deletion and moving.
214
214
215
215
216 Notebook.prototype.delete_cell = function (index) {
216 Notebook.prototype.delete_cell = function (index) {
217 var i = index || this.selected_index();
217 var i = index || this.selected_index();
218 if (i !== null && i >= 0 && i < this.ncells()) {
218 if (i !== null && i >= 0 && i < this.ncells()) {
219 this.cell_elements().eq(i).remove();
219 this.cell_elements().eq(i).remove();
220 if (i === (this.ncells())) {
220 if (i === (this.ncells())) {
221 this.select(i-1);
221 this.select(i-1);
222 } else {
222 } else {
223 this.select(i);
223 this.select(i);
224 };
224 };
225 };
225 };
226 this.dirty = true;
226 this.dirty = true;
227 return this;
227 return this;
228 };
228 };
229
229
230
230
231 Notebook.prototype.append_cell = function (cell) {
231 Notebook.prototype.append_cell = function (cell) {
232 this.element.find('div.end_space').before(cell.element);
232 this.element.find('div.end_space').before(cell.element);
233 this.dirty = true;
233 this.dirty = true;
234 return this;
234 return this;
235 };
235 };
236
236
237
237
238 Notebook.prototype.insert_cell_after = function (cell, index) {
238 Notebook.prototype.insert_cell_after = function (cell, index) {
239 var ncells = this.ncells();
239 var ncells = this.ncells();
240 if (ncells === 0) {
240 if (ncells === 0) {
241 this.append_cell(cell);
241 this.append_cell(cell);
242 return this;
242 return this;
243 };
243 };
244 if (index >= 0 && index < ncells) {
244 if (index >= 0 && index < ncells) {
245 this.cell_elements().eq(index).after(cell.element);
245 this.cell_elements().eq(index).after(cell.element);
246 };
246 };
247 this.dirty = true;
247 this.dirty = true;
248 return this
248 return this
249 };
249 };
250
250
251
251
252 Notebook.prototype.insert_cell_before = function (cell, index) {
252 Notebook.prototype.insert_cell_before = function (cell, index) {
253 var ncells = this.ncells();
253 var ncells = this.ncells();
254 if (ncells === 0) {
254 if (ncells === 0) {
255 this.append_cell(cell);
255 this.append_cell(cell);
256 return this;
256 return this;
257 };
257 };
258 if (index >= 0 && index < ncells) {
258 if (index >= 0 && index < ncells) {
259 this.cell_elements().eq(index).before(cell.element);
259 this.cell_elements().eq(index).before(cell.element);
260 };
260 };
261 this.dirty = true;
261 this.dirty = true;
262 return this;
262 return this;
263 };
263 };
264
264
265
265
266 Notebook.prototype.move_cell_up = function (index) {
266 Notebook.prototype.move_cell_up = function (index) {
267 var i = index || this.selected_index();
267 var i = index || this.selected_index();
268 if (i !== null && i < this.ncells() && i > 0) {
268 if (i !== null && i < this.ncells() && i > 0) {
269 var pivot = this.cell_elements().eq(i-1);
269 var pivot = this.cell_elements().eq(i-1);
270 var tomove = this.cell_elements().eq(i);
270 var tomove = this.cell_elements().eq(i);
271 if (pivot !== null && tomove !== null) {
271 if (pivot !== null && tomove !== null) {
272 tomove.detach();
272 tomove.detach();
273 pivot.before(tomove);
273 pivot.before(tomove);
274 this.select(i-1);
274 this.select(i-1);
275 };
275 };
276 };
276 };
277 this.dirty = true;
277 this.dirty = true;
278 return this;
278 return this;
279 }
279 }
280
280
281
281
282 Notebook.prototype.move_cell_down = function (index) {
282 Notebook.prototype.move_cell_down = function (index) {
283 var i = index || this.selected_index();
283 var i = index || this.selected_index();
284 if (i !== null && i < (this.ncells()-1) && i >= 0) {
284 if (i !== null && i < (this.ncells()-1) && i >= 0) {
285 var pivot = this.cell_elements().eq(i+1)
285 var pivot = this.cell_elements().eq(i+1)
286 var tomove = this.cell_elements().eq(i)
286 var tomove = this.cell_elements().eq(i)
287 if (pivot !== null && tomove !== null) {
287 if (pivot !== null && tomove !== null) {
288 tomove.detach();
288 tomove.detach();
289 pivot.after(tomove);
289 pivot.after(tomove);
290 this.select(i+1);
290 this.select(i+1);
291 };
291 };
292 };
292 };
293 this.dirty = true;
293 this.dirty = true;
294 return this;
294 return this;
295 }
295 }
296
296
297
297
298 Notebook.prototype.sort_cells = function () {
298 Notebook.prototype.sort_cells = function () {
299 var ncells = this.ncells();
299 var ncells = this.ncells();
300 var sindex = this.selected_index();
300 var sindex = this.selected_index();
301 var swapped;
301 var swapped;
302 do {
302 do {
303 swapped = false
303 swapped = false
304 for (var i=1; i<ncells; i++) {
304 for (var i=1; i<ncells; i++) {
305 current = this.cell_elements().eq(i).data("cell");
305 current = this.cell_elements().eq(i).data("cell");
306 previous = this.cell_elements().eq(i-1).data("cell");
306 previous = this.cell_elements().eq(i-1).data("cell");
307 if (previous.input_prompt_number > current.input_prompt_number) {
307 if (previous.input_prompt_number > current.input_prompt_number) {
308 this.move_cell_up(i);
308 this.move_cell_up(i);
309 swapped = true;
309 swapped = true;
310 };
310 };
311 };
311 };
312 } while (swapped);
312 } while (swapped);
313 this.select(sindex);
313 this.select(sindex);
314 return this;
314 return this;
315 };
315 };
316
316
317
317
318 Notebook.prototype.insert_code_cell_before = function (index) {
318 Notebook.prototype.insert_code_cell_before = function (index) {
319 // TODO: Bounds check for i
319 // TODO: Bounds check for i
320 var i = this.index_or_selected(index);
320 var i = this.index_or_selected(index);
321 var cell = new IPython.CodeCell(this);
321 var cell = new IPython.CodeCell(this);
322 cell.set_input_prompt();
322 cell.set_input_prompt();
323 this.insert_cell_before(cell, i);
323 this.insert_cell_before(cell, i);
324 this.select(this.find_cell_index(cell));
324 this.select(this.find_cell_index(cell));
325 return cell;
325 return cell;
326 }
326 }
327
327
328
328
329 Notebook.prototype.insert_code_cell_after = function (index) {
329 Notebook.prototype.insert_code_cell_after = function (index) {
330 // TODO: Bounds check for i
330 // TODO: Bounds check for i
331 var i = this.index_or_selected(index);
331 var i = this.index_or_selected(index);
332 var cell = new IPython.CodeCell(this);
332 var cell = new IPython.CodeCell(this);
333 cell.set_input_prompt();
333 cell.set_input_prompt();
334 this.insert_cell_after(cell, i);
334 this.insert_cell_after(cell, i);
335 this.select(this.find_cell_index(cell));
335 this.select(this.find_cell_index(cell));
336 return cell;
336 return cell;
337 }
337 }
338
338
339
339
340 Notebook.prototype.insert_html_cell_before = function (index) {
340 Notebook.prototype.insert_html_cell_before = function (index) {
341 // TODO: Bounds check for i
341 // TODO: Bounds check for i
342 var i = this.index_or_selected(index);
342 var i = this.index_or_selected(index);
343 var cell = new IPython.HTMLCell(this);
343 var cell = new IPython.HTMLCell(this);
344 cell.config_mathjax();
344 cell.config_mathjax();
345 this.insert_cell_before(cell, i);
345 this.insert_cell_before(cell, i);
346 this.select(this.find_cell_index(cell));
346 this.select(this.find_cell_index(cell));
347 return cell;
347 return cell;
348 }
348 }
349
349
350
350
351 Notebook.prototype.insert_html_cell_after = function (index) {
351 Notebook.prototype.insert_html_cell_after = function (index) {
352 // TODO: Bounds check for i
352 // TODO: Bounds check for i
353 var i = this.index_or_selected(index);
353 var i = this.index_or_selected(index);
354 var cell = new IPython.HTMLCell(this);
354 var cell = new IPython.HTMLCell(this);
355 cell.config_mathjax();
355 cell.config_mathjax();
356 this.insert_cell_after(cell, i);
356 this.insert_cell_after(cell, i);
357 this.select(this.find_cell_index(cell));
357 this.select(this.find_cell_index(cell));
358 return cell;
358 return cell;
359 }
359 }
360
360
361
361
362 Notebook.prototype.insert_markdown_cell_before = function (index) {
362 Notebook.prototype.insert_markdown_cell_before = function (index) {
363 // TODO: Bounds check for i
363 // TODO: Bounds check for i
364 var i = this.index_or_selected(index);
364 var i = this.index_or_selected(index);
365 var cell = new IPython.MarkdownCell(this);
365 var cell = new IPython.MarkdownCell(this);
366 cell.config_mathjax();
366 cell.config_mathjax();
367 this.insert_cell_before(cell, i);
367 this.insert_cell_before(cell, i);
368 this.select(this.find_cell_index(cell));
368 this.select(this.find_cell_index(cell));
369 return cell;
369 return cell;
370 }
370 }
371
371
372
372
373 Notebook.prototype.insert_markdown_cell_after = function (index) {
373 Notebook.prototype.insert_markdown_cell_after = function (index) {
374 // TODO: Bounds check for i
374 // TODO: Bounds check for i
375 var i = this.index_or_selected(index);
375 var i = this.index_or_selected(index);
376 var cell = new IPython.MarkdownCell(this);
376 var cell = new IPython.MarkdownCell(this);
377 cell.config_mathjax();
377 cell.config_mathjax();
378 this.insert_cell_after(cell, i);
378 this.insert_cell_after(cell, i);
379 this.select(this.find_cell_index(cell));
379 this.select(this.find_cell_index(cell));
380 return cell;
380 return cell;
381 }
381 }
382
382
383
383
384 Notebook.prototype.to_code = function (index) {
384 Notebook.prototype.to_code = function (index) {
385 // TODO: Bounds check for i
385 // TODO: Bounds check for i
386 var i = this.index_or_selected(index);
386 var i = this.index_or_selected(index);
387 var source_element = this.cell_elements().eq(i);
387 var source_element = this.cell_elements().eq(i);
388 var source_cell = source_element.data("cell");
388 var source_cell = source_element.data("cell");
389 if (source_cell instanceof IPython.HTMLCell ||
389 if (source_cell instanceof IPython.HTMLCell ||
390 source_cell instanceof IPython.MarkdownCell) {
390 source_cell instanceof IPython.MarkdownCell) {
391 this.insert_code_cell_after(i);
391 this.insert_code_cell_after(i);
392 var target_cell = this.cells()[i+1];
392 var target_cell = this.cells()[i+1];
393 target_cell.set_code(source_cell.get_source());
393 target_cell.set_code(source_cell.get_source());
394 source_element.remove();
394 source_element.remove();
395 };
395 };
396 this.dirty = true;
396 this.dirty = true;
397 };
397 };
398
398
399
399
400 Notebook.prototype.to_markdown = function (index) {
400 Notebook.prototype.to_markdown = function (index) {
401 // TODO: Bounds check for i
401 // TODO: Bounds check for i
402 var i = this.index_or_selected(index);
402 var i = this.index_or_selected(index);
403 var source_element = this.cell_elements().eq(i);
403 var source_element = this.cell_elements().eq(i);
404 var source_cell = source_element.data("cell");
404 var source_cell = source_element.data("cell");
405 var target_cell = null;
405 var target_cell = null;
406 if (source_cell instanceof IPython.CodeCell) {
406 if (source_cell instanceof IPython.CodeCell) {
407 this.insert_markdown_cell_after(i);
407 this.insert_markdown_cell_after(i);
408 var target_cell = this.cells()[i+1];
408 var target_cell = this.cells()[i+1];
409 var text = source_cell.get_code();
409 var text = source_cell.get_code();
410 } else if (source_cell instanceof IPython.HTMLCell) {
410 } else if (source_cell instanceof IPython.HTMLCell) {
411 this.insert_markdown_cell_after(i);
411 this.insert_markdown_cell_after(i);
412 var target_cell = this.cells()[i+1];
412 var target_cell = this.cells()[i+1];
413 var text = source_cell.get_source();
413 var text = source_cell.get_source();
414 if (text === source_cell.placeholder) {
414 if (text === source_cell.placeholder) {
415 text = target_cell.placeholder;
415 text = target_cell.placeholder;
416 }
416 }
417 }
417 }
418 if (target_cell !== null) {
418 if (target_cell !== null) {
419 if (text === "") {text = target_cell.placeholder;};
419 if (text === "") {text = target_cell.placeholder;};
420 target_cell.set_source(text);
420 target_cell.set_source(text);
421 source_element.remove();
421 source_element.remove();
422 target_cell.edit();
422 target_cell.edit();
423 }
423 }
424 this.dirty = true;
424 this.dirty = true;
425 };
425 };
426
426
427
427
428 Notebook.prototype.to_html = function (index) {
428 Notebook.prototype.to_html = function (index) {
429 // TODO: Bounds check for i
429 // TODO: Bounds check for i
430 var i = this.index_or_selected(index);
430 var i = this.index_or_selected(index);
431 var source_element = this.cell_elements().eq(i);
431 var source_element = this.cell_elements().eq(i);
432 var source_cell = source_element.data("cell");
432 var source_cell = source_element.data("cell");
433 var target_cell = null;
433 var target_cell = null;
434 if (source_cell instanceof IPython.CodeCell) {
434 if (source_cell instanceof IPython.CodeCell) {
435 this.insert_html_cell_after(i);
435 this.insert_html_cell_after(i);
436 var target_cell = this.cells()[i+1];
436 var target_cell = this.cells()[i+1];
437 var text = source_cell.get_code();
437 var text = source_cell.get_code();
438 } else if (source_cell instanceof IPython.MarkdownCell) {
438 } else if (source_cell instanceof IPython.MarkdownCell) {
439 this.insert_html_cell_after(i);
439 this.insert_html_cell_after(i);
440 var target_cell = this.cells()[i+1];
440 var target_cell = this.cells()[i+1];
441 var text = source_cell.get_source();
441 var text = source_cell.get_source();
442 if (text === source_cell.placeholder) {
442 if (text === source_cell.placeholder) {
443 text = target_cell.placeholder;
443 text = target_cell.placeholder;
444 }
444 }
445 }
445 }
446 if (target_cell !== null) {
446 if (target_cell !== null) {
447 if (text === "") {text = target_cell.placeholder;};
447 if (text === "") {text = target_cell.placeholder;};
448 target_cell.set_source(text);
448 target_cell.set_source(text);
449 source_element.remove();
449 source_element.remove();
450 target_cell.edit();
450 target_cell.edit();
451 }
451 }
452 this.dirty = true;
452 this.dirty = true;
453 };
453 };
454
454
455
455
456 // Cell collapsing and output clearing
456 // Cell collapsing and output clearing
457
457
458 Notebook.prototype.collapse = function (index) {
458 Notebook.prototype.collapse = function (index) {
459 var i = this.index_or_selected(index);
459 var i = this.index_or_selected(index);
460 this.cells()[i].collapse();
460 this.cells()[i].collapse();
461 this.dirty = true;
461 this.dirty = true;
462 };
462 };
463
463
464
464
465 Notebook.prototype.expand = function (index) {
465 Notebook.prototype.expand = function (index) {
466 var i = this.index_or_selected(index);
466 var i = this.index_or_selected(index);
467 this.cells()[i].expand();
467 this.cells()[i].expand();
468 this.dirty = true;
468 this.dirty = true;
469 };
469 };
470
470
471
471
472 Notebook.prototype.set_autoindent = function (state) {
472 Notebook.prototype.set_autoindent = function (state) {
473 var cells = this.cells();
473 var cells = this.cells();
474 len = cells.length;
474 len = cells.length;
475 for (var i=0; i<len; i++) {
475 for (var i=0; i<len; i++) {
476 cells[i].set_autoindent(state)
476 cells[i].set_autoindent(state)
477 };
477 };
478 };
478 };
479
479
480
480
481 Notebook.prototype.clear_all_output = function () {
481 Notebook.prototype.clear_all_output = function () {
482 var ncells = this.ncells();
482 var ncells = this.ncells();
483 var cells = this.cells();
483 var cells = this.cells();
484 for (var i=0; i<ncells; i++) {
484 for (var i=0; i<ncells; i++) {
485 if (cells[i] instanceof IPython.CodeCell) {
485 if (cells[i] instanceof IPython.CodeCell) {
486 cells[i].clear_output();
486 cells[i].clear_output();
487 }
487 }
488 };
488 };
489 this.dirty = true;
489 this.dirty = true;
490 };
490 };
491
491
492
492
493 // Kernel related things
493 // Kernel related things
494
494
495 Notebook.prototype.start_kernel = function () {
495 Notebook.prototype.start_kernel = function () {
496 this.kernel = new IPython.Kernel();
496 this.kernel = new IPython.Kernel();
497 var notebook_id = IPython.save_widget.get_notebook_id();
497 var notebook_id = IPython.save_widget.get_notebook_id();
498 this.kernel.start(notebook_id, $.proxy(this.kernel_started, this));
498 this.kernel.start(notebook_id, $.proxy(this.kernel_started, this));
499 };
499 };
500
500
501
501
502 Notebook.prototype.restart_kernel = function () {
502 Notebook.prototype.restart_kernel = function () {
503 var notebook_id = IPython.save_widget.get_notebook_id();
503 var notebook_id = IPython.save_widget.get_notebook_id();
504 this.kernel.restart($.proxy(this.kernel_started, this));
504 this.kernel.restart($.proxy(this.kernel_started, this));
505 };
505 };
506
506
507
507
508 Notebook.prototype.kernel_started = function () {
508 Notebook.prototype.kernel_started = function () {
509 console.log("Kernel started: ", this.kernel.kernel_id);
509 console.log("Kernel started: ", this.kernel.kernel_id);
510 this.kernel.shell_channel.onmessage = $.proxy(this.handle_shell_reply,this);
510 this.kernel.shell_channel.onmessage = $.proxy(this.handle_shell_reply,this);
511 this.kernel.iopub_channel.onmessage = $.proxy(this.handle_iopub_reply,this);
511 this.kernel.iopub_channel.onmessage = $.proxy(this.handle_iopub_reply,this);
512 };
512 };
513
513
514
514
515 Notebook.prototype.handle_shell_reply = function (e) {
515 Notebook.prototype.handle_shell_reply = function (e) {
516 reply = $.parseJSON(e.data);
516 reply = $.parseJSON(e.data);
517 var header = reply.header;
517 var header = reply.header;
518 var content = reply.content;
518 var content = reply.content;
519 var msg_type = header.msg_type;
519 var msg_type = header.msg_type;
520 // console.log(reply);
520 // console.log(reply);
521 var cell = this.cell_for_msg(reply.parent_header.msg_id);
521 var cell = this.cell_for_msg(reply.parent_header.msg_id);
522 if (msg_type === "execute_reply") {
522 if (msg_type === "execute_reply") {
523 cell.set_input_prompt(content.execution_count);
523 cell.set_input_prompt(content.execution_count);
524 this.dirty = true;
524 this.dirty = true;
525 } else if (msg_type === "complete_reply") {
525 } else if (msg_type === "complete_reply") {
526 cell.finish_completing(content.matched_text, content.matches);
526 cell.finish_completing(content.matched_text, content.matches);
527 };
527 };
528 var payload = content.payload || [];
528 var payload = content.payload || [];
529 this.handle_payload(cell, payload);
529 this.handle_payload(cell, payload);
530 };
530 };
531
531
532
532
533 Notebook.prototype.handle_payload = function (cell, payload) {
533 Notebook.prototype.handle_payload = function (cell, payload) {
534 var l = payload.length;
534 var l = payload.length;
535 for (var i=0; i<l; i++) {
535 for (var i=0; i<l; i++) {
536 if (payload[i].source === 'IPython.zmq.page.page') {
536 if (payload[i].source === 'IPython.zmq.page.page') {
537 IPython.pager.clear();
537 IPython.pager.clear();
538 IPython.pager.expand();
538 IPython.pager.expand();
539 IPython.pager.append_text(payload[i].text);
539 IPython.pager.append_text(payload[i].text);
540 } else if (payload[i].source === 'IPython.zmq.zmqshell.ZMQInteractiveShell.set_next_input') {
540 } else if (payload[i].source === 'IPython.zmq.zmqshell.ZMQInteractiveShell.set_next_input') {
541 var index = this.find_cell_index(cell);
541 var index = this.find_cell_index(cell);
542 var new_cell = this.insert_code_cell_after(index);
542 var new_cell = this.insert_code_cell_after(index);
543 new_cell.set_code(payload[i].text);
543 new_cell.set_code(payload[i].text);
544 this.dirty = true;
544 this.dirty = true;
545 }
545 }
546 };
546 };
547 };
547 };
548
548
549
549
550 Notebook.prototype.handle_iopub_reply = function (e) {
550 Notebook.prototype.handle_iopub_reply = function (e) {
551 reply = $.parseJSON(e.data);
551 reply = $.parseJSON(e.data);
552 var content = reply.content;
552 var content = reply.content;
553 // console.log(reply);
553 // console.log(reply);
554 var msg_type = reply.header.msg_type;
554 var msg_type = reply.header.msg_type;
555 var cell = this.cell_for_msg(reply.parent_header.msg_id);
555 var cell = this.cell_for_msg(reply.parent_header.msg_id);
556 var output_types = ['stream','display_data','pyout','pyerr'];
556 var output_types = ['stream','display_data','pyout','pyerr'];
557 if (output_types.indexOf(msg_type) >= 0) {
557 if (output_types.indexOf(msg_type) >= 0) {
558 this.handle_output(cell, msg_type, content);
558 this.handle_output(cell, msg_type, content);
559 } else if (msg_type === 'status') {
559 } else if (msg_type === 'status') {
560 if (content.execution_state === 'busy') {
560 if (content.execution_state === 'busy') {
561 IPython.kernel_status_widget.status_busy();
561 IPython.kernel_status_widget.status_busy();
562 } else if (content.execution_state === 'idle') {
562 } else if (content.execution_state === 'idle') {
563 IPython.kernel_status_widget.status_idle();
563 IPython.kernel_status_widget.status_idle();
564 } else if (content.execution_state === 'dead') {
564 } else if (content.execution_state === 'dead') {
565 this.handle_status_dead();
565 this.handle_status_dead();
566 };
566 };
567 }
567 }
568 };
568 };
569
569
570
570
571 Notebook.prototype.handle_status_dead = function () {
571 Notebook.prototype.handle_status_dead = function () {
572 var that = this;
572 var that = this;
573 this.kernel.stop_channels();
573 this.kernel.stop_channels();
574 var dialog = $('<div/>');
574 var dialog = $('<div/>');
575 dialog.html('The kernel has died, would you like to restart it? If you do not restart the kernel, you will be able to save the notebook, but running code will not work until the notebook is reopened.');
575 dialog.html('The kernel has died, would you like to restart it? If you do not restart the kernel, you will be able to save the notebook, but running code will not work until the notebook is reopened.');
576 $(document).append(dialog);
576 $(document).append(dialog);
577 dialog.dialog({
577 dialog.dialog({
578 resizable: false,
578 resizable: false,
579 modal: true,
579 modal: true,
580 title: "Dead kernel",
580 title: "Dead kernel",
581 buttons : {
581 buttons : {
582 "Yes": function () {
582 "Yes": function () {
583 that.start_kernel();
583 that.start_kernel();
584 $(this).dialog('close');
584 $(this).dialog('close');
585 },
585 },
586 "No": function () {
586 "No": function () {
587 $(this).dialog('close');
587 $(this).dialog('close');
588 }
588 }
589 }
589 }
590 });
590 });
591 };
591 };
592
592
593
593
594 Notebook.prototype.handle_output = function (cell, msg_type, content) {
594 Notebook.prototype.handle_output = function (cell, msg_type, content) {
595 var json = {};
595 var json = {};
596 json.output_type = msg_type;
596 json.output_type = msg_type;
597 if (msg_type === "stream") {
597 if (msg_type === "stream") {
598 json.text = utils.fixConsole(content.data + '\n');
598 json.text = utils.fixConsole(content.data + '\n');
599 } else if (msg_type === "display_data") {
599 } else if (msg_type === "display_data") {
600 json = this.convert_mime_types(json, content.data);
600 json = this.convert_mime_types(json, content.data);
601 } else if (msg_type === "pyout") {
601 } else if (msg_type === "pyout") {
602 json.prompt_number = content.execution_count;
602 json.prompt_number = content.execution_count;
603 json = this.convert_mime_types(json, content.data);
603 json = this.convert_mime_types(json, content.data);
604 } else if (msg_type === "pyerr") {
604 } else if (msg_type === "pyerr") {
605 json.ename = content.ename;
605 json.ename = content.ename;
606 json.evalue = content.evalue;
606 json.evalue = content.evalue;
607 var traceback = [];
607 var traceback = [];
608 for (var i=0; i<content.traceback.length; i++) {
608 for (var i=0; i<content.traceback.length; i++) {
609 traceback.push(utils.fixConsole(content.traceback[i]));
609 traceback.push(utils.fixConsole(content.traceback[i]));
610 }
610 }
611 json.traceback = traceback;
611 json.traceback = traceback;
612 };
612 };
613 cell.append_output(json);
613 cell.append_output(json);
614 this.dirty = true;
614 this.dirty = true;
615 };
615 };
616
616
617
617
618 Notebook.prototype.convert_mime_types = function (json, data) {
618 Notebook.prototype.convert_mime_types = function (json, data) {
619 if (data['text/plain'] !== undefined) {
619 if (data['text/plain'] !== undefined) {
620 json.text = utils.fixConsole(data['text/plain']);
620 json.text = utils.fixConsole(data['text/plain']);
621 };
621 };
622 if (data['text/html'] !== undefined) {
622 if (data['text/html'] !== undefined) {
623 json.html = data['text/html'];
623 json.html = data['text/html'];
624 };
624 };
625 if (data['image/svg+xml'] !== undefined) {
625 if (data['image/svg+xml'] !== undefined) {
626 json.svg = data['image/svg+xml'];
626 json.svg = data['image/svg+xml'];
627 };
627 };
628 if (data['image/png'] !== undefined) {
628 if (data['image/png'] !== undefined) {
629 json.png = data['image/png'];
629 json.png = data['image/png'];
630 };
630 };
631 if (data['image/jpeg'] !== undefined) {
631 if (data['image/jpeg'] !== undefined) {
632 json.jpeg = data['image/jpeg'];
632 json.jpeg = data['image/jpeg'];
633 };
633 };
634 if (data['text/latex'] !== undefined) {
634 if (data['text/latex'] !== undefined) {
635 json.latex = data['text/latex'];
635 json.latex = data['text/latex'];
636 };
636 };
637 if (data['application/json'] !== undefined) {
637 if (data['application/json'] !== undefined) {
638 json.json = data['application/json'];
638 json.json = data['application/json'];
639 };
639 };
640 if (data['application/javascript'] !== undefined) {
640 if (data['application/javascript'] !== undefined) {
641 json.javascript = data['application/javascript'];
641 json.javascript = data['application/javascript'];
642 }
642 }
643 return json;
643 return json;
644 };
644 };
645
645
646
646
647 Notebook.prototype.execute_selected_cell = function (options) {
647 Notebook.prototype.execute_selected_cell = function (options) {
648 // add_new: should a new cell be added if we are at the end of the nb
648 // add_new: should a new cell be added if we are at the end of the nb
649 // terminal: execute in terminal mode, which stays in the current cell
649 // terminal: execute in terminal mode, which stays in the current cell
650 default_options = {terminal: false, add_new: true}
650 default_options = {terminal: false, add_new: true}
651 $.extend(default_options, options)
651 $.extend(default_options, options)
652 var that = this;
652 var that = this;
653 var cell = that.selected_cell();
653 var cell = that.selected_cell();
654 var cell_index = that.find_cell_index(cell);
654 var cell_index = that.find_cell_index(cell);
655 if (cell instanceof IPython.CodeCell) {
655 if (cell instanceof IPython.CodeCell) {
656 cell.clear_output();
656 cell.clear_output();
657 var code = cell.get_code();
657 var code = cell.get_code();
658 var msg_id = that.kernel.execute(cell.get_code());
658 var msg_id = that.kernel.execute(cell.get_code());
659 that.msg_cell_map[msg_id] = cell.cell_id;
659 that.msg_cell_map[msg_id] = cell.cell_id;
660 } else if (cell instanceof IPython.HTMLCell) {
660 } else if (cell instanceof IPython.HTMLCell) {
661 cell.render();
661 cell.render();
662 }
662 }
663 if (default_options.terminal) {
663 if (default_options.terminal) {
664 cell.clear_input();
664 cell.clear_input();
665 } else {
665 } else {
666 if ((cell_index === (that.ncells()-1)) && default_options.add_new) {
666 if ((cell_index === (that.ncells()-1)) && default_options.add_new) {
667 that.insert_code_cell_after();
667 that.insert_code_cell_after();
668 // If we are adding a new cell at the end, scroll down to show it.
668 // If we are adding a new cell at the end, scroll down to show it.
669 that.scroll_to_bottom();
669 that.scroll_to_bottom();
670 } else {
670 } else {
671 that.select(cell_index+1);
671 that.select(cell_index+1);
672 };
672 };
673 };
673 };
674 this.dirty = true;
674 this.dirty = true;
675 };
675 };
676
676
677
677
678 Notebook.prototype.execute_all_cells = function () {
678 Notebook.prototype.execute_all_cells = function () {
679 var ncells = this.ncells();
679 var ncells = this.ncells();
680 for (var i=0; i<ncells; i++) {
680 for (var i=0; i<ncells; i++) {
681 this.select(i);
681 this.select(i);
682 this.execute_selected_cell({add_new:false});
682 this.execute_selected_cell({add_new:false});
683 };
683 };
684 this.scroll_to_bottom();
684 this.scroll_to_bottom();
685 };
685 };
686
686
687
687
688 Notebook.prototype.complete_cell = function (cell, line, cursor_pos) {
688 Notebook.prototype.complete_cell = function (cell, line, cursor_pos) {
689 var msg_id = this.kernel.complete(line, cursor_pos);
689 var msg_id = this.kernel.complete(line, cursor_pos);
690 this.msg_cell_map[msg_id] = cell.cell_id;
690 this.msg_cell_map[msg_id] = cell.cell_id;
691 };
691 };
692
692
693 // Persistance and loading
693 // Persistance and loading
694
694
695
695
696 Notebook.prototype.fromJSON = function (data) {
696 Notebook.prototype.fromJSON = function (data) {
697 var ncells = this.ncells();
697 var ncells = this.ncells();
698 for (var i=0; i<ncells; i++) {
698 for (var i=0; i<ncells; i++) {
699 // Always delete cell 0 as they get renumbered as they are deleted.
699 // Always delete cell 0 as they get renumbered as they are deleted.
700 this.delete_cell(0);
700 this.delete_cell(0);
701 };
701 };
702 // Only handle 1 worksheet for now.
702 // Only handle 1 worksheet for now.
703 var worksheet = data.worksheets[0];
703 var worksheet = data.worksheets[0];
704 if (worksheet !== undefined) {
704 if (worksheet !== undefined) {
705 var new_cells = worksheet.cells;
705 var new_cells = worksheet.cells;
706 ncells = new_cells.length;
706 ncells = new_cells.length;
707 var cell_data = null;
707 var cell_data = null;
708 var new_cell = null;
708 var new_cell = null;
709 for (var i=0; i<ncells; i++) {
709 for (var i=0; i<ncells; i++) {
710 cell_data = new_cells[i];
710 cell_data = new_cells[i];
711 if (cell_data.cell_type == 'code') {
711 if (cell_data.cell_type == 'code') {
712 new_cell = this.insert_code_cell_after();
712 new_cell = this.insert_code_cell_after();
713 new_cell.fromJSON(cell_data);
713 new_cell.fromJSON(cell_data);
714 } else if (cell_data.cell_type === 'html') {
714 } else if (cell_data.cell_type === 'html') {
715 new_cell = this.insert_html_cell_after();
715 new_cell = this.insert_html_cell_after();
716 new_cell.fromJSON(cell_data);
716 new_cell.fromJSON(cell_data);
717 } else if (cell_data.cell_type === 'markdown') {
717 } else if (cell_data.cell_type === 'markdown') {
718 new_cell = this.insert_markdown_cell_after();
718 new_cell = this.insert_markdown_cell_after();
719 new_cell.fromJSON(cell_data);
719 new_cell.fromJSON(cell_data);
720 };
720 };
721 };
721 };
722 };
722 };
723 };
723 };
724
724
725
725
726 Notebook.prototype.toJSON = function () {
726 Notebook.prototype.toJSON = function () {
727 var cells = this.cells();
727 var cells = this.cells();
728 var ncells = cells.length;
728 var ncells = cells.length;
729 cell_array = new Array(ncells);
729 cell_array = new Array(ncells);
730 for (var i=0; i<ncells; i++) {
730 for (var i=0; i<ncells; i++) {
731 cell_array[i] = cells[i].toJSON();
731 cell_array[i] = cells[i].toJSON();
732 };
732 };
733 data = {
733 data = {
734 // Only handle 1 worksheet for now.
734 // Only handle 1 worksheet for now.
735 worksheets : [{cells:cell_array}]
735 worksheets : [{cells:cell_array}]
736 }
736 }
737 return data
737 return data
738 };
738 };
739
739
740 Notebook.prototype.save_notebook = function () {
740 Notebook.prototype.save_notebook = function () {
741 if (IPython.save_widget.test_notebook_name()) {
741 if (IPython.save_widget.test_notebook_name()) {
742 var notebook_id = IPython.save_widget.get_notebook_id();
742 var notebook_id = IPython.save_widget.get_notebook_id();
743 var nbname = IPython.save_widget.get_notebook_name();
743 var nbname = IPython.save_widget.get_notebook_name();
744 // We may want to move the name/id/nbformat logic inside toJSON?
744 // We may want to move the name/id/nbformat logic inside toJSON?
745 var data = this.toJSON();
745 var data = this.toJSON();
746 data.name = nbname;
746 data.name = nbname;
747 data.nbformat = 2;
747 data.nbformat = 2;
748 data.id = notebook_id
749 // We do the call with settings so we can set cache to false.
748 // We do the call with settings so we can set cache to false.
750 var settings = {
749 var settings = {
751 processData : false,
750 processData : false,
752 cache : false,
751 cache : false,
753 type : "PUT",
752 type : "PUT",
754 data : JSON.stringify(data),
753 data : JSON.stringify(data),
755 headers : {'Content-Type': 'application/json'},
754 headers : {'Content-Type': 'application/json'},
756 success : $.proxy(this.notebook_saved,this)
755 success : $.proxy(this.notebook_saved,this)
757 };
756 };
758 IPython.save_widget.status_saving();
757 IPython.save_widget.status_saving();
759 $.ajax("/notebooks/" + notebook_id, settings);
758 $.ajax("/notebooks/" + notebook_id, settings);
760 };
759 };
761 };
760 };
762
761
763
762
764 Notebook.prototype.notebook_saved = function (data, status, xhr) {
763 Notebook.prototype.notebook_saved = function (data, status, xhr) {
765 this.dirty = false;
764 this.dirty = false;
766 setTimeout($.proxy(IPython.save_widget.status_save,IPython.save_widget),500);
765 setTimeout($.proxy(IPython.save_widget.status_save,IPython.save_widget),500);
767 }
766 }
768
767
769
768
770 Notebook.prototype.load_notebook = function (callback) {
769 Notebook.prototype.load_notebook = function (callback) {
771 var that = this;
770 var that = this;
772 var notebook_id = IPython.save_widget.get_notebook_id();
771 var notebook_id = IPython.save_widget.get_notebook_id();
773 // We do the call with settings so we can set cache to false.
772 // We do the call with settings so we can set cache to false.
774 var settings = {
773 var settings = {
775 processData : false,
774 processData : false,
776 cache : false,
775 cache : false,
777 type : "GET",
776 type : "GET",
778 dataType : "json",
777 dataType : "json",
779 success : function (data, status, xhr) {
778 success : function (data, status, xhr) {
780 that.notebook_loaded(data, status, xhr);
779 that.notebook_loaded(data, status, xhr);
781 if (callback !== undefined) {
780 if (callback !== undefined) {
782 callback();
781 callback();
783 };
782 };
784 }
783 }
785 };
784 };
786 IPython.save_widget.status_loading();
785 IPython.save_widget.status_loading();
787 $.ajax("/notebooks/" + notebook_id, settings);
786 $.ajax("/notebooks/" + notebook_id, settings);
788 }
787 }
789
788
790
789
791 Notebook.prototype.notebook_loaded = function (data, status, xhr) {
790 Notebook.prototype.notebook_loaded = function (data, status, xhr) {
792 this.fromJSON(data);
791 this.fromJSON(data);
793 if (this.ncells() === 0) {
792 if (this.ncells() === 0) {
794 this.insert_code_cell_after();
793 this.insert_code_cell_after();
795 };
794 };
796 IPython.save_widget.status_save();
795 IPython.save_widget.status_save();
797 IPython.save_widget.set_notebook_name(data.name);
796 IPython.save_widget.set_notebook_name(data.name);
798 this.start_kernel();
797 this.start_kernel();
799 this.dirty = false;
798 this.dirty = false;
800 // fromJSON always selects the last cell inserted. We need to wait
799 // fromJSON always selects the last cell inserted. We need to wait
801 // until that is done before scrolling to the top.
800 // until that is done before scrolling to the top.
802 setTimeout(function () {
801 setTimeout(function () {
803 IPython.notebook.select(0);
802 IPython.notebook.select(0);
804 IPython.notebook.scroll_to_top();
803 IPython.notebook.scroll_to_top();
805 }, 50);
804 }, 50);
806 };
805 };
807
806
808 IPython.Notebook = Notebook;
807 IPython.Notebook = Notebook;
809
808
810 return IPython;
809 return IPython;
811
810
812 }(IPython));
811 }(IPython));
813
812
@@ -1,136 +1,132
1 """The basic dict based notebook format."""
1 """The basic dict based notebook format."""
2
2
3 import pprint
3 import pprint
4 import uuid
4 import uuid
5
5
6 from IPython.utils.ipstruct import Struct
6 from IPython.utils.ipstruct import Struct
7
7
8
8
9 class NotebookNode(Struct):
9 class NotebookNode(Struct):
10 pass
10 pass
11
11
12
12
13 def from_dict(d):
13 def from_dict(d):
14 if isinstance(d, dict):
14 if isinstance(d, dict):
15 newd = NotebookNode()
15 newd = NotebookNode()
16 for k,v in d.items():
16 for k,v in d.items():
17 newd[k] = from_dict(v)
17 newd[k] = from_dict(v)
18 return newd
18 return newd
19 elif isinstance(d, (tuple, list)):
19 elif isinstance(d, (tuple, list)):
20 return [from_dict(i) for i in d]
20 return [from_dict(i) for i in d]
21 else:
21 else:
22 return d
22 return d
23
23
24
24
25 def new_output(output_type=None, output_text=None, output_png=None,
25 def new_output(output_type=None, output_text=None, output_png=None,
26 output_html=None, output_svg=None, output_latex=None, output_json=None,
26 output_html=None, output_svg=None, output_latex=None, output_json=None,
27 output_javascript=None, output_jpeg=None, prompt_number=None,
27 output_javascript=None, output_jpeg=None, prompt_number=None,
28 etype=None, evalue=None, traceback=None):
28 etype=None, evalue=None, traceback=None):
29 """Create a new code cell with input and output"""
29 """Create a new code cell with input and output"""
30 output = NotebookNode()
30 output = NotebookNode()
31 if output_type is not None:
31 if output_type is not None:
32 output.output_type = unicode(output_type)
32 output.output_type = unicode(output_type)
33
33
34 if output_type != 'pyerr':
34 if output_type != 'pyerr':
35 if output_text is not None:
35 if output_text is not None:
36 output.text = unicode(output_text)
36 output.text = unicode(output_text)
37 if output_png is not None:
37 if output_png is not None:
38 output.png = bytes(output_png)
38 output.png = bytes(output_png)
39 if output_jpeg is not None:
39 if output_jpeg is not None:
40 output.jpeg = bytes(output_jpeg)
40 output.jpeg = bytes(output_jpeg)
41 if output_html is not None:
41 if output_html is not None:
42 output.html = unicode(output_html)
42 output.html = unicode(output_html)
43 if output_svg is not None:
43 if output_svg is not None:
44 output.svg = unicode(output_svg)
44 output.svg = unicode(output_svg)
45 if output_latex is not None:
45 if output_latex is not None:
46 output.latex = unicode(output_latex)
46 output.latex = unicode(output_latex)
47 if output_json is not None:
47 if output_json is not None:
48 output.json = unicode(output_json)
48 output.json = unicode(output_json)
49 if output_javascript is not None:
49 if output_javascript is not None:
50 output.javascript = unicode(output_javascript)
50 output.javascript = unicode(output_javascript)
51
51
52 if output_type == u'pyout':
52 if output_type == u'pyout':
53 if prompt_number is not None:
53 if prompt_number is not None:
54 output.prompt_number = int(prompt_number)
54 output.prompt_number = int(prompt_number)
55
55
56 if output_type == u'pyerr':
56 if output_type == u'pyerr':
57 if etype is not None:
57 if etype is not None:
58 output.etype = unicode(etype)
58 output.etype = unicode(etype)
59 if evalue is not None:
59 if evalue is not None:
60 output.evalue = unicode(evalue)
60 output.evalue = unicode(evalue)
61 if traceback is not None:
61 if traceback is not None:
62 output.traceback = [unicode(frame) for frame in list(traceback)]
62 output.traceback = [unicode(frame) for frame in list(traceback)]
63
63
64 return output
64 return output
65
65
66
66
67 def new_code_cell(input=None, prompt_number=None, outputs=None,
67 def new_code_cell(input=None, prompt_number=None, outputs=None,
68 language=u'python', collapsed=False):
68 language=u'python', collapsed=False):
69 """Create a new code cell with input and output"""
69 """Create a new code cell with input and output"""
70 cell = NotebookNode()
70 cell = NotebookNode()
71 cell.cell_type = u'code'
71 cell.cell_type = u'code'
72 if language is not None:
72 if language is not None:
73 cell.language = unicode(language)
73 cell.language = unicode(language)
74 if input is not None:
74 if input is not None:
75 cell.input = unicode(input)
75 cell.input = unicode(input)
76 if prompt_number is not None:
76 if prompt_number is not None:
77 cell.prompt_number = int(prompt_number)
77 cell.prompt_number = int(prompt_number)
78 if outputs is None:
78 if outputs is None:
79 cell.outputs = []
79 cell.outputs = []
80 else:
80 else:
81 cell.outputs = outputs
81 cell.outputs = outputs
82 if collapsed is not None:
82 if collapsed is not None:
83 cell.collapsed = bool(collapsed)
83 cell.collapsed = bool(collapsed)
84
84
85 return cell
85 return cell
86
86
87 def new_text_cell(cell_type, source=None, rendered=None):
87 def new_text_cell(cell_type, source=None, rendered=None):
88 """Create a new text cell."""
88 """Create a new text cell."""
89 cell = NotebookNode()
89 cell = NotebookNode()
90 if source is not None:
90 if source is not None:
91 cell.source = unicode(source)
91 cell.source = unicode(source)
92 if rendered is not None:
92 if rendered is not None:
93 cell.rendered = unicode(rendered)
93 cell.rendered = unicode(rendered)
94 cell.cell_type = cell_type
94 cell.cell_type = cell_type
95 return cell
95 return cell
96
96
97
97
98 def new_worksheet(name=None, cells=None):
98 def new_worksheet(name=None, cells=None):
99 """Create a worksheet by name with with a list of cells."""
99 """Create a worksheet by name with with a list of cells."""
100 ws = NotebookNode()
100 ws = NotebookNode()
101 if name is not None:
101 if name is not None:
102 ws.name = unicode(name)
102 ws.name = unicode(name)
103 if cells is None:
103 if cells is None:
104 ws.cells = []
104 ws.cells = []
105 else:
105 else:
106 ws.cells = list(cells)
106 ws.cells = list(cells)
107 return ws
107 return ws
108
108
109
109
110 def new_notebook(name=None, id=None, worksheets=None, author=None, email=None,
110 def new_notebook(name=None, worksheets=None, author=None, email=None,
111 created=None, saved=None, license=None):
111 created=None, saved=None, license=None):
112 """Create a notebook by name, id and a list of worksheets."""
112 """Create a notebook by name, id and a list of worksheets."""
113 nb = NotebookNode()
113 nb = NotebookNode()
114 nb.nbformat = 2
114 nb.nbformat = 2
115 if name is not None:
115 if name is not None:
116 nb.name = unicode(name)
116 nb.name = unicode(name)
117 if id is None:
118 nb.id = unicode(uuid.uuid4())
119 else:
120 nb.id = unicode(id)
121 if worksheets is None:
117 if worksheets is None:
122 nb.worksheets = []
118 nb.worksheets = []
123 else:
119 else:
124 nb.worksheets = list(worksheets)
120 nb.worksheets = list(worksheets)
125 if author is not None:
121 if author is not None:
126 nb.author = unicode(author)
122 nb.author = unicode(author)
127 if email is not None:
123 if email is not None:
128 nb.email = unicode(email)
124 nb.email = unicode(email)
129 if created is not None:
125 if created is not None:
130 nb.created = unicode(created)
126 nb.created = unicode(created)
131 if saved is not None:
127 if saved is not None:
132 nb.saved = unicode(saved)
128 nb.saved = unicode(saved)
133 if license is not None:
129 if license is not None:
134 nb.license = unicode(license)
130 nb.license = unicode(license)
135 return nb
131 return nb
136
132
@@ -1,228 +1,226
1 """Read and write notebook files as XML."""
1 """Read and write notebook files as XML."""
2
2
3 from base64 import encodestring, decodestring
3 from base64 import encodestring, decodestring
4 from xml.etree import ElementTree as ET
4 from xml.etree import ElementTree as ET
5
5
6 from .rwbase import NotebookReader, NotebookWriter
6 from .rwbase import NotebookReader, NotebookWriter
7 from .nbbase import (
7 from .nbbase import (
8 new_code_cell, new_text_cell, new_worksheet, new_notebook, new_output
8 new_code_cell, new_text_cell, new_worksheet, new_notebook, new_output
9 )
9 )
10
10
11 def indent(elem, level=0):
11 def indent(elem, level=0):
12 i = "\n" + level*" "
12 i = "\n" + level*" "
13 if len(elem):
13 if len(elem):
14 if not elem.text or not elem.text.strip():
14 if not elem.text or not elem.text.strip():
15 elem.text = i + " "
15 elem.text = i + " "
16 if not elem.tail or not elem.tail.strip():
16 if not elem.tail or not elem.tail.strip():
17 elem.tail = i
17 elem.tail = i
18 for elem in elem:
18 for elem in elem:
19 indent(elem, level+1)
19 indent(elem, level+1)
20 if not elem.tail or not elem.tail.strip():
20 if not elem.tail or not elem.tail.strip():
21 elem.tail = i
21 elem.tail = i
22 else:
22 else:
23 if level and (not elem.tail or not elem.tail.strip()):
23 if level and (not elem.tail or not elem.tail.strip()):
24 elem.tail = i
24 elem.tail = i
25
25
26
26
27 def _get_text(e, tag):
27 def _get_text(e, tag):
28 sub_e = e.find(tag)
28 sub_e = e.find(tag)
29 if sub_e is None:
29 if sub_e is None:
30 return None
30 return None
31 else:
31 else:
32 return sub_e.text
32 return sub_e.text
33
33
34
34
35 def _set_text(nbnode, attr, parent, tag):
35 def _set_text(nbnode, attr, parent, tag):
36 if attr in nbnode:
36 if attr in nbnode:
37 e = ET.SubElement(parent, tag)
37 e = ET.SubElement(parent, tag)
38 e.text = nbnode[attr]
38 e.text = nbnode[attr]
39
39
40
40
41 def _get_int(e, tag):
41 def _get_int(e, tag):
42 sub_e = e.find(tag)
42 sub_e = e.find(tag)
43 if sub_e is None:
43 if sub_e is None:
44 return None
44 return None
45 else:
45 else:
46 return int(sub_e.text)
46 return int(sub_e.text)
47
47
48
48
49 def _set_int(nbnode, attr, parent, tag):
49 def _set_int(nbnode, attr, parent, tag):
50 if attr in nbnode:
50 if attr in nbnode:
51 e = ET.SubElement(parent, tag)
51 e = ET.SubElement(parent, tag)
52 e.text = unicode(nbnode[attr])
52 e.text = unicode(nbnode[attr])
53
53
54
54
55 def _get_bool(e, tag):
55 def _get_bool(e, tag):
56 sub_e = e.find(tag)
56 sub_e = e.find(tag)
57 if sub_e is None:
57 if sub_e is None:
58 return None
58 return None
59 else:
59 else:
60 return bool(int(sub_e.text))
60 return bool(int(sub_e.text))
61
61
62
62
63 def _set_bool(nbnode, attr, parent, tag):
63 def _set_bool(nbnode, attr, parent, tag):
64 if attr in nbnode:
64 if attr in nbnode:
65 e = ET.SubElement(parent, tag)
65 e = ET.SubElement(parent, tag)
66 if nbnode[attr]:
66 if nbnode[attr]:
67 e.text = u'1'
67 e.text = u'1'
68 else:
68 else:
69 e.text = u'0'
69 e.text = u'0'
70
70
71
71
72 def _get_binary(e, tag):
72 def _get_binary(e, tag):
73 sub_e = e.find(tag)
73 sub_e = e.find(tag)
74 if sub_e is None:
74 if sub_e is None:
75 return None
75 return None
76 else:
76 else:
77 return decodestring(sub_e.text)
77 return decodestring(sub_e.text)
78
78
79
79
80 def _set_binary(nbnode, attr, parent, tag):
80 def _set_binary(nbnode, attr, parent, tag):
81 if attr in nbnode:
81 if attr in nbnode:
82 e = ET.SubElement(parent, tag)
82 e = ET.SubElement(parent, tag)
83 e.text = encodestring(nbnode[attr])
83 e.text = encodestring(nbnode[attr])
84
84
85
85
86 class XMLReader(NotebookReader):
86 class XMLReader(NotebookReader):
87
87
88 def reads(self, s, **kwargs):
88 def reads(self, s, **kwargs):
89 root = ET.fromstring(s)
89 root = ET.fromstring(s)
90 return self.to_notebook(root, **kwargs)
90 return self.to_notebook(root, **kwargs)
91
91
92 def to_notebook(self, root, **kwargs):
92 def to_notebook(self, root, **kwargs):
93 nbname = _get_text(root,u'name')
93 nbname = _get_text(root,u'name')
94 nbid = _get_text(root,u'id')
95 nbauthor = _get_text(root,u'author')
94 nbauthor = _get_text(root,u'author')
96 nbemail = _get_text(root,u'email')
95 nbemail = _get_text(root,u'email')
97 nblicense = _get_text(root,u'license')
96 nblicense = _get_text(root,u'license')
98 nbcreated = _get_text(root,u'created')
97 nbcreated = _get_text(root,u'created')
99 nbsaved = _get_text(root,u'saved')
98 nbsaved = _get_text(root,u'saved')
100
99
101 worksheets = []
100 worksheets = []
102 for ws_e in root.find(u'worksheets').getiterator(u'worksheet'):
101 for ws_e in root.find(u'worksheets').getiterator(u'worksheet'):
103 wsname = _get_text(ws_e,u'name')
102 wsname = _get_text(ws_e,u'name')
104 cells = []
103 cells = []
105 for cell_e in ws_e.find(u'cells').getiterator():
104 for cell_e in ws_e.find(u'cells').getiterator():
106 if cell_e.tag == u'codecell':
105 if cell_e.tag == u'codecell':
107 input = _get_text(cell_e,u'input')
106 input = _get_text(cell_e,u'input')
108 prompt_number = _get_int(cell_e,u'prompt_number')
107 prompt_number = _get_int(cell_e,u'prompt_number')
109 collapsed = _get_bool(cell_e,u'collapsed')
108 collapsed = _get_bool(cell_e,u'collapsed')
110 language = _get_text(cell_e,u'language')
109 language = _get_text(cell_e,u'language')
111 outputs = []
110 outputs = []
112 for output_e in cell_e.find(u'outputs').getiterator(u'output'):
111 for output_e in cell_e.find(u'outputs').getiterator(u'output'):
113 output_type = _get_text(output_e,u'output_type')
112 output_type = _get_text(output_e,u'output_type')
114 output_text = _get_text(output_e,u'text')
113 output_text = _get_text(output_e,u'text')
115 output_png = _get_binary(output_e,u'png')
114 output_png = _get_binary(output_e,u'png')
116 output_jpeg = _get_binary(output_e,u'jpeg')
115 output_jpeg = _get_binary(output_e,u'jpeg')
117 output_svg = _get_text(output_e,u'svg')
116 output_svg = _get_text(output_e,u'svg')
118 output_html = _get_text(output_e,u'html')
117 output_html = _get_text(output_e,u'html')
119 output_latex = _get_text(output_e,u'latex')
118 output_latex = _get_text(output_e,u'latex')
120 output_json = _get_text(output_e,u'json')
119 output_json = _get_text(output_e,u'json')
121 output_javascript = _get_text(output_e,u'javascript')
120 output_javascript = _get_text(output_e,u'javascript')
122
121
123 out_prompt_number = _get_int(output_e,u'prompt_number')
122 out_prompt_number = _get_int(output_e,u'prompt_number')
124 etype = _get_text(output_e,u'etype')
123 etype = _get_text(output_e,u'etype')
125 evalue = _get_text(output_e,u'evalue')
124 evalue = _get_text(output_e,u'evalue')
126 traceback = []
125 traceback = []
127 traceback_e = output_e.find(u'traceback')
126 traceback_e = output_e.find(u'traceback')
128 if traceback_e is not None:
127 if traceback_e is not None:
129 for frame_e in traceback_e.getiterator(u'frame'):
128 for frame_e in traceback_e.getiterator(u'frame'):
130 traceback.append(frame_e.text)
129 traceback.append(frame_e.text)
131 if len(traceback) == 0:
130 if len(traceback) == 0:
132 traceback = None
131 traceback = None
133 output = new_output(output_type=output_type,output_png=output_png,
132 output = new_output(output_type=output_type,output_png=output_png,
134 output_text=output_text, output_svg=output_svg,
133 output_text=output_text, output_svg=output_svg,
135 output_html=output_html, output_latex=output_latex,
134 output_html=output_html, output_latex=output_latex,
136 output_json=output_json, output_javascript=output_javascript,
135 output_json=output_json, output_javascript=output_javascript,
137 output_jpeg=output_jpeg, prompt_number=out_prompt_number,
136 output_jpeg=output_jpeg, prompt_number=out_prompt_number,
138 etype=etype, evalue=evalue, traceback=traceback
137 etype=etype, evalue=evalue, traceback=traceback
139 )
138 )
140 outputs.append(output)
139 outputs.append(output)
141 cc = new_code_cell(input=input,prompt_number=prompt_number,
140 cc = new_code_cell(input=input,prompt_number=prompt_number,
142 language=language,outputs=outputs,collapsed=collapsed)
141 language=language,outputs=outputs,collapsed=collapsed)
143 cells.append(cc)
142 cells.append(cc)
144 if cell_e.tag == u'htmlcell':
143 if cell_e.tag == u'htmlcell':
145 source = _get_text(cell_e,u'source')
144 source = _get_text(cell_e,u'source')
146 rendered = _get_text(cell_e,u'rendered')
145 rendered = _get_text(cell_e,u'rendered')
147 cells.append(new_text_cell(u'html', source=source, rendered=rendered))
146 cells.append(new_text_cell(u'html', source=source, rendered=rendered))
148 if cell_e.tag == u'markdowncell':
147 if cell_e.tag == u'markdowncell':
149 source = _get_text(cell_e,u'source')
148 source = _get_text(cell_e,u'source')
150 rendered = _get_text(cell_e,u'rendered')
149 rendered = _get_text(cell_e,u'rendered')
151 cells.append(new_text_cell(u'markdown', source=source, rendered=rendered))
150 cells.append(new_text_cell(u'markdown', source=source, rendered=rendered))
152 ws = new_worksheet(name=wsname,cells=cells)
151 ws = new_worksheet(name=wsname,cells=cells)
153 worksheets.append(ws)
152 worksheets.append(ws)
154
153
155 nb = new_notebook(name=nbname,id=nbid,worksheets=worksheets,author=nbauthor,
154 nb = new_notebook(name=nbname,worksheets=worksheets,author=nbauthor,
156 email=nbemail,license=nblicense,saved=nbsaved,created=nbcreated)
155 email=nbemail,license=nblicense,saved=nbsaved,created=nbcreated)
157 return nb
156 return nb
158
157
159
158
160 class XMLWriter(NotebookWriter):
159 class XMLWriter(NotebookWriter):
161
160
162 def writes(self, nb, **kwargs):
161 def writes(self, nb, **kwargs):
163 nb_e = ET.Element(u'notebook')
162 nb_e = ET.Element(u'notebook')
164 _set_text(nb,u'name',nb_e,u'name')
163 _set_text(nb,u'name',nb_e,u'name')
165 _set_text(nb,u'id',nb_e,u'id')
166 _set_text(nb,u'author',nb_e,u'author')
164 _set_text(nb,u'author',nb_e,u'author')
167 _set_text(nb,u'email',nb_e,u'email')
165 _set_text(nb,u'email',nb_e,u'email')
168 _set_text(nb,u'license',nb_e,u'license')
166 _set_text(nb,u'license',nb_e,u'license')
169 _set_text(nb,u'created',nb_e,u'created')
167 _set_text(nb,u'created',nb_e,u'created')
170 _set_text(nb,u'saved',nb_e,u'saved')
168 _set_text(nb,u'saved',nb_e,u'saved')
171 _set_int(nb,u'nbformat',nb_e,u'nbformat')
169 _set_int(nb,u'nbformat',nb_e,u'nbformat')
172 wss_e = ET.SubElement(nb_e,u'worksheets')
170 wss_e = ET.SubElement(nb_e,u'worksheets')
173 for ws in nb.worksheets:
171 for ws in nb.worksheets:
174 ws_e = ET.SubElement(wss_e, u'worksheet')
172 ws_e = ET.SubElement(wss_e, u'worksheet')
175 _set_text(ws,u'name',ws_e,u'name')
173 _set_text(ws,u'name',ws_e,u'name')
176 cells_e = ET.SubElement(ws_e,u'cells')
174 cells_e = ET.SubElement(ws_e,u'cells')
177 for cell in ws.cells:
175 for cell in ws.cells:
178 cell_type = cell.cell_type
176 cell_type = cell.cell_type
179 if cell_type == u'code':
177 if cell_type == u'code':
180 cell_e = ET.SubElement(cells_e, u'codecell')
178 cell_e = ET.SubElement(cells_e, u'codecell')
181 _set_text(cell,u'input',cell_e,u'input')
179 _set_text(cell,u'input',cell_e,u'input')
182 _set_text(cell,u'language',cell_e,u'language')
180 _set_text(cell,u'language',cell_e,u'language')
183 _set_int(cell,u'prompt_number',cell_e,u'prompt_number')
181 _set_int(cell,u'prompt_number',cell_e,u'prompt_number')
184 _set_bool(cell,u'collapsed',cell_e,u'collapsed')
182 _set_bool(cell,u'collapsed',cell_e,u'collapsed')
185 outputs_e = ET.SubElement(cell_e, u'outputs')
183 outputs_e = ET.SubElement(cell_e, u'outputs')
186 for output in cell.outputs:
184 for output in cell.outputs:
187 output_e = ET.SubElement(outputs_e, u'output')
185 output_e = ET.SubElement(outputs_e, u'output')
188 _set_text(output,u'output_type',output_e,u'output_type')
186 _set_text(output,u'output_type',output_e,u'output_type')
189 _set_text(output,u'text',output_e,u'text')
187 _set_text(output,u'text',output_e,u'text')
190 _set_binary(output,u'png',output_e,u'png')
188 _set_binary(output,u'png',output_e,u'png')
191 _set_binary(output,u'jpeg',output_e,u'jpeg')
189 _set_binary(output,u'jpeg',output_e,u'jpeg')
192 _set_text(output,u'html',output_e,u'html')
190 _set_text(output,u'html',output_e,u'html')
193 _set_text(output,u'svg',output_e,u'svg')
191 _set_text(output,u'svg',output_e,u'svg')
194 _set_text(output,u'latex',output_e,u'latex')
192 _set_text(output,u'latex',output_e,u'latex')
195 _set_text(output,u'json',output_e,u'json')
193 _set_text(output,u'json',output_e,u'json')
196 _set_text(output,u'javascript',output_e,u'javascript')
194 _set_text(output,u'javascript',output_e,u'javascript')
197 _set_int(output,u'prompt_number',output_e,u'prompt_number')
195 _set_int(output,u'prompt_number',output_e,u'prompt_number')
198 _set_text(output,u'etype',output_e,u'etype')
196 _set_text(output,u'etype',output_e,u'etype')
199 _set_text(output,u'evalue',output_e,u'evalue')
197 _set_text(output,u'evalue',output_e,u'evalue')
200 if u'traceback' in output:
198 if u'traceback' in output:
201 tb_e = ET.SubElement(output_e, u'traceback')
199 tb_e = ET.SubElement(output_e, u'traceback')
202 for frame in output.traceback:
200 for frame in output.traceback:
203 frame_e = ET.SubElement(tb_e, u'frame')
201 frame_e = ET.SubElement(tb_e, u'frame')
204 frame_e.text = frame
202 frame_e.text = frame
205 elif cell_type == u'html':
203 elif cell_type == u'html':
206 cell_e = ET.SubElement(cells_e, u'htmlcell')
204 cell_e = ET.SubElement(cells_e, u'htmlcell')
207 _set_text(cell,u'source',cell_e,u'source')
205 _set_text(cell,u'source',cell_e,u'source')
208 _set_text(cell,u'rendered',cell_e,u'rendered')
206 _set_text(cell,u'rendered',cell_e,u'rendered')
209 elif cell_type == u'markdown':
207 elif cell_type == u'markdown':
210 cell_e = ET.SubElement(cells_e, u'markdowncell')
208 cell_e = ET.SubElement(cells_e, u'markdowncell')
211 _set_text(cell,u'source',cell_e,u'source')
209 _set_text(cell,u'source',cell_e,u'source')
212 _set_text(cell,u'rendered',cell_e,u'rendered')
210 _set_text(cell,u'rendered',cell_e,u'rendered')
213
211
214 indent(nb_e)
212 indent(nb_e)
215 txt = ET.tostring(nb_e, encoding="utf-8")
213 txt = ET.tostring(nb_e, encoding="utf-8")
216 txt = '<?xml version="1.0" encoding="utf-8"?>\n' + txt
214 txt = '<?xml version="1.0" encoding="utf-8"?>\n' + txt
217 return txt
215 return txt
218
216
219
217
220 _reader = XMLReader()
218 _reader = XMLReader()
221 _writer = XMLWriter()
219 _writer = XMLWriter()
222
220
223 reads = _reader.reads
221 reads = _reader.reads
224 read = _reader.read
222 read = _reader.read
225 to_notebook = _reader.to_notebook
223 to_notebook = _reader.to_notebook
226 write = _writer.write
224 write = _writer.write
227 writes = _writer.writes
225 writes = _writer.writes
228
226
@@ -1,90 +1,89
1 from unittest import TestCase
1 from unittest import TestCase
2
2
3 from ..nbbase import (
3 from ..nbbase import (
4 NotebookNode,
4 NotebookNode,
5 new_code_cell, new_text_cell, new_worksheet, new_notebook, new_output
5 new_code_cell, new_text_cell, new_worksheet, new_notebook, new_output
6 )
6 )
7
7
8 class TestCell(TestCase):
8 class TestCell(TestCase):
9
9
10 def test_empty_code_cell(self):
10 def test_empty_code_cell(self):
11 cc = new_code_cell()
11 cc = new_code_cell()
12 self.assertEquals(cc.cell_type,'code')
12 self.assertEquals(cc.cell_type,'code')
13 self.assertEquals('input' not in cc, True)
13 self.assertEquals('input' not in cc, True)
14 self.assertEquals('prompt_number' not in cc, True)
14 self.assertEquals('prompt_number' not in cc, True)
15 self.assertEquals(cc.outputs, [])
15 self.assertEquals(cc.outputs, [])
16 self.assertEquals(cc.collapsed, False)
16 self.assertEquals(cc.collapsed, False)
17
17
18 def test_code_cell(self):
18 def test_code_cell(self):
19 cc = new_code_cell(input='a=10', prompt_number=0, collapsed=True)
19 cc = new_code_cell(input='a=10', prompt_number=0, collapsed=True)
20 cc.outputs = [new_output(output_type='pyout',
20 cc.outputs = [new_output(output_type='pyout',
21 output_svg='foo',output_text='10',prompt_number=0)]
21 output_svg='foo',output_text='10',prompt_number=0)]
22 self.assertEquals(cc.input, u'a=10')
22 self.assertEquals(cc.input, u'a=10')
23 self.assertEquals(cc.prompt_number, 0)
23 self.assertEquals(cc.prompt_number, 0)
24 self.assertEquals(cc.language, u'python')
24 self.assertEquals(cc.language, u'python')
25 self.assertEquals(cc.outputs[0].svg, u'foo')
25 self.assertEquals(cc.outputs[0].svg, u'foo')
26 self.assertEquals(cc.outputs[0].text, u'10')
26 self.assertEquals(cc.outputs[0].text, u'10')
27 self.assertEquals(cc.outputs[0].prompt_number, 0)
27 self.assertEquals(cc.outputs[0].prompt_number, 0)
28 self.assertEquals(cc.collapsed, True)
28 self.assertEquals(cc.collapsed, True)
29
29
30 def test_pyerr(self):
30 def test_pyerr(self):
31 o = new_output(output_type=u'pyerr', etype=u'NameError',
31 o = new_output(output_type=u'pyerr', etype=u'NameError',
32 evalue=u'Name not found', traceback=[u'frame 0', u'frame 1', u'frame 2']
32 evalue=u'Name not found', traceback=[u'frame 0', u'frame 1', u'frame 2']
33 )
33 )
34 self.assertEquals(o.output_type, u'pyerr')
34 self.assertEquals(o.output_type, u'pyerr')
35 self.assertEquals(o.etype, u'NameError')
35 self.assertEquals(o.etype, u'NameError')
36 self.assertEquals(o.evalue, u'Name not found')
36 self.assertEquals(o.evalue, u'Name not found')
37 self.assertEquals(o.traceback, [u'frame 0', u'frame 1', u'frame 2'])
37 self.assertEquals(o.traceback, [u'frame 0', u'frame 1', u'frame 2'])
38
38
39 def test_empty_html_cell(self):
39 def test_empty_html_cell(self):
40 tc = new_text_cell(u'html')
40 tc = new_text_cell(u'html')
41 self.assertEquals(tc.cell_type, u'html')
41 self.assertEquals(tc.cell_type, u'html')
42 self.assertEquals('source' not in tc, True)
42 self.assertEquals('source' not in tc, True)
43 self.assertEquals('rendered' not in tc, True)
43 self.assertEquals('rendered' not in tc, True)
44
44
45 def test_html_cell(self):
45 def test_html_cell(self):
46 tc = new_text_cell(u'html', 'hi', 'hi')
46 tc = new_text_cell(u'html', 'hi', 'hi')
47 self.assertEquals(tc.source, u'hi')
47 self.assertEquals(tc.source, u'hi')
48 self.assertEquals(tc.rendered, u'hi')
48 self.assertEquals(tc.rendered, u'hi')
49
49
50 def test_empty_markdown_cell(self):
50 def test_empty_markdown_cell(self):
51 tc = new_text_cell(u'markdown')
51 tc = new_text_cell(u'markdown')
52 self.assertEquals(tc.cell_type, u'markdown')
52 self.assertEquals(tc.cell_type, u'markdown')
53 self.assertEquals('source' not in tc, True)
53 self.assertEquals('source' not in tc, True)
54 self.assertEquals('rendered' not in tc, True)
54 self.assertEquals('rendered' not in tc, True)
55
55
56 def test_markdown_cell(self):
56 def test_markdown_cell(self):
57 tc = new_text_cell(u'markdown', 'hi', 'hi')
57 tc = new_text_cell(u'markdown', 'hi', 'hi')
58 self.assertEquals(tc.source, u'hi')
58 self.assertEquals(tc.source, u'hi')
59 self.assertEquals(tc.rendered, u'hi')
59 self.assertEquals(tc.rendered, u'hi')
60
60
61
61
62 class TestWorksheet(TestCase):
62 class TestWorksheet(TestCase):
63
63
64 def test_empty_worksheet(self):
64 def test_empty_worksheet(self):
65 ws = new_worksheet()
65 ws = new_worksheet()
66 self.assertEquals(ws.cells,[])
66 self.assertEquals(ws.cells,[])
67 self.assertEquals('name' not in ws, True)
67 self.assertEquals('name' not in ws, True)
68
68
69 def test_worksheet(self):
69 def test_worksheet(self):
70 cells = [new_code_cell(), new_text_cell(u'html')]
70 cells = [new_code_cell(), new_text_cell(u'html')]
71 ws = new_worksheet(cells=cells,name='foo')
71 ws = new_worksheet(cells=cells,name='foo')
72 self.assertEquals(ws.cells,cells)
72 self.assertEquals(ws.cells,cells)
73 self.assertEquals(ws.name,u'foo')
73 self.assertEquals(ws.name,u'foo')
74
74
75 class TestNotebook(TestCase):
75 class TestNotebook(TestCase):
76
76
77 def test_empty_notebook(self):
77 def test_empty_notebook(self):
78 nb = new_notebook()
78 nb = new_notebook()
79 self.assertEquals('id' in nb, True)
80 self.assertEquals(nb.worksheets, [])
79 self.assertEquals(nb.worksheets, [])
81 self.assertEquals('name' not in nb, True)
80 self.assertEquals('name' not in nb, True)
82 self.assertEquals(nb.nbformat,2)
81 self.assertEquals(nb.nbformat,2)
83
82
84 def test_notebook(self):
83 def test_notebook(self):
85 worksheets = [new_worksheet(),new_worksheet()]
84 worksheets = [new_worksheet(),new_worksheet()]
86 nb = new_notebook(name='foo',worksheets=worksheets)
85 nb = new_notebook(name='foo',worksheets=worksheets)
87 self.assertEquals(nb.name,u'foo')
86 self.assertEquals(nb.name,u'foo')
88 self.assertEquals(nb.worksheets,worksheets)
87 self.assertEquals(nb.worksheets,worksheets)
89 self.assertEquals(nb.nbformat,2)
88 self.assertEquals(nb.nbformat,2)
90
89
General Comments 0
You need to be logged in to leave comments. Login now