Show More
@@ -1,1069 +1,1074 | |||||
1 | //---------------------------------------------------------------------------- |
|
1 | //---------------------------------------------------------------------------- | |
2 | // Copyright (C) 2008-2011 The IPython Development Team |
|
2 | // Copyright (C) 2008-2011 The IPython Development Team | |
3 | // |
|
3 | // | |
4 | // Distributed under the terms of the BSD License. The full license is in |
|
4 | // Distributed under the terms of the BSD License. The full license is in | |
5 | // the file COPYING, distributed as part of this software. |
|
5 | // the file COPYING, distributed as part of this software. | |
6 | //---------------------------------------------------------------------------- |
|
6 | //---------------------------------------------------------------------------- | |
7 |
|
7 | |||
8 | //============================================================================ |
|
8 | //============================================================================ | |
9 | // Notebook |
|
9 | // Notebook | |
10 | //============================================================================ |
|
10 | //============================================================================ | |
11 |
|
11 | |||
12 | var IPython = (function (IPython) { |
|
12 | var IPython = (function (IPython) { | |
13 |
|
13 | |||
14 | var utils = IPython.utils; |
|
14 | var utils = IPython.utils; | |
15 |
|
15 | |||
16 | var Notebook = function (selector) { |
|
16 | var Notebook = function (selector) { | |
17 | this.read_only = IPython.read_only; |
|
17 | this.read_only = IPython.read_only; | |
18 | this.element = $(selector); |
|
18 | this.element = $(selector); | |
19 | this.element.scroll(); |
|
19 | this.element.scroll(); | |
20 | this.element.data("notebook", this); |
|
20 | this.element.data("notebook", this); | |
21 | this.next_prompt_number = 1; |
|
21 | this.next_prompt_number = 1; | |
22 | this.kernel = null; |
|
22 | this.kernel = null; | |
23 | this.dirty = false; |
|
23 | this.dirty = false; | |
24 | this.msg_cell_map = {}; |
|
24 | this.msg_cell_map = {}; | |
25 | this.metadata = {}; |
|
25 | this.metadata = {}; | |
26 | this.control_key_active = false; |
|
26 | this.control_key_active = false; | |
27 | this.style(); |
|
27 | this.style(); | |
28 | this.create_elements(); |
|
28 | this.create_elements(); | |
29 | this.bind_events(); |
|
29 | this.bind_events(); | |
30 | this.set_tooltipontab(true); |
|
30 | this.set_tooltipontab(true); | |
31 | this.set_smartcompleter(true); |
|
31 | this.set_smartcompleter(true); | |
32 | this.set_timebeforetooltip(1200); |
|
32 | this.set_timebeforetooltip(1200); | |
33 | }; |
|
33 | }; | |
34 |
|
34 | |||
35 |
|
35 | |||
36 | Notebook.prototype.style = function () { |
|
36 | Notebook.prototype.style = function () { | |
37 | $('div#notebook').addClass('border-box-sizing'); |
|
37 | $('div#notebook').addClass('border-box-sizing'); | |
38 | }; |
|
38 | }; | |
39 |
|
39 | |||
40 |
|
40 | |||
41 | Notebook.prototype.create_elements = function () { |
|
41 | Notebook.prototype.create_elements = function () { | |
42 | // We add this end_space div to the end of the notebook div to: |
|
42 | // We add this end_space div to the end of the notebook div to: | |
43 | // i) provide a margin between the last cell and the end of the notebook |
|
43 | // i) provide a margin between the last cell and the end of the notebook | |
44 | // ii) to prevent the div from scrolling up when the last cell is being |
|
44 | // ii) to prevent the div from scrolling up when the last cell is being | |
45 | // edited, but is too low on the page, which browsers will do automatically. |
|
45 | // edited, but is too low on the page, which browsers will do automatically. | |
46 | var that = this; |
|
46 | var that = this; | |
47 | var end_space = $('<div class="end_space"></div>').height(150); |
|
47 | var end_space = $('<div class="end_space"></div>').height(150); | |
48 | end_space.dblclick(function (e) { |
|
48 | end_space.dblclick(function (e) { | |
49 | if (that.read_only) return; |
|
49 | if (that.read_only) return; | |
50 | var ncells = that.ncells(); |
|
50 | var ncells = that.ncells(); | |
51 | that.insert_code_cell_below(ncells-1); |
|
51 | that.insert_code_cell_below(ncells-1); | |
52 | }); |
|
52 | }); | |
53 | this.element.append(end_space); |
|
53 | this.element.append(end_space); | |
54 | $('div#notebook').addClass('border-box-sizing'); |
|
54 | $('div#notebook').addClass('border-box-sizing'); | |
55 | }; |
|
55 | }; | |
56 |
|
56 | |||
57 |
|
57 | |||
58 | Notebook.prototype.bind_events = function () { |
|
58 | Notebook.prototype.bind_events = function () { | |
59 | var that = this; |
|
59 | var that = this; | |
60 | $(document).keydown(function (event) { |
|
60 | $(document).keydown(function (event) { | |
61 | // console.log(event); |
|
61 | // console.log(event); | |
62 | if (that.read_only) return; |
|
62 | if (that.read_only) return; | |
63 | if (event.which === 27) { |
|
63 | if (event.which === 27) { | |
64 | // Intercept escape at highest level to avoid closing |
|
64 | // Intercept escape at highest level to avoid closing | |
65 | // websocket connection with firefox |
|
65 | // websocket connection with firefox | |
66 | event.preventDefault(); |
|
66 | event.preventDefault(); | |
67 | } |
|
67 | } | |
68 | if (event.which === 38 && !event.shiftKey) { |
|
68 | if (event.which === 38 && !event.shiftKey) { | |
69 | var cell = that.selected_cell(); |
|
69 | var cell = that.selected_cell(); | |
70 | if (cell.at_top()) { |
|
70 | if (cell.at_top()) { | |
71 | event.preventDefault(); |
|
71 | event.preventDefault(); | |
72 | that.select_prev(); |
|
72 | that.select_prev(); | |
73 | }; |
|
73 | }; | |
74 | } else if (event.which === 40 && !event.shiftKey) { |
|
74 | } else if (event.which === 40 && !event.shiftKey) { | |
75 | var cell = that.selected_cell(); |
|
75 | var cell = that.selected_cell(); | |
76 | if (cell.at_bottom()) { |
|
76 | if (cell.at_bottom()) { | |
77 | event.preventDefault(); |
|
77 | event.preventDefault(); | |
78 | that.select_next(); |
|
78 | that.select_next(); | |
79 | }; |
|
79 | }; | |
80 | } else if (event.which === 13 && event.shiftKey) { |
|
80 | } else if (event.which === 13 && event.shiftKey) { | |
81 | that.execute_selected_cell(); |
|
81 | that.execute_selected_cell(); | |
82 | return false; |
|
82 | return false; | |
83 | } else if (event.which === 13 && event.ctrlKey) { |
|
83 | } else if (event.which === 13 && event.ctrlKey) { | |
84 | that.execute_selected_cell({terminal:true}); |
|
84 | that.execute_selected_cell({terminal:true}); | |
85 | return false; |
|
85 | return false; | |
86 | } else if (event.which === 77 && event.ctrlKey) { |
|
86 | } else if (event.which === 77 && event.ctrlKey) { | |
87 | that.control_key_active = true; |
|
87 | that.control_key_active = true; | |
88 | return false; |
|
88 | return false; | |
89 | } else if (event.which === 68 && that.control_key_active) { |
|
89 | } else if (event.which === 68 && that.control_key_active) { | |
90 | // Delete selected cell = d |
|
90 | // Delete selected cell = d | |
91 | that.delete_cell(); |
|
91 | that.delete_cell(); | |
92 | that.control_key_active = false; |
|
92 | that.control_key_active = false; | |
93 | return false; |
|
93 | return false; | |
94 | } else if (event.which === 65 && that.control_key_active) { |
|
94 | } else if (event.which === 65 && that.control_key_active) { | |
95 | // Insert code cell above selected = a |
|
95 | // Insert code cell above selected = a | |
96 | that.insert_code_cell_above(); |
|
96 | that.insert_code_cell_above(); | |
97 | that.control_key_active = false; |
|
97 | that.control_key_active = false; | |
98 | return false; |
|
98 | return false; | |
99 | } else if (event.which === 66 && that.control_key_active) { |
|
99 | } else if (event.which === 66 && that.control_key_active) { | |
100 | // Insert code cell below selected = b |
|
100 | // Insert code cell below selected = b | |
101 | that.insert_code_cell_below(); |
|
101 | that.insert_code_cell_below(); | |
102 | that.control_key_active = false; |
|
102 | that.control_key_active = false; | |
103 | return false; |
|
103 | return false; | |
104 | } else if (event.which === 67 && that.control_key_active) { |
|
104 | } else if (event.which === 67 && that.control_key_active) { | |
105 | // To code = c |
|
105 | // To code = c | |
106 | that.to_code(); |
|
106 | that.to_code(); | |
107 | that.control_key_active = false; |
|
107 | that.control_key_active = false; | |
108 | return false; |
|
108 | return false; | |
109 | } else if (event.which === 77 && that.control_key_active) { |
|
109 | } else if (event.which === 77 && that.control_key_active) { | |
110 | // To markdown = m |
|
110 | // To markdown = m | |
111 | that.to_markdown(); |
|
111 | that.to_markdown(); | |
112 | that.control_key_active = false; |
|
112 | that.control_key_active = false; | |
113 | return false; |
|
113 | return false; | |
114 | } else if (event.which === 84 && that.control_key_active) { |
|
114 | } else if (event.which === 84 && that.control_key_active) { | |
115 | // Toggle output = t |
|
115 | // Toggle output = t | |
116 | that.toggle_output(); |
|
116 | that.toggle_output(); | |
117 | that.control_key_active = false; |
|
117 | that.control_key_active = false; | |
118 | return false; |
|
118 | return false; | |
119 | } else if (event.which === 83 && that.control_key_active) { |
|
119 | } else if (event.which === 83 && that.control_key_active) { | |
120 | // Save notebook = s |
|
120 | // Save notebook = s | |
121 | IPython.save_widget.save_notebook(); |
|
121 | IPython.save_widget.save_notebook(); | |
122 | that.control_key_active = false; |
|
122 | that.control_key_active = false; | |
123 | return false; |
|
123 | return false; | |
124 | } else if (event.which === 74 && that.control_key_active) { |
|
124 | } else if (event.which === 74 && that.control_key_active) { | |
125 | // Move cell down = j |
|
125 | // Move cell down = j | |
126 | that.move_cell_down(); |
|
126 | that.move_cell_down(); | |
127 | that.control_key_active = false; |
|
127 | that.control_key_active = false; | |
128 | return false; |
|
128 | return false; | |
129 | } else if (event.which === 75 && that.control_key_active) { |
|
129 | } else if (event.which === 75 && that.control_key_active) { | |
130 | // Move cell up = k |
|
130 | // Move cell up = k | |
131 | that.move_cell_up(); |
|
131 | that.move_cell_up(); | |
132 | that.control_key_active = false; |
|
132 | that.control_key_active = false; | |
133 | return false; |
|
133 | return false; | |
134 | } else if (event.which === 80 && that.control_key_active) { |
|
134 | } else if (event.which === 80 && that.control_key_active) { | |
135 | // Select previous = p |
|
135 | // Select previous = p | |
136 | that.select_prev(); |
|
136 | that.select_prev(); | |
137 | that.control_key_active = false; |
|
137 | that.control_key_active = false; | |
138 | return false; |
|
138 | return false; | |
139 | } else if (event.which === 78 && that.control_key_active) { |
|
139 | } else if (event.which === 78 && that.control_key_active) { | |
140 | // Select next = n |
|
140 | // Select next = n | |
141 | that.select_next(); |
|
141 | that.select_next(); | |
142 | that.control_key_active = false; |
|
142 | that.control_key_active = false; | |
143 | return false; |
|
143 | return false; | |
144 | } else if (event.which === 76 && that.control_key_active) { |
|
144 | } else if (event.which === 76 && that.control_key_active) { | |
145 | // Toggle line numbers = l |
|
145 | // Toggle line numbers = l | |
146 | that.cell_toggle_line_numbers(); |
|
146 | that.cell_toggle_line_numbers(); | |
147 | that.control_key_active = false; |
|
147 | that.control_key_active = false; | |
148 | return false; |
|
148 | return false; | |
149 | } else if (event.which === 73 && that.control_key_active) { |
|
149 | } else if (event.which === 73 && that.control_key_active) { | |
150 | // Interrupt kernel = i |
|
150 | // Interrupt kernel = i | |
151 | IPython.notebook.kernel.interrupt(); |
|
151 | IPython.notebook.kernel.interrupt(); | |
152 | that.control_key_active = false; |
|
152 | that.control_key_active = false; | |
153 | return false; |
|
153 | return false; | |
154 | } else if (event.which === 190 && that.control_key_active) { |
|
154 | } else if (event.which === 190 && that.control_key_active) { | |
155 | // Restart kernel = . # matches qt console |
|
155 | // Restart kernel = . # matches qt console | |
156 | IPython.notebook.restart_kernel(); |
|
156 | IPython.notebook.restart_kernel(); | |
157 | that.control_key_active = false; |
|
157 | that.control_key_active = false; | |
158 | return false; |
|
158 | return false; | |
159 | } else if (event.which === 72 && that.control_key_active) { |
|
159 | } else if (event.which === 72 && that.control_key_active) { | |
160 | // Show keyboard shortcuts = h |
|
160 | // Show keyboard shortcuts = h | |
161 | that.toggle_keyboard_shortcuts(); |
|
161 | that.toggle_keyboard_shortcuts(); | |
162 | that.control_key_active = false; |
|
162 | that.control_key_active = false; | |
163 | return false; |
|
163 | return false; | |
164 | } else if (that.control_key_active) { |
|
164 | } else if (that.control_key_active) { | |
165 | that.control_key_active = false; |
|
165 | that.control_key_active = false; | |
166 | return true; |
|
166 | return true; | |
167 | }; |
|
167 | }; | |
168 | }); |
|
168 | }); | |
169 |
|
169 | |||
170 | this.element.bind('collapse_pager', function () { |
|
170 | this.element.bind('collapse_pager', function () { | |
171 | var app_height = $('div#main_app').height(); // content height |
|
171 | var app_height = $('div#main_app').height(); // content height | |
172 | var splitter_height = $('div#pager_splitter').outerHeight(true); |
|
172 | var splitter_height = $('div#pager_splitter').outerHeight(true); | |
173 | var new_height = app_height - splitter_height; |
|
173 | var new_height = app_height - splitter_height; | |
174 | that.element.animate({height : new_height + 'px'}, 'fast'); |
|
174 | that.element.animate({height : new_height + 'px'}, 'fast'); | |
175 | }); |
|
175 | }); | |
176 |
|
176 | |||
177 | this.element.bind('expand_pager', function () { |
|
177 | this.element.bind('expand_pager', function () { | |
178 | var app_height = $('div#main_app').height(); // content height |
|
178 | var app_height = $('div#main_app').height(); // content height | |
179 | var splitter_height = $('div#pager_splitter').outerHeight(true); |
|
179 | var splitter_height = $('div#pager_splitter').outerHeight(true); | |
180 | var pager_height = $('div#pager').outerHeight(true); |
|
180 | var pager_height = $('div#pager').outerHeight(true); | |
181 | var new_height = app_height - pager_height - splitter_height; |
|
181 | var new_height = app_height - pager_height - splitter_height; | |
182 | that.element.animate({height : new_height + 'px'}, 'fast'); |
|
182 | that.element.animate({height : new_height + 'px'}, 'fast'); | |
183 | }); |
|
183 | }); | |
184 |
|
184 | |||
185 | this.element.bind('collapse_left_panel', function () { |
|
185 | this.element.bind('collapse_left_panel', function () { | |
186 | var splitter_width = $('div#left_panel_splitter').outerWidth(true); |
|
186 | var splitter_width = $('div#left_panel_splitter').outerWidth(true); | |
187 | var new_margin = splitter_width; |
|
187 | var new_margin = splitter_width; | |
188 | $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast'); |
|
188 | $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast'); | |
189 | }); |
|
189 | }); | |
190 |
|
190 | |||
191 | this.element.bind('expand_left_panel', function () { |
|
191 | this.element.bind('expand_left_panel', function () { | |
192 | var splitter_width = $('div#left_panel_splitter').outerWidth(true); |
|
192 | var splitter_width = $('div#left_panel_splitter').outerWidth(true); | |
193 | var left_panel_width = IPython.left_panel.width; |
|
193 | var left_panel_width = IPython.left_panel.width; | |
194 | var new_margin = splitter_width + left_panel_width; |
|
194 | var new_margin = splitter_width + left_panel_width; | |
195 | $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast'); |
|
195 | $('div#notebook_panel').animate({marginLeft : new_margin + 'px'}, 'fast'); | |
196 | }); |
|
196 | }); | |
197 |
|
197 | |||
198 | $(window).bind('beforeunload', function () { |
|
198 | $(window).bind('beforeunload', function () { | |
199 | var kill_kernel = $('#kill_kernel').prop('checked'); |
|
199 | var kill_kernel = $('#kill_kernel').prop('checked'); | |
200 | if (kill_kernel) { |
|
200 | if (kill_kernel) { | |
201 | that.kernel.kill(); |
|
201 | that.kernel.kill(); | |
202 | } |
|
202 | } | |
203 | if (that.dirty && ! that.read_only) { |
|
203 | if (that.dirty && ! that.read_only) { | |
204 | return "You have unsaved changes that will be lost if you leave this page."; |
|
204 | return "You have unsaved changes that will be lost if you leave this page."; | |
205 | }; |
|
205 | }; | |
206 | }); |
|
206 | }); | |
207 | }; |
|
207 | }; | |
208 |
|
208 | |||
209 |
|
209 | |||
210 | Notebook.prototype.toggle_keyboard_shortcuts = function () { |
|
210 | Notebook.prototype.toggle_keyboard_shortcuts = function () { | |
211 | // toggles display of keyboard shortcut dialog |
|
211 | // toggles display of keyboard shortcut dialog | |
212 | var that = this; |
|
212 | var that = this; | |
213 | if ( this.shortcut_dialog ){ |
|
213 | if ( this.shortcut_dialog ){ | |
214 | // if dialog is already shown, close it |
|
214 | // if dialog is already shown, close it | |
215 | this.shortcut_dialog.dialog("close"); |
|
215 | this.shortcut_dialog.dialog("close"); | |
216 | this.shortcut_dialog = null; |
|
216 | this.shortcut_dialog = null; | |
217 | return; |
|
217 | return; | |
218 | } |
|
218 | } | |
219 | var dialog = $('<div/>'); |
|
219 | var dialog = $('<div/>'); | |
220 | this.shortcut_dialog = dialog; |
|
220 | this.shortcut_dialog = dialog; | |
221 | var shortcuts = [ |
|
221 | var shortcuts = [ | |
222 | {key: 'Shift-Enter', help: 'run cell'}, |
|
222 | {key: 'Shift-Enter', help: 'run cell'}, | |
223 | {key: 'Ctrl-Enter', help: 'run cell in-place'}, |
|
223 | {key: 'Ctrl-Enter', help: 'run cell in-place'}, | |
224 | {key: 'Ctrl-m d', help: 'delete cell'}, |
|
224 | {key: 'Ctrl-m d', help: 'delete cell'}, | |
225 | {key: 'Ctrl-m a', help: 'insert cell above'}, |
|
225 | {key: 'Ctrl-m a', help: 'insert cell above'}, | |
226 | {key: 'Ctrl-m b', help: 'insert cell below'}, |
|
226 | {key: 'Ctrl-m b', help: 'insert cell below'}, | |
227 | {key: 'Ctrl-m t', help: 'toggle output'}, |
|
227 | {key: 'Ctrl-m t', help: 'toggle output'}, | |
228 | {key: 'Ctrl-m l', help: 'toggle line numbers'}, |
|
228 | {key: 'Ctrl-m l', help: 'toggle line numbers'}, | |
229 | {key: 'Ctrl-m s', help: 'save notebook'}, |
|
229 | {key: 'Ctrl-m s', help: 'save notebook'}, | |
230 | {key: 'Ctrl-m j', help: 'move cell down'}, |
|
230 | {key: 'Ctrl-m j', help: 'move cell down'}, | |
231 | {key: 'Ctrl-m k', help: 'move cell up'}, |
|
231 | {key: 'Ctrl-m k', help: 'move cell up'}, | |
232 | {key: 'Ctrl-m c', help: 'code cell'}, |
|
232 | {key: 'Ctrl-m c', help: 'code cell'}, | |
233 | {key: 'Ctrl-m m', help: 'markdown cell'}, |
|
233 | {key: 'Ctrl-m m', help: 'markdown cell'}, | |
234 | {key: 'Ctrl-m p', help: 'select previous'}, |
|
234 | {key: 'Ctrl-m p', help: 'select previous'}, | |
235 | {key: 'Ctrl-m n', help: 'select next'}, |
|
235 | {key: 'Ctrl-m n', help: 'select next'}, | |
236 | {key: 'Ctrl-m i', help: 'interrupt kernel'}, |
|
236 | {key: 'Ctrl-m i', help: 'interrupt kernel'}, | |
237 | {key: 'Ctrl-m .', help: 'restart kernel'}, |
|
237 | {key: 'Ctrl-m .', help: 'restart kernel'}, | |
238 | {key: 'Ctrl-m h', help: 'show keyboard shortcuts'} |
|
238 | {key: 'Ctrl-m h', help: 'show keyboard shortcuts'} | |
239 | ]; |
|
239 | ]; | |
240 | for (var i=0; i<shortcuts.length; i++) { |
|
240 | for (var i=0; i<shortcuts.length; i++) { | |
241 | dialog.append($('<div>'). |
|
241 | dialog.append($('<div>'). | |
242 | append($('<span/>').addClass('shortcut_key').html(shortcuts[i].key)). |
|
242 | append($('<span/>').addClass('shortcut_key').html(shortcuts[i].key)). | |
243 | append($('<span/>').addClass('shortcut_descr').html(' : ' + shortcuts[i].help)) |
|
243 | append($('<span/>').addClass('shortcut_descr').html(' : ' + shortcuts[i].help)) | |
244 | ); |
|
244 | ); | |
245 | }; |
|
245 | }; | |
246 | dialog.bind('dialogclose', function(event) { |
|
246 | dialog.bind('dialogclose', function(event) { | |
247 | // dialog has been closed, allow it to be drawn again. |
|
247 | // dialog has been closed, allow it to be drawn again. | |
248 | that.shortcut_dialog = null; |
|
248 | that.shortcut_dialog = null; | |
249 | }); |
|
249 | }); | |
250 | dialog.dialog({title: 'Keyboard shortcuts'}); |
|
250 | dialog.dialog({title: 'Keyboard shortcuts'}); | |
251 | }; |
|
251 | }; | |
252 |
|
252 | |||
253 |
|
253 | |||
254 | Notebook.prototype.scroll_to_bottom = function () { |
|
254 | Notebook.prototype.scroll_to_bottom = function () { | |
255 | this.element.animate({scrollTop:this.element.get(0).scrollHeight}, 0); |
|
255 | this.element.animate({scrollTop:this.element.get(0).scrollHeight}, 0); | |
256 | }; |
|
256 | }; | |
257 |
|
257 | |||
258 |
|
258 | |||
259 | Notebook.prototype.scroll_to_top = function () { |
|
259 | Notebook.prototype.scroll_to_top = function () { | |
260 | this.element.animate({scrollTop:0}, 0); |
|
260 | this.element.animate({scrollTop:0}, 0); | |
261 | }; |
|
261 | }; | |
262 |
|
262 | |||
263 |
|
263 | |||
264 | // Cell indexing, retrieval, etc. |
|
264 | // Cell indexing, retrieval, etc. | |
265 |
|
265 | |||
266 |
|
266 | |||
267 | Notebook.prototype.cell_elements = function () { |
|
267 | Notebook.prototype.cell_elements = function () { | |
268 | return this.element.children("div.cell"); |
|
268 | return this.element.children("div.cell"); | |
269 | } |
|
269 | } | |
270 |
|
270 | |||
271 |
|
271 | |||
272 | Notebook.prototype.ncells = function (cell) { |
|
272 | Notebook.prototype.ncells = function (cell) { | |
273 | return this.cell_elements().length; |
|
273 | return this.cell_elements().length; | |
274 | } |
|
274 | } | |
275 |
|
275 | |||
276 |
|
276 | |||
277 | // TODO: we are often calling cells as cells()[i], which we should optimize |
|
277 | // TODO: we are often calling cells as cells()[i], which we should optimize | |
278 | // to cells(i) or a new method. |
|
278 | // to cells(i) or a new method. | |
279 | Notebook.prototype.cells = function () { |
|
279 | Notebook.prototype.cells = function () { | |
280 | return this.cell_elements().toArray().map(function (e) { |
|
280 | return this.cell_elements().toArray().map(function (e) { | |
281 | return $(e).data("cell"); |
|
281 | return $(e).data("cell"); | |
282 | }); |
|
282 | }); | |
283 | } |
|
283 | } | |
284 |
|
284 | |||
285 |
|
285 | |||
286 | Notebook.prototype.find_cell_index = function (cell) { |
|
286 | Notebook.prototype.find_cell_index = function (cell) { | |
287 | var result = null; |
|
287 | var result = null; | |
288 | this.cell_elements().filter(function (index) { |
|
288 | this.cell_elements().filter(function (index) { | |
289 | if ($(this).data("cell") === cell) { |
|
289 | if ($(this).data("cell") === cell) { | |
290 | result = index; |
|
290 | result = index; | |
291 | }; |
|
291 | }; | |
292 | }); |
|
292 | }); | |
293 | return result; |
|
293 | return result; | |
294 | }; |
|
294 | }; | |
295 |
|
295 | |||
296 |
|
296 | |||
297 | Notebook.prototype.index_or_selected = function (index) { |
|
297 | Notebook.prototype.index_or_selected = function (index) { | |
298 | return index || this.selected_index() || 0; |
|
298 | return index || this.selected_index() || 0; | |
299 | } |
|
299 | } | |
300 |
|
300 | |||
301 |
|
301 | |||
302 | Notebook.prototype.select = function (index) { |
|
302 | Notebook.prototype.select = function (index) { | |
303 | if (index !== undefined && index >= 0 && index < this.ncells()) { |
|
303 | if (index !== undefined && index >= 0 && index < this.ncells()) { | |
304 | if (this.selected_index() !== null) { |
|
304 | if (this.selected_index() !== null) { | |
305 | this.selected_cell().unselect(); |
|
305 | this.selected_cell().unselect(); | |
306 | }; |
|
306 | }; | |
307 | this.cells()[index].select(); |
|
307 | this.cells()[index].select(); | |
308 | }; |
|
308 | }; | |
309 | return this; |
|
309 | return this; | |
310 | }; |
|
310 | }; | |
311 |
|
311 | |||
312 |
|
312 | |||
313 | Notebook.prototype.select_next = function () { |
|
313 | Notebook.prototype.select_next = function () { | |
314 | var index = this.selected_index(); |
|
314 | var index = this.selected_index(); | |
315 | if (index !== null && index >= 0 && (index+1) < this.ncells()) { |
|
315 | if (index !== null && index >= 0 && (index+1) < this.ncells()) { | |
316 | this.select(index+1); |
|
316 | this.select(index+1); | |
317 | }; |
|
317 | }; | |
318 | return this; |
|
318 | return this; | |
319 | }; |
|
319 | }; | |
320 |
|
320 | |||
321 |
|
321 | |||
322 | Notebook.prototype.select_prev = function () { |
|
322 | Notebook.prototype.select_prev = function () { | |
323 | var index = this.selected_index(); |
|
323 | var index = this.selected_index(); | |
324 | if (index !== null && index >= 0 && (index-1) < this.ncells()) { |
|
324 | if (index !== null && index >= 0 && (index-1) < this.ncells()) { | |
325 | this.select(index-1); |
|
325 | this.select(index-1); | |
326 | }; |
|
326 | }; | |
327 | return this; |
|
327 | return this; | |
328 | }; |
|
328 | }; | |
329 |
|
329 | |||
330 |
|
330 | |||
331 | Notebook.prototype.selected_index = function () { |
|
331 | Notebook.prototype.selected_index = function () { | |
332 | var result = null; |
|
332 | var result = null; | |
333 | this.cell_elements().filter(function (index) { |
|
333 | this.cell_elements().filter(function (index) { | |
334 | if ($(this).data("cell").selected === true) { |
|
334 | if ($(this).data("cell").selected === true) { | |
335 | result = index; |
|
335 | result = index; | |
336 | }; |
|
336 | }; | |
337 | }); |
|
337 | }); | |
338 | return result; |
|
338 | return result; | |
339 | }; |
|
339 | }; | |
340 |
|
340 | |||
341 |
|
341 | |||
342 | Notebook.prototype.cell_for_msg = function (msg_id) { |
|
342 | Notebook.prototype.cell_for_msg = function (msg_id) { | |
343 | var cell_id = this.msg_cell_map[msg_id]; |
|
343 | var cell_id = this.msg_cell_map[msg_id]; | |
344 | var result = null; |
|
344 | var result = null; | |
345 | this.cell_elements().filter(function (index) { |
|
345 | this.cell_elements().filter(function (index) { | |
346 | cell = $(this).data("cell"); |
|
346 | cell = $(this).data("cell"); | |
347 | if (cell.cell_id === cell_id) { |
|
347 | if (cell.cell_id === cell_id) { | |
348 | result = cell; |
|
348 | result = cell; | |
349 | }; |
|
349 | }; | |
350 | }); |
|
350 | }); | |
351 | return result; |
|
351 | return result; | |
352 | }; |
|
352 | }; | |
353 |
|
353 | |||
354 |
|
354 | |||
355 | Notebook.prototype.selected_cell = function () { |
|
355 | Notebook.prototype.selected_cell = function () { | |
356 | return this.cell_elements().eq(this.selected_index()).data("cell"); |
|
356 | return this.cell_elements().eq(this.selected_index()).data("cell"); | |
357 | } |
|
357 | } | |
358 |
|
358 | |||
359 |
|
359 | |||
360 | // Cell insertion, deletion and moving. |
|
360 | // Cell insertion, deletion and moving. | |
361 |
|
361 | |||
362 |
|
362 | |||
363 | Notebook.prototype.delete_cell = function (index) { |
|
363 | Notebook.prototype.delete_cell = function (index) { | |
364 | var i = index || this.selected_index(); |
|
364 | var i = index || this.selected_index(); | |
365 | if (i !== null && i >= 0 && i < this.ncells()) { |
|
365 | if (i !== null && i >= 0 && i < this.ncells()) { | |
366 | this.cell_elements().eq(i).remove(); |
|
366 | this.cell_elements().eq(i).remove(); | |
367 | if (i === (this.ncells())) { |
|
367 | if (i === (this.ncells())) { | |
368 | this.select(i-1); |
|
368 | this.select(i-1); | |
369 | } else { |
|
369 | } else { | |
370 | this.select(i); |
|
370 | this.select(i); | |
371 | }; |
|
371 | }; | |
372 | }; |
|
372 | }; | |
373 | this.dirty = true; |
|
373 | this.dirty = true; | |
374 | return this; |
|
374 | return this; | |
375 | }; |
|
375 | }; | |
376 |
|
376 | |||
377 |
|
377 | |||
378 | Notebook.prototype.append_cell = function (cell) { |
|
378 | Notebook.prototype.append_cell = function (cell) { | |
379 | this.element.find('div.end_space').before(cell.element); |
|
379 | this.element.find('div.end_space').before(cell.element); | |
380 | this.dirty = true; |
|
380 | this.dirty = true; | |
381 | return this; |
|
381 | return this; | |
382 | }; |
|
382 | }; | |
383 |
|
383 | |||
384 |
|
384 | |||
385 | Notebook.prototype.insert_cell_below = function (cell, index) { |
|
385 | Notebook.prototype.insert_cell_below = function (cell, index) { | |
386 | var ncells = this.ncells(); |
|
386 | var ncells = this.ncells(); | |
387 | if (ncells === 0) { |
|
387 | if (ncells === 0) { | |
388 | this.append_cell(cell); |
|
388 | this.append_cell(cell); | |
389 | return this; |
|
389 | return this; | |
390 | }; |
|
390 | }; | |
391 | if (index >= 0 && index < ncells) { |
|
391 | if (index >= 0 && index < ncells) { | |
392 | this.cell_elements().eq(index).after(cell.element); |
|
392 | this.cell_elements().eq(index).after(cell.element); | |
393 | }; |
|
393 | }; | |
394 | this.dirty = true; |
|
394 | this.dirty = true; | |
395 | return this |
|
395 | return this | |
396 | }; |
|
396 | }; | |
397 |
|
397 | |||
398 |
|
398 | |||
399 | Notebook.prototype.insert_cell_above = function (cell, index) { |
|
399 | Notebook.prototype.insert_cell_above = function (cell, index) { | |
400 | var ncells = this.ncells(); |
|
400 | var ncells = this.ncells(); | |
401 | if (ncells === 0) { |
|
401 | if (ncells === 0) { | |
402 | this.append_cell(cell); |
|
402 | this.append_cell(cell); | |
403 | return this; |
|
403 | return this; | |
404 | }; |
|
404 | }; | |
405 | if (index >= 0 && index < ncells) { |
|
405 | if (index >= 0 && index < ncells) { | |
406 | this.cell_elements().eq(index).before(cell.element); |
|
406 | this.cell_elements().eq(index).before(cell.element); | |
407 | }; |
|
407 | }; | |
408 | this.dirty = true; |
|
408 | this.dirty = true; | |
409 | return this; |
|
409 | return this; | |
410 | }; |
|
410 | }; | |
411 |
|
411 | |||
412 |
|
412 | |||
413 | Notebook.prototype.move_cell_up = function (index) { |
|
413 | Notebook.prototype.move_cell_up = function (index) { | |
414 | var i = index || this.selected_index(); |
|
414 | var i = index || this.selected_index(); | |
415 | if (i !== null && i < this.ncells() && i > 0) { |
|
415 | if (i !== null && i < this.ncells() && i > 0) { | |
416 | var pivot = this.cell_elements().eq(i-1); |
|
416 | var pivot = this.cell_elements().eq(i-1); | |
417 | var tomove = this.cell_elements().eq(i); |
|
417 | var tomove = this.cell_elements().eq(i); | |
418 | if (pivot !== null && tomove !== null) { |
|
418 | if (pivot !== null && tomove !== null) { | |
419 | tomove.detach(); |
|
419 | tomove.detach(); | |
420 | pivot.before(tomove); |
|
420 | pivot.before(tomove); | |
421 | this.select(i-1); |
|
421 | this.select(i-1); | |
422 | }; |
|
422 | }; | |
423 | }; |
|
423 | }; | |
424 | this.dirty = true; |
|
424 | this.dirty = true; | |
425 | return this; |
|
425 | return this; | |
426 | } |
|
426 | } | |
427 |
|
427 | |||
428 |
|
428 | |||
429 | Notebook.prototype.move_cell_down = function (index) { |
|
429 | Notebook.prototype.move_cell_down = function (index) { | |
430 | var i = index || this.selected_index(); |
|
430 | var i = index || this.selected_index(); | |
431 | if (i !== null && i < (this.ncells()-1) && i >= 0) { |
|
431 | if (i !== null && i < (this.ncells()-1) && i >= 0) { | |
432 | var pivot = this.cell_elements().eq(i+1) |
|
432 | var pivot = this.cell_elements().eq(i+1) | |
433 | var tomove = this.cell_elements().eq(i) |
|
433 | var tomove = this.cell_elements().eq(i) | |
434 | if (pivot !== null && tomove !== null) { |
|
434 | if (pivot !== null && tomove !== null) { | |
435 | tomove.detach(); |
|
435 | tomove.detach(); | |
436 | pivot.after(tomove); |
|
436 | pivot.after(tomove); | |
437 | this.select(i+1); |
|
437 | this.select(i+1); | |
438 | }; |
|
438 | }; | |
439 | }; |
|
439 | }; | |
440 | this.dirty = true; |
|
440 | this.dirty = true; | |
441 | return this; |
|
441 | return this; | |
442 | } |
|
442 | } | |
443 |
|
443 | |||
444 |
|
444 | |||
445 | Notebook.prototype.sort_cells = function () { |
|
445 | Notebook.prototype.sort_cells = function () { | |
446 | var ncells = this.ncells(); |
|
446 | var ncells = this.ncells(); | |
447 | var sindex = this.selected_index(); |
|
447 | var sindex = this.selected_index(); | |
448 | var swapped; |
|
448 | var swapped; | |
449 | do { |
|
449 | do { | |
450 | swapped = false |
|
450 | swapped = false | |
451 | for (var i=1; i<ncells; i++) { |
|
451 | for (var i=1; i<ncells; i++) { | |
452 | current = this.cell_elements().eq(i).data("cell"); |
|
452 | current = this.cell_elements().eq(i).data("cell"); | |
453 | previous = this.cell_elements().eq(i-1).data("cell"); |
|
453 | previous = this.cell_elements().eq(i-1).data("cell"); | |
454 | if (previous.input_prompt_number > current.input_prompt_number) { |
|
454 | if (previous.input_prompt_number > current.input_prompt_number) { | |
455 | this.move_cell_up(i); |
|
455 | this.move_cell_up(i); | |
456 | swapped = true; |
|
456 | swapped = true; | |
457 | }; |
|
457 | }; | |
458 | }; |
|
458 | }; | |
459 | } while (swapped); |
|
459 | } while (swapped); | |
460 | this.select(sindex); |
|
460 | this.select(sindex); | |
461 | return this; |
|
461 | return this; | |
462 | }; |
|
462 | }; | |
463 |
|
463 | |||
464 |
|
464 | |||
465 | Notebook.prototype.insert_code_cell_above = function (index) { |
|
465 | Notebook.prototype.insert_code_cell_above = function (index) { | |
466 | // TODO: Bounds check for i |
|
466 | // TODO: Bounds check for i | |
467 | var i = this.index_or_selected(index); |
|
467 | var i = this.index_or_selected(index); | |
468 | var cell = new IPython.CodeCell(this); |
|
468 | var cell = new IPython.CodeCell(this); | |
469 | cell.set_input_prompt(); |
|
469 | cell.set_input_prompt(); | |
470 | this.insert_cell_above(cell, i); |
|
470 | this.insert_cell_above(cell, i); | |
471 | this.select(this.find_cell_index(cell)); |
|
471 | this.select(this.find_cell_index(cell)); | |
472 | return cell; |
|
472 | return cell; | |
473 | } |
|
473 | } | |
474 |
|
474 | |||
475 |
|
475 | |||
476 | Notebook.prototype.insert_code_cell_below = function (index) { |
|
476 | Notebook.prototype.insert_code_cell_below = function (index) { | |
477 | // TODO: Bounds check for i |
|
477 | // TODO: Bounds check for i | |
478 | var i = this.index_or_selected(index); |
|
478 | var i = this.index_or_selected(index); | |
479 | var cell = new IPython.CodeCell(this); |
|
479 | var cell = new IPython.CodeCell(this); | |
480 | cell.set_input_prompt(); |
|
480 | cell.set_input_prompt(); | |
481 | this.insert_cell_below(cell, i); |
|
481 | this.insert_cell_below(cell, i); | |
482 | this.select(this.find_cell_index(cell)); |
|
482 | this.select(this.find_cell_index(cell)); | |
483 | return cell; |
|
483 | return cell; | |
484 | } |
|
484 | } | |
485 |
|
485 | |||
486 |
|
486 | |||
487 | Notebook.prototype.insert_html_cell_above = function (index) { |
|
487 | Notebook.prototype.insert_html_cell_above = function (index) { | |
488 | // TODO: Bounds check for i |
|
488 | // TODO: Bounds check for i | |
489 | var i = this.index_or_selected(index); |
|
489 | var i = this.index_or_selected(index); | |
490 | var cell = new IPython.HTMLCell(this); |
|
490 | var cell = new IPython.HTMLCell(this); | |
491 | cell.config_mathjax(); |
|
491 | cell.config_mathjax(); | |
492 | this.insert_cell_above(cell, i); |
|
492 | this.insert_cell_above(cell, i); | |
493 | this.select(this.find_cell_index(cell)); |
|
493 | this.select(this.find_cell_index(cell)); | |
494 | return cell; |
|
494 | return cell; | |
495 | } |
|
495 | } | |
496 |
|
496 | |||
497 |
|
497 | |||
498 | Notebook.prototype.insert_html_cell_below = function (index) { |
|
498 | Notebook.prototype.insert_html_cell_below = function (index) { | |
499 | // TODO: Bounds check for i |
|
499 | // TODO: Bounds check for i | |
500 | var i = this.index_or_selected(index); |
|
500 | var i = this.index_or_selected(index); | |
501 | var cell = new IPython.HTMLCell(this); |
|
501 | var cell = new IPython.HTMLCell(this); | |
502 | cell.config_mathjax(); |
|
502 | cell.config_mathjax(); | |
503 | this.insert_cell_below(cell, i); |
|
503 | this.insert_cell_below(cell, i); | |
504 | this.select(this.find_cell_index(cell)); |
|
504 | this.select(this.find_cell_index(cell)); | |
505 | return cell; |
|
505 | return cell; | |
506 | } |
|
506 | } | |
507 |
|
507 | |||
508 |
|
508 | |||
509 | Notebook.prototype.insert_markdown_cell_above = function (index) { |
|
509 | Notebook.prototype.insert_markdown_cell_above = function (index) { | |
510 | // TODO: Bounds check for i |
|
510 | // TODO: Bounds check for i | |
511 | var i = this.index_or_selected(index); |
|
511 | var i = this.index_or_selected(index); | |
512 | var cell = new IPython.MarkdownCell(this); |
|
512 | var cell = new IPython.MarkdownCell(this); | |
513 | cell.config_mathjax(); |
|
513 | cell.config_mathjax(); | |
514 | this.insert_cell_above(cell, i); |
|
514 | this.insert_cell_above(cell, i); | |
515 | this.select(this.find_cell_index(cell)); |
|
515 | this.select(this.find_cell_index(cell)); | |
516 | return cell; |
|
516 | return cell; | |
517 | } |
|
517 | } | |
518 |
|
518 | |||
519 |
|
519 | |||
520 | Notebook.prototype.insert_markdown_cell_below = function (index) { |
|
520 | Notebook.prototype.insert_markdown_cell_below = function (index) { | |
521 | // TODO: Bounds check for i |
|
521 | // TODO: Bounds check for i | |
522 | var i = this.index_or_selected(index); |
|
522 | var i = this.index_or_selected(index); | |
523 | var cell = new IPython.MarkdownCell(this); |
|
523 | var cell = new IPython.MarkdownCell(this); | |
524 | cell.config_mathjax(); |
|
524 | cell.config_mathjax(); | |
525 | this.insert_cell_below(cell, i); |
|
525 | this.insert_cell_below(cell, i); | |
526 | this.select(this.find_cell_index(cell)); |
|
526 | this.select(this.find_cell_index(cell)); | |
527 | return cell; |
|
527 | return cell; | |
528 | } |
|
528 | } | |
529 |
|
529 | |||
530 |
|
530 | |||
531 | Notebook.prototype.to_code = function (index) { |
|
531 | Notebook.prototype.to_code = function (index) { | |
532 | // TODO: Bounds check for i |
|
532 | // TODO: Bounds check for i | |
533 | var i = this.index_or_selected(index); |
|
533 | var i = this.index_or_selected(index); | |
534 | var source_element = this.cell_elements().eq(i); |
|
534 | var source_element = this.cell_elements().eq(i); | |
535 | var source_cell = source_element.data("cell"); |
|
535 | var source_cell = source_element.data("cell"); | |
536 | if (source_cell instanceof IPython.HTMLCell || |
|
536 | if (source_cell instanceof IPython.HTMLCell || | |
537 | source_cell instanceof IPython.MarkdownCell) { |
|
537 | source_cell instanceof IPython.MarkdownCell) { | |
538 | this.insert_code_cell_below(i); |
|
538 | this.insert_code_cell_below(i); | |
539 | var target_cell = this.cells()[i+1]; |
|
539 | var target_cell = this.cells()[i+1]; | |
540 | target_cell.set_code(source_cell.get_source()); |
|
540 | target_cell.set_code(source_cell.get_source()); | |
541 | source_element.remove(); |
|
541 | source_element.remove(); | |
542 | target_cell.select(); |
|
542 | target_cell.select(); | |
543 | }; |
|
543 | }; | |
544 | this.dirty = true; |
|
544 | this.dirty = true; | |
545 | }; |
|
545 | }; | |
546 |
|
546 | |||
547 |
|
547 | |||
548 | Notebook.prototype.to_markdown = function (index) { |
|
548 | Notebook.prototype.to_markdown = function (index) { | |
549 | // TODO: Bounds check for i |
|
549 | // TODO: Bounds check for i | |
550 | var i = this.index_or_selected(index); |
|
550 | var i = this.index_or_selected(index); | |
551 | var source_element = this.cell_elements().eq(i); |
|
551 | var source_element = this.cell_elements().eq(i); | |
552 | var source_cell = source_element.data("cell"); |
|
552 | var source_cell = source_element.data("cell"); | |
553 | var target_cell = null; |
|
553 | var target_cell = null; | |
554 | if (source_cell instanceof IPython.CodeCell) { |
|
554 | if (source_cell instanceof IPython.CodeCell) { | |
555 | this.insert_markdown_cell_below(i); |
|
555 | this.insert_markdown_cell_below(i); | |
556 | var target_cell = this.cells()[i+1]; |
|
556 | var target_cell = this.cells()[i+1]; | |
557 | var text = source_cell.get_code(); |
|
557 | var text = source_cell.get_code(); | |
558 | } else if (source_cell instanceof IPython.HTMLCell) { |
|
558 | } else if (source_cell instanceof IPython.HTMLCell) { | |
559 | this.insert_markdown_cell_below(i); |
|
559 | this.insert_markdown_cell_below(i); | |
560 | var target_cell = this.cells()[i+1]; |
|
560 | var target_cell = this.cells()[i+1]; | |
561 | var text = source_cell.get_source(); |
|
561 | var text = source_cell.get_source(); | |
562 | if (text === source_cell.placeholder) { |
|
562 | if (text === source_cell.placeholder) { | |
563 | text = target_cell.placeholder; |
|
563 | text = target_cell.placeholder; | |
564 | } |
|
564 | } | |
565 | } |
|
565 | } | |
566 | if (target_cell !== null) { |
|
566 | if (target_cell !== null) { | |
567 | if (text === "") {text = target_cell.placeholder;}; |
|
567 | if (text === "") {text = target_cell.placeholder;}; | |
568 | target_cell.set_source(text); |
|
568 | target_cell.set_source(text); | |
569 | source_element.remove(); |
|
569 | source_element.remove(); | |
570 | target_cell.edit(); |
|
570 | target_cell.edit(); | |
571 | } |
|
571 | } | |
572 | this.dirty = true; |
|
572 | this.dirty = true; | |
573 | }; |
|
573 | }; | |
574 |
|
574 | |||
575 |
|
575 | |||
576 | Notebook.prototype.to_html = function (index) { |
|
576 | Notebook.prototype.to_html = function (index) { | |
577 | // TODO: Bounds check for i |
|
577 | // TODO: Bounds check for i | |
578 | var i = this.index_or_selected(index); |
|
578 | var i = this.index_or_selected(index); | |
579 | var source_element = this.cell_elements().eq(i); |
|
579 | var source_element = this.cell_elements().eq(i); | |
580 | var source_cell = source_element.data("cell"); |
|
580 | var source_cell = source_element.data("cell"); | |
581 | var target_cell = null; |
|
581 | var target_cell = null; | |
582 | if (source_cell instanceof IPython.CodeCell) { |
|
582 | if (source_cell instanceof IPython.CodeCell) { | |
583 | this.insert_html_cell_below(i); |
|
583 | this.insert_html_cell_below(i); | |
584 | var target_cell = this.cells()[i+1]; |
|
584 | var target_cell = this.cells()[i+1]; | |
585 | var text = source_cell.get_code(); |
|
585 | var text = source_cell.get_code(); | |
586 | } else if (source_cell instanceof IPython.MarkdownCell) { |
|
586 | } else if (source_cell instanceof IPython.MarkdownCell) { | |
587 | this.insert_html_cell_below(i); |
|
587 | this.insert_html_cell_below(i); | |
588 | var target_cell = this.cells()[i+1]; |
|
588 | var target_cell = this.cells()[i+1]; | |
589 | var text = source_cell.get_source(); |
|
589 | var text = source_cell.get_source(); | |
590 | if (text === source_cell.placeholder) { |
|
590 | if (text === source_cell.placeholder) { | |
591 | text = target_cell.placeholder; |
|
591 | text = target_cell.placeholder; | |
592 | } |
|
592 | } | |
593 | } |
|
593 | } | |
594 | if (target_cell !== null) { |
|
594 | if (target_cell !== null) { | |
595 | if (text === "") {text = target_cell.placeholder;}; |
|
595 | if (text === "") {text = target_cell.placeholder;}; | |
596 | target_cell.set_source(text); |
|
596 | target_cell.set_source(text); | |
597 | source_element.remove(); |
|
597 | source_element.remove(); | |
598 | target_cell.edit(); |
|
598 | target_cell.edit(); | |
599 | } |
|
599 | } | |
600 | this.dirty = true; |
|
600 | this.dirty = true; | |
601 | }; |
|
601 | }; | |
602 |
|
602 | |||
603 |
|
603 | |||
604 | // Cell collapsing and output clearing |
|
604 | // Cell collapsing and output clearing | |
605 |
|
605 | |||
606 | Notebook.prototype.collapse = function (index) { |
|
606 | Notebook.prototype.collapse = function (index) { | |
607 | var i = this.index_or_selected(index); |
|
607 | var i = this.index_or_selected(index); | |
608 | this.cells()[i].collapse(); |
|
608 | this.cells()[i].collapse(); | |
609 | this.dirty = true; |
|
609 | this.dirty = true; | |
610 | }; |
|
610 | }; | |
611 |
|
611 | |||
612 |
|
612 | |||
613 | Notebook.prototype.expand = function (index) { |
|
613 | Notebook.prototype.expand = function (index) { | |
614 | var i = this.index_or_selected(index); |
|
614 | var i = this.index_or_selected(index); | |
615 | this.cells()[i].expand(); |
|
615 | this.cells()[i].expand(); | |
616 | this.dirty = true; |
|
616 | this.dirty = true; | |
617 | }; |
|
617 | }; | |
618 |
|
618 | |||
619 |
|
619 | |||
620 | Notebook.prototype.toggle_output = function (index) { |
|
620 | Notebook.prototype.toggle_output = function (index) { | |
621 | var i = this.index_or_selected(index); |
|
621 | var i = this.index_or_selected(index); | |
622 | this.cells()[i].toggle_output(); |
|
622 | this.cells()[i].toggle_output(); | |
623 | this.dirty = true; |
|
623 | this.dirty = true; | |
624 | }; |
|
624 | }; | |
625 |
|
625 | |||
626 |
|
626 | |||
627 | Notebook.prototype.set_timebeforetooltip = function (time) { |
|
627 | Notebook.prototype.set_timebeforetooltip = function (time) { | |
628 | console.log("change time before tooltip to : "+time); |
|
628 | console.log("change time before tooltip to : "+time); | |
629 | this.time_before_tooltip = time; |
|
629 | this.time_before_tooltip = time; | |
630 | }; |
|
630 | }; | |
631 |
|
631 | |||
632 | Notebook.prototype.set_tooltipontab = function (state) { |
|
632 | Notebook.prototype.set_tooltipontab = function (state) { | |
633 | console.log("change tooltip on tab to : "+state); |
|
633 | console.log("change tooltip on tab to : "+state); | |
634 | this.tooltip_on_tab = state; |
|
634 | this.tooltip_on_tab = state; | |
635 | }; |
|
635 | }; | |
636 |
|
636 | |||
637 | Notebook.prototype.set_smartcompleter = function (state) { |
|
637 | Notebook.prototype.set_smartcompleter = function (state) { | |
638 | console.log("Smart completion (kwargs first) changed to to : "+state); |
|
638 | console.log("Smart completion (kwargs first) changed to to : "+state); | |
639 | this.smart_completer = state; |
|
639 | this.smart_completer = state; | |
640 | }; |
|
640 | }; | |
641 |
|
641 | |||
642 | Notebook.prototype.set_autoindent = function (state) { |
|
642 | Notebook.prototype.set_autoindent = function (state) { | |
643 | var cells = this.cells(); |
|
643 | var cells = this.cells(); | |
644 | len = cells.length; |
|
644 | len = cells.length; | |
645 | for (var i=0; i<len; i++) { |
|
645 | for (var i=0; i<len; i++) { | |
646 | cells[i].set_autoindent(state) |
|
646 | cells[i].set_autoindent(state) | |
647 | }; |
|
647 | }; | |
648 | }; |
|
648 | }; | |
649 |
|
649 | |||
650 |
|
650 | |||
651 | Notebook.prototype.clear_all_output = function () { |
|
651 | Notebook.prototype.clear_all_output = function () { | |
652 | var ncells = this.ncells(); |
|
652 | var ncells = this.ncells(); | |
653 | var cells = this.cells(); |
|
653 | var cells = this.cells(); | |
654 | for (var i=0; i<ncells; i++) { |
|
654 | for (var i=0; i<ncells; i++) { | |
655 | if (cells[i] instanceof IPython.CodeCell) { |
|
655 | if (cells[i] instanceof IPython.CodeCell) { | |
656 | cells[i].clear_output(true,true,true); |
|
656 | cells[i].clear_output(true,true,true); | |
657 | } |
|
657 | } | |
658 | }; |
|
658 | }; | |
659 | this.dirty = true; |
|
659 | this.dirty = true; | |
660 | }; |
|
660 | }; | |
661 |
|
661 | |||
662 | // Other cell functions: line numbers, ... |
|
662 | // Other cell functions: line numbers, ... | |
663 |
|
663 | |||
664 | Notebook.prototype.cell_toggle_line_numbers = function() { |
|
664 | Notebook.prototype.cell_toggle_line_numbers = function() { | |
665 | this.selected_cell().toggle_line_numbers() |
|
665 | this.selected_cell().toggle_line_numbers() | |
666 | }; |
|
666 | }; | |
667 |
|
667 | |||
668 | // Kernel related things |
|
668 | // Kernel related things | |
669 |
|
669 | |||
670 | Notebook.prototype.start_kernel = function () { |
|
670 | Notebook.prototype.start_kernel = function () { | |
671 | this.kernel = new IPython.Kernel(); |
|
671 | this.kernel = new IPython.Kernel(); | |
672 | var notebook_id = IPython.save_widget.get_notebook_id(); |
|
672 | var notebook_id = IPython.save_widget.get_notebook_id(); | |
673 | this.kernel.start(notebook_id, $.proxy(this.kernel_started, this)); |
|
673 | this.kernel.start(notebook_id, $.proxy(this.kernel_started, this)); | |
674 | }; |
|
674 | }; | |
675 |
|
675 | |||
676 |
|
676 | |||
677 | Notebook.prototype.restart_kernel = function () { |
|
677 | Notebook.prototype.restart_kernel = function () { | |
678 | var that = this; |
|
678 | var that = this; | |
679 | var notebook_id = IPython.save_widget.get_notebook_id(); |
|
679 | var notebook_id = IPython.save_widget.get_notebook_id(); | |
680 |
|
680 | |||
681 | var dialog = $('<div/>'); |
|
681 | var dialog = $('<div/>'); | |
682 | dialog.html('Do you want to restart the current kernel? You will lose all variables defined in it.'); |
|
682 | dialog.html('Do you want to restart the current kernel? You will lose all variables defined in it.'); | |
683 | $(document).append(dialog); |
|
683 | $(document).append(dialog); | |
684 | dialog.dialog({ |
|
684 | dialog.dialog({ | |
685 | resizable: false, |
|
685 | resizable: false, | |
686 | modal: true, |
|
686 | modal: true, | |
687 | title: "Restart kernel or continue running?", |
|
687 | title: "Restart kernel or continue running?", | |
688 | buttons : { |
|
688 | buttons : { | |
689 | "Restart": function () { |
|
689 | "Restart": function () { | |
690 | that.kernel.restart($.proxy(that.kernel_started, that)); |
|
690 | that.kernel.restart($.proxy(that.kernel_started, that)); | |
691 | $(this).dialog('close'); |
|
691 | $(this).dialog('close'); | |
692 | }, |
|
692 | }, | |
693 | "Continue running": function () { |
|
693 | "Continue running": function () { | |
694 | $(this).dialog('close'); |
|
694 | $(this).dialog('close'); | |
695 | } |
|
695 | } | |
696 | } |
|
696 | } | |
697 | }); |
|
697 | }); | |
698 | }; |
|
698 | }; | |
699 |
|
699 | |||
700 |
|
700 | |||
701 | Notebook.prototype.kernel_started = function () { |
|
701 | Notebook.prototype.kernel_started = function () { | |
702 | console.log("Kernel started: ", this.kernel.kernel_id); |
|
702 | console.log("Kernel started: ", this.kernel.kernel_id); | |
703 | this.kernel.shell_channel.onmessage = $.proxy(this.handle_shell_reply,this); |
|
703 | this.kernel.shell_channel.onmessage = $.proxy(this.handle_shell_reply,this); | |
704 | this.kernel.iopub_channel.onmessage = $.proxy(this.handle_iopub_reply,this); |
|
704 | this.kernel.iopub_channel.onmessage = $.proxy(this.handle_iopub_reply,this); | |
705 | }; |
|
705 | }; | |
706 |
|
706 | |||
707 |
|
707 | |||
708 | Notebook.prototype.handle_shell_reply = function (e) { |
|
708 | Notebook.prototype.handle_shell_reply = function (e) { | |
709 | reply = $.parseJSON(e.data); |
|
709 | reply = $.parseJSON(e.data); | |
710 | var header = reply.header; |
|
710 | var header = reply.header; | |
711 | var content = reply.content; |
|
711 | var content = reply.content; | |
712 | var msg_type = header.msg_type; |
|
712 | var msg_type = header.msg_type; | |
713 | // console.log(reply); |
|
713 | // console.log(reply); | |
714 | var cell = this.cell_for_msg(reply.parent_header.msg_id); |
|
714 | var cell = this.cell_for_msg(reply.parent_header.msg_id); | |
715 | if (msg_type === "execute_reply") { |
|
715 | if (msg_type === "execute_reply") { | |
716 | cell.set_input_prompt(content.execution_count); |
|
716 | cell.set_input_prompt(content.execution_count); | |
717 | cell.element.removeClass("running"); |
|
717 | cell.element.removeClass("running"); | |
718 | this.dirty = true; |
|
718 | this.dirty = true; | |
719 | } else if (msg_type === "complete_reply") { |
|
719 | } else if (msg_type === "complete_reply") { | |
720 | cell.finish_completing(content.matched_text, content.matches); |
|
720 | cell.finish_completing(content.matched_text, content.matches); | |
721 | } else if (msg_type === "object_info_reply"){ |
|
721 | } else if (msg_type === "object_info_reply"){ | |
722 | //console.log('back from object_info_request : ') |
|
722 | //console.log('back from object_info_request : ') | |
723 | rep = reply.content; |
|
723 | rep = reply.content; | |
724 | if(rep.found) |
|
724 | if(rep.found) | |
725 | { |
|
725 | { | |
726 | cell.finish_tooltip(rep); |
|
726 | cell.finish_tooltip(rep); | |
727 | } |
|
727 | } | |
728 | } else { |
|
728 | } else { | |
729 | //console.log("unknown reply:"+msg_type); |
|
729 | //console.log("unknown reply:"+msg_type); | |
730 | } |
|
730 | } | |
731 | // when having a rely from object_info_reply, |
|
731 | // when having a rely from object_info_reply, | |
732 | // no payload so no nned to handle it |
|
732 | // no payload so no nned to handle it | |
733 | if(typeof(content.payload)!='undefined') { |
|
733 | if(typeof(content.payload)!='undefined') { | |
734 | var payload = content.payload || []; |
|
734 | var payload = content.payload || []; | |
735 | this.handle_payload(cell, payload); |
|
735 | this.handle_payload(cell, payload); | |
736 | } |
|
736 | } | |
737 | }; |
|
737 | }; | |
738 |
|
738 | |||
739 |
|
739 | |||
740 | Notebook.prototype.handle_payload = function (cell, payload) { |
|
740 | Notebook.prototype.handle_payload = function (cell, payload) { | |
741 | var l = payload.length; |
|
741 | var l = payload.length; | |
742 | for (var i=0; i<l; i++) { |
|
742 | for (var i=0; i<l; i++) { | |
743 | if (payload[i].source === 'IPython.zmq.page.page') { |
|
743 | if (payload[i].source === 'IPython.zmq.page.page') { | |
744 | if (payload[i].text.trim() !== '') { |
|
744 | if (payload[i].text.trim() !== '') { | |
745 | IPython.pager.clear(); |
|
745 | IPython.pager.clear(); | |
746 | IPython.pager.expand(); |
|
746 | IPython.pager.expand(); | |
747 | IPython.pager.append_text(payload[i].text); |
|
747 | IPython.pager.append_text(payload[i].text); | |
748 | } |
|
748 | } | |
749 | } else if (payload[i].source === 'IPython.zmq.zmqshell.ZMQInteractiveShell.set_next_input') { |
|
749 | } else if (payload[i].source === 'IPython.zmq.zmqshell.ZMQInteractiveShell.set_next_input') { | |
750 | var index = this.find_cell_index(cell); |
|
750 | var index = this.find_cell_index(cell); | |
751 | var new_cell = this.insert_code_cell_below(index); |
|
751 | var new_cell = this.insert_code_cell_below(index); | |
752 | new_cell.set_code(payload[i].text); |
|
752 | new_cell.set_code(payload[i].text); | |
753 | this.dirty = true; |
|
753 | this.dirty = true; | |
754 | } |
|
754 | } | |
755 | }; |
|
755 | }; | |
756 | }; |
|
756 | }; | |
757 |
|
757 | |||
758 |
|
758 | |||
759 | Notebook.prototype.handle_iopub_reply = function (e) { |
|
759 | Notebook.prototype.handle_iopub_reply = function (e) { | |
760 | reply = $.parseJSON(e.data); |
|
760 | reply = $.parseJSON(e.data); | |
761 | var content = reply.content; |
|
761 | var content = reply.content; | |
762 | // console.log(reply); |
|
762 | // console.log(reply); | |
763 | var msg_type = reply.header.msg_type; |
|
763 | var msg_type = reply.header.msg_type; | |
764 | var cell = this.cell_for_msg(reply.parent_header.msg_id); |
|
764 | var cell = this.cell_for_msg(reply.parent_header.msg_id); | |
|
765 | if (!cell){ | |||
|
766 | // message not from this notebook | |||
|
767 | console.log("Received IOPub message not caused by one of my cells"); | |||
|
768 | return; | |||
|
769 | } | |||
765 | var output_types = ['stream','display_data','pyout','pyerr']; |
|
770 | var output_types = ['stream','display_data','pyout','pyerr']; | |
766 | if (output_types.indexOf(msg_type) >= 0) { |
|
771 | if (output_types.indexOf(msg_type) >= 0) { | |
767 | this.handle_output(cell, msg_type, content); |
|
772 | this.handle_output(cell, msg_type, content); | |
768 | } else if (msg_type === 'status') { |
|
773 | } else if (msg_type === 'status') { | |
769 | if (content.execution_state === 'busy') { |
|
774 | if (content.execution_state === 'busy') { | |
770 | IPython.kernel_status_widget.status_busy(); |
|
775 | IPython.kernel_status_widget.status_busy(); | |
771 | } else if (content.execution_state === 'idle') { |
|
776 | } else if (content.execution_state === 'idle') { | |
772 | IPython.kernel_status_widget.status_idle(); |
|
777 | IPython.kernel_status_widget.status_idle(); | |
773 | } else if (content.execution_state === 'dead') { |
|
778 | } else if (content.execution_state === 'dead') { | |
774 | this.handle_status_dead(); |
|
779 | this.handle_status_dead(); | |
775 | }; |
|
780 | }; | |
776 | } else if (msg_type === 'clear_output') { |
|
781 | } else if (msg_type === 'clear_output') { | |
777 | cell.clear_output(content.stdout, content.stderr, content.other); |
|
782 | cell.clear_output(content.stdout, content.stderr, content.other); | |
778 | }; |
|
783 | }; | |
779 | }; |
|
784 | }; | |
780 |
|
785 | |||
781 |
|
786 | |||
782 | Notebook.prototype.handle_status_dead = function () { |
|
787 | Notebook.prototype.handle_status_dead = function () { | |
783 | var that = this; |
|
788 | var that = this; | |
784 | this.kernel.stop_channels(); |
|
789 | this.kernel.stop_channels(); | |
785 | var dialog = $('<div/>'); |
|
790 | var dialog = $('<div/>'); | |
786 | 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.'); |
|
791 | 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.'); | |
787 | $(document).append(dialog); |
|
792 | $(document).append(dialog); | |
788 | dialog.dialog({ |
|
793 | dialog.dialog({ | |
789 | resizable: false, |
|
794 | resizable: false, | |
790 | modal: true, |
|
795 | modal: true, | |
791 | title: "Dead kernel", |
|
796 | title: "Dead kernel", | |
792 | buttons : { |
|
797 | buttons : { | |
793 | "Restart": function () { |
|
798 | "Restart": function () { | |
794 | that.start_kernel(); |
|
799 | that.start_kernel(); | |
795 | $(this).dialog('close'); |
|
800 | $(this).dialog('close'); | |
796 | }, |
|
801 | }, | |
797 | "Continue running": function () { |
|
802 | "Continue running": function () { | |
798 | $(this).dialog('close'); |
|
803 | $(this).dialog('close'); | |
799 | } |
|
804 | } | |
800 | } |
|
805 | } | |
801 | }); |
|
806 | }); | |
802 | }; |
|
807 | }; | |
803 |
|
808 | |||
804 |
|
809 | |||
805 | Notebook.prototype.handle_output = function (cell, msg_type, content) { |
|
810 | Notebook.prototype.handle_output = function (cell, msg_type, content) { | |
806 | var json = {}; |
|
811 | var json = {}; | |
807 | json.output_type = msg_type; |
|
812 | json.output_type = msg_type; | |
808 | if (msg_type === "stream") { |
|
813 | if (msg_type === "stream") { | |
809 | json.text = utils.fixConsole(content.data); |
|
814 | json.text = utils.fixConsole(content.data); | |
810 | json.stream = content.name; |
|
815 | json.stream = content.name; | |
811 | } else if (msg_type === "display_data") { |
|
816 | } else if (msg_type === "display_data") { | |
812 | json = this.convert_mime_types(json, content.data); |
|
817 | json = this.convert_mime_types(json, content.data); | |
813 | } else if (msg_type === "pyout") { |
|
818 | } else if (msg_type === "pyout") { | |
814 | json.prompt_number = content.execution_count; |
|
819 | json.prompt_number = content.execution_count; | |
815 | json = this.convert_mime_types(json, content.data); |
|
820 | json = this.convert_mime_types(json, content.data); | |
816 | } else if (msg_type === "pyerr") { |
|
821 | } else if (msg_type === "pyerr") { | |
817 | json.ename = content.ename; |
|
822 | json.ename = content.ename; | |
818 | json.evalue = content.evalue; |
|
823 | json.evalue = content.evalue; | |
819 | var traceback = []; |
|
824 | var traceback = []; | |
820 | for (var i=0; i<content.traceback.length; i++) { |
|
825 | for (var i=0; i<content.traceback.length; i++) { | |
821 | traceback.push(utils.fixConsole(content.traceback[i])); |
|
826 | traceback.push(utils.fixConsole(content.traceback[i])); | |
822 | } |
|
827 | } | |
823 | json.traceback = traceback; |
|
828 | json.traceback = traceback; | |
824 | }; |
|
829 | }; | |
825 | cell.append_output(json); |
|
830 | cell.append_output(json); | |
826 | this.dirty = true; |
|
831 | this.dirty = true; | |
827 | }; |
|
832 | }; | |
828 |
|
833 | |||
829 |
|
834 | |||
830 | Notebook.prototype.convert_mime_types = function (json, data) { |
|
835 | Notebook.prototype.convert_mime_types = function (json, data) { | |
831 | if (data['text/plain'] !== undefined) { |
|
836 | if (data['text/plain'] !== undefined) { | |
832 | json.text = utils.fixConsole(data['text/plain']); |
|
837 | json.text = utils.fixConsole(data['text/plain']); | |
833 | }; |
|
838 | }; | |
834 | if (data['text/html'] !== undefined) { |
|
839 | if (data['text/html'] !== undefined) { | |
835 | json.html = data['text/html']; |
|
840 | json.html = data['text/html']; | |
836 | }; |
|
841 | }; | |
837 | if (data['image/svg+xml'] !== undefined) { |
|
842 | if (data['image/svg+xml'] !== undefined) { | |
838 | json.svg = data['image/svg+xml']; |
|
843 | json.svg = data['image/svg+xml']; | |
839 | }; |
|
844 | }; | |
840 | if (data['image/png'] !== undefined) { |
|
845 | if (data['image/png'] !== undefined) { | |
841 | json.png = data['image/png']; |
|
846 | json.png = data['image/png']; | |
842 | }; |
|
847 | }; | |
843 | if (data['image/jpeg'] !== undefined) { |
|
848 | if (data['image/jpeg'] !== undefined) { | |
844 | json.jpeg = data['image/jpeg']; |
|
849 | json.jpeg = data['image/jpeg']; | |
845 | }; |
|
850 | }; | |
846 | if (data['text/latex'] !== undefined) { |
|
851 | if (data['text/latex'] !== undefined) { | |
847 | json.latex = data['text/latex']; |
|
852 | json.latex = data['text/latex']; | |
848 | }; |
|
853 | }; | |
849 | if (data['application/json'] !== undefined) { |
|
854 | if (data['application/json'] !== undefined) { | |
850 | json.json = data['application/json']; |
|
855 | json.json = data['application/json']; | |
851 | }; |
|
856 | }; | |
852 | if (data['application/javascript'] !== undefined) { |
|
857 | if (data['application/javascript'] !== undefined) { | |
853 | json.javascript = data['application/javascript']; |
|
858 | json.javascript = data['application/javascript']; | |
854 | } |
|
859 | } | |
855 | return json; |
|
860 | return json; | |
856 | }; |
|
861 | }; | |
857 |
|
862 | |||
858 |
|
863 | |||
859 | Notebook.prototype.execute_selected_cell = function (options) { |
|
864 | Notebook.prototype.execute_selected_cell = function (options) { | |
860 | // add_new: should a new cell be added if we are at the end of the nb |
|
865 | // add_new: should a new cell be added if we are at the end of the nb | |
861 | // terminal: execute in terminal mode, which stays in the current cell |
|
866 | // terminal: execute in terminal mode, which stays in the current cell | |
862 | default_options = {terminal: false, add_new: true} |
|
867 | default_options = {terminal: false, add_new: true} | |
863 | $.extend(default_options, options) |
|
868 | $.extend(default_options, options) | |
864 | var that = this; |
|
869 | var that = this; | |
865 | var cell = that.selected_cell(); |
|
870 | var cell = that.selected_cell(); | |
866 | var cell_index = that.find_cell_index(cell); |
|
871 | var cell_index = that.find_cell_index(cell); | |
867 | if (cell instanceof IPython.CodeCell) { |
|
872 | if (cell instanceof IPython.CodeCell) { | |
868 | cell.clear_output(true, true, true); |
|
873 | cell.clear_output(true, true, true); | |
869 | cell.set_input_prompt('*'); |
|
874 | cell.set_input_prompt('*'); | |
870 | cell.element.addClass("running"); |
|
875 | cell.element.addClass("running"); | |
871 | var code = cell.get_code(); |
|
876 | var code = cell.get_code(); | |
872 | var msg_id = that.kernel.execute(cell.get_code()); |
|
877 | var msg_id = that.kernel.execute(cell.get_code()); | |
873 | that.msg_cell_map[msg_id] = cell.cell_id; |
|
878 | that.msg_cell_map[msg_id] = cell.cell_id; | |
874 | } else if (cell instanceof IPython.HTMLCell) { |
|
879 | } else if (cell instanceof IPython.HTMLCell) { | |
875 | cell.render(); |
|
880 | cell.render(); | |
876 | } |
|
881 | } | |
877 | if (default_options.terminal) { |
|
882 | if (default_options.terminal) { | |
878 | cell.select_all(); |
|
883 | cell.select_all(); | |
879 | } else { |
|
884 | } else { | |
880 | if ((cell_index === (that.ncells()-1)) && default_options.add_new) { |
|
885 | if ((cell_index === (that.ncells()-1)) && default_options.add_new) { | |
881 | that.insert_code_cell_below(); |
|
886 | that.insert_code_cell_below(); | |
882 | // If we are adding a new cell at the end, scroll down to show it. |
|
887 | // If we are adding a new cell at the end, scroll down to show it. | |
883 | that.scroll_to_bottom(); |
|
888 | that.scroll_to_bottom(); | |
884 | } else { |
|
889 | } else { | |
885 | that.select(cell_index+1); |
|
890 | that.select(cell_index+1); | |
886 | }; |
|
891 | }; | |
887 | }; |
|
892 | }; | |
888 | this.dirty = true; |
|
893 | this.dirty = true; | |
889 | }; |
|
894 | }; | |
890 |
|
895 | |||
891 |
|
896 | |||
892 | Notebook.prototype.execute_all_cells = function () { |
|
897 | Notebook.prototype.execute_all_cells = function () { | |
893 | var ncells = this.ncells(); |
|
898 | var ncells = this.ncells(); | |
894 | for (var i=0; i<ncells; i++) { |
|
899 | for (var i=0; i<ncells; i++) { | |
895 | this.select(i); |
|
900 | this.select(i); | |
896 | this.execute_selected_cell({add_new:false}); |
|
901 | this.execute_selected_cell({add_new:false}); | |
897 | }; |
|
902 | }; | |
898 | this.scroll_to_bottom(); |
|
903 | this.scroll_to_bottom(); | |
899 | }; |
|
904 | }; | |
900 |
|
905 | |||
901 |
|
906 | |||
902 | Notebook.prototype.request_tool_tip = function (cell,func) { |
|
907 | Notebook.prototype.request_tool_tip = function (cell,func) { | |
903 | // Feel free to shorten this logic if you are better |
|
908 | // Feel free to shorten this logic if you are better | |
904 | // than me in regEx |
|
909 | // than me in regEx | |
905 | // basicaly you shoul be able to get xxx.xxx.xxx from |
|
910 | // basicaly you shoul be able to get xxx.xxx.xxx from | |
906 | // something(range(10), kwarg=smth) ; xxx.xxx.xxx( firstarg, rand(234,23), kwarg1=2, |
|
911 | // something(range(10), kwarg=smth) ; xxx.xxx.xxx( firstarg, rand(234,23), kwarg1=2, | |
907 | // remove everything between matchin bracket (need to iterate) |
|
912 | // remove everything between matchin bracket (need to iterate) | |
908 | matchBracket = /\([^\(\)]+\)/g; |
|
913 | matchBracket = /\([^\(\)]+\)/g; | |
909 | oldfunc = func; |
|
914 | oldfunc = func; | |
910 | func = func.replace(matchBracket,""); |
|
915 | func = func.replace(matchBracket,""); | |
911 | while( oldfunc != func ) |
|
916 | while( oldfunc != func ) | |
912 | { |
|
917 | { | |
913 | oldfunc = func; |
|
918 | oldfunc = func; | |
914 | func = func.replace(matchBracket,""); |
|
919 | func = func.replace(matchBracket,""); | |
915 | } |
|
920 | } | |
916 | // remove everythin after last open bracket |
|
921 | // remove everythin after last open bracket | |
917 | endBracket = /\([^\(]*$/g; |
|
922 | endBracket = /\([^\(]*$/g; | |
918 | func = func.replace(endBracket,""); |
|
923 | func = func.replace(endBracket,""); | |
919 | var re = /[a-zA-Z._]+$/g; |
|
924 | var re = /[a-zA-Z._]+$/g; | |
920 | var msg_id = this.kernel.object_info_request(re.exec(func)); |
|
925 | var msg_id = this.kernel.object_info_request(re.exec(func)); | |
921 | if(typeof(msg_id)!='undefined'){ |
|
926 | if(typeof(msg_id)!='undefined'){ | |
922 | this.msg_cell_map[msg_id] = cell.cell_id; |
|
927 | this.msg_cell_map[msg_id] = cell.cell_id; | |
923 | } |
|
928 | } | |
924 | }; |
|
929 | }; | |
925 |
|
930 | |||
926 | Notebook.prototype.complete_cell = function (cell, line, cursor_pos) { |
|
931 | Notebook.prototype.complete_cell = function (cell, line, cursor_pos) { | |
927 | var msg_id = this.kernel.complete(line, cursor_pos); |
|
932 | var msg_id = this.kernel.complete(line, cursor_pos); | |
928 | this.msg_cell_map[msg_id] = cell.cell_id; |
|
933 | this.msg_cell_map[msg_id] = cell.cell_id; | |
929 | }; |
|
934 | }; | |
930 |
|
935 | |||
931 | // Persistance and loading |
|
936 | // Persistance and loading | |
932 |
|
937 | |||
933 |
|
938 | |||
934 | Notebook.prototype.fromJSON = function (data) { |
|
939 | Notebook.prototype.fromJSON = function (data) { | |
935 | var ncells = this.ncells(); |
|
940 | var ncells = this.ncells(); | |
936 | for (var i=0; i<ncells; i++) { |
|
941 | for (var i=0; i<ncells; i++) { | |
937 | // Always delete cell 0 as they get renumbered as they are deleted. |
|
942 | // Always delete cell 0 as they get renumbered as they are deleted. | |
938 | this.delete_cell(0); |
|
943 | this.delete_cell(0); | |
939 | }; |
|
944 | }; | |
940 | // Save the metadata |
|
945 | // Save the metadata | |
941 | this.metadata = data.metadata; |
|
946 | this.metadata = data.metadata; | |
942 | // Only handle 1 worksheet for now. |
|
947 | // Only handle 1 worksheet for now. | |
943 | var worksheet = data.worksheets[0]; |
|
948 | var worksheet = data.worksheets[0]; | |
944 | if (worksheet !== undefined) { |
|
949 | if (worksheet !== undefined) { | |
945 | var new_cells = worksheet.cells; |
|
950 | var new_cells = worksheet.cells; | |
946 | ncells = new_cells.length; |
|
951 | ncells = new_cells.length; | |
947 | var cell_data = null; |
|
952 | var cell_data = null; | |
948 | var new_cell = null; |
|
953 | var new_cell = null; | |
949 | for (var i=0; i<ncells; i++) { |
|
954 | for (var i=0; i<ncells; i++) { | |
950 | cell_data = new_cells[i]; |
|
955 | cell_data = new_cells[i]; | |
951 | if (cell_data.cell_type == 'code') { |
|
956 | if (cell_data.cell_type == 'code') { | |
952 | new_cell = this.insert_code_cell_below(); |
|
957 | new_cell = this.insert_code_cell_below(); | |
953 | new_cell.fromJSON(cell_data); |
|
958 | new_cell.fromJSON(cell_data); | |
954 | } else if (cell_data.cell_type === 'html') { |
|
959 | } else if (cell_data.cell_type === 'html') { | |
955 | new_cell = this.insert_html_cell_below(); |
|
960 | new_cell = this.insert_html_cell_below(); | |
956 | new_cell.fromJSON(cell_data); |
|
961 | new_cell.fromJSON(cell_data); | |
957 | } else if (cell_data.cell_type === 'markdown') { |
|
962 | } else if (cell_data.cell_type === 'markdown') { | |
958 | new_cell = this.insert_markdown_cell_below(); |
|
963 | new_cell = this.insert_markdown_cell_below(); | |
959 | new_cell.fromJSON(cell_data); |
|
964 | new_cell.fromJSON(cell_data); | |
960 | }; |
|
965 | }; | |
961 | }; |
|
966 | }; | |
962 | }; |
|
967 | }; | |
963 | }; |
|
968 | }; | |
964 |
|
969 | |||
965 |
|
970 | |||
966 | Notebook.prototype.toJSON = function () { |
|
971 | Notebook.prototype.toJSON = function () { | |
967 | var cells = this.cells(); |
|
972 | var cells = this.cells(); | |
968 | var ncells = cells.length; |
|
973 | var ncells = cells.length; | |
969 | cell_array = new Array(ncells); |
|
974 | cell_array = new Array(ncells); | |
970 | for (var i=0; i<ncells; i++) { |
|
975 | for (var i=0; i<ncells; i++) { | |
971 | cell_array[i] = cells[i].toJSON(); |
|
976 | cell_array[i] = cells[i].toJSON(); | |
972 | }; |
|
977 | }; | |
973 | data = { |
|
978 | data = { | |
974 | // Only handle 1 worksheet for now. |
|
979 | // Only handle 1 worksheet for now. | |
975 | worksheets : [{cells:cell_array}], |
|
980 | worksheets : [{cells:cell_array}], | |
976 | metadata : this.metadata |
|
981 | metadata : this.metadata | |
977 | } |
|
982 | } | |
978 | return data |
|
983 | return data | |
979 | }; |
|
984 | }; | |
980 |
|
985 | |||
981 | Notebook.prototype.save_notebook = function () { |
|
986 | Notebook.prototype.save_notebook = function () { | |
982 | if (IPython.save_widget.test_notebook_name()) { |
|
987 | if (IPython.save_widget.test_notebook_name()) { | |
983 | var notebook_id = IPython.save_widget.get_notebook_id(); |
|
988 | var notebook_id = IPython.save_widget.get_notebook_id(); | |
984 | var nbname = IPython.save_widget.get_notebook_name(); |
|
989 | var nbname = IPython.save_widget.get_notebook_name(); | |
985 | // We may want to move the name/id/nbformat logic inside toJSON? |
|
990 | // We may want to move the name/id/nbformat logic inside toJSON? | |
986 | var data = this.toJSON(); |
|
991 | var data = this.toJSON(); | |
987 | data.metadata.name = nbname; |
|
992 | data.metadata.name = nbname; | |
988 | data.nbformat = 2; |
|
993 | data.nbformat = 2; | |
989 | // We do the call with settings so we can set cache to false. |
|
994 | // We do the call with settings so we can set cache to false. | |
990 | var settings = { |
|
995 | var settings = { | |
991 | processData : false, |
|
996 | processData : false, | |
992 | cache : false, |
|
997 | cache : false, | |
993 | type : "PUT", |
|
998 | type : "PUT", | |
994 | data : JSON.stringify(data), |
|
999 | data : JSON.stringify(data), | |
995 | headers : {'Content-Type': 'application/json'}, |
|
1000 | headers : {'Content-Type': 'application/json'}, | |
996 | success : $.proxy(this.notebook_saved,this), |
|
1001 | success : $.proxy(this.notebook_saved,this), | |
997 | error : $.proxy(this.notebook_save_failed,this) |
|
1002 | error : $.proxy(this.notebook_save_failed,this) | |
998 | }; |
|
1003 | }; | |
999 | IPython.save_widget.status_saving(); |
|
1004 | IPython.save_widget.status_saving(); | |
1000 | var url = $('body').data('baseProjectUrl') + 'notebooks/' + notebook_id |
|
1005 | var url = $('body').data('baseProjectUrl') + 'notebooks/' + notebook_id | |
1001 | $.ajax(url, settings); |
|
1006 | $.ajax(url, settings); | |
1002 | }; |
|
1007 | }; | |
1003 | }; |
|
1008 | }; | |
1004 |
|
1009 | |||
1005 |
|
1010 | |||
1006 | Notebook.prototype.notebook_saved = function (data, status, xhr) { |
|
1011 | Notebook.prototype.notebook_saved = function (data, status, xhr) { | |
1007 | this.dirty = false; |
|
1012 | this.dirty = false; | |
1008 | IPython.save_widget.notebook_saved(); |
|
1013 | IPython.save_widget.notebook_saved(); | |
1009 | IPython.save_widget.status_save(); |
|
1014 | IPython.save_widget.status_save(); | |
1010 | } |
|
1015 | } | |
1011 |
|
1016 | |||
1012 |
|
1017 | |||
1013 | Notebook.prototype.notebook_save_failed = function (xhr, status, error_msg) { |
|
1018 | Notebook.prototype.notebook_save_failed = function (xhr, status, error_msg) { | |
1014 | // Notify the user and reset the save button |
|
1019 | // Notify the user and reset the save button | |
1015 | // TODO: Handle different types of errors (timeout etc.) |
|
1020 | // TODO: Handle different types of errors (timeout etc.) | |
1016 | alert('An unexpected error occured while saving the notebook.'); |
|
1021 | alert('An unexpected error occured while saving the notebook.'); | |
1017 | IPython.save_widget.reset_status(); |
|
1022 | IPython.save_widget.reset_status(); | |
1018 | } |
|
1023 | } | |
1019 |
|
1024 | |||
1020 |
|
1025 | |||
1021 | Notebook.prototype.load_notebook = function (callback) { |
|
1026 | Notebook.prototype.load_notebook = function (callback) { | |
1022 | var that = this; |
|
1027 | var that = this; | |
1023 | var notebook_id = IPython.save_widget.get_notebook_id(); |
|
1028 | var notebook_id = IPython.save_widget.get_notebook_id(); | |
1024 | // We do the call with settings so we can set cache to false. |
|
1029 | // We do the call with settings so we can set cache to false. | |
1025 | var settings = { |
|
1030 | var settings = { | |
1026 | processData : false, |
|
1031 | processData : false, | |
1027 | cache : false, |
|
1032 | cache : false, | |
1028 | type : "GET", |
|
1033 | type : "GET", | |
1029 | dataType : "json", |
|
1034 | dataType : "json", | |
1030 | success : function (data, status, xhr) { |
|
1035 | success : function (data, status, xhr) { | |
1031 | that.notebook_loaded(data, status, xhr); |
|
1036 | that.notebook_loaded(data, status, xhr); | |
1032 | if (callback !== undefined) { |
|
1037 | if (callback !== undefined) { | |
1033 | callback(); |
|
1038 | callback(); | |
1034 | }; |
|
1039 | }; | |
1035 | } |
|
1040 | } | |
1036 | }; |
|
1041 | }; | |
1037 | IPython.save_widget.status_loading(); |
|
1042 | IPython.save_widget.status_loading(); | |
1038 | var url = $('body').data('baseProjectUrl') + 'notebooks/' + notebook_id |
|
1043 | var url = $('body').data('baseProjectUrl') + 'notebooks/' + notebook_id | |
1039 | $.ajax(url, settings); |
|
1044 | $.ajax(url, settings); | |
1040 | } |
|
1045 | } | |
1041 |
|
1046 | |||
1042 |
|
1047 | |||
1043 | Notebook.prototype.notebook_loaded = function (data, status, xhr) { |
|
1048 | Notebook.prototype.notebook_loaded = function (data, status, xhr) { | |
1044 | var allowed = xhr.getResponseHeader('Allow'); |
|
1049 | var allowed = xhr.getResponseHeader('Allow'); | |
1045 | this.fromJSON(data); |
|
1050 | this.fromJSON(data); | |
1046 | if (this.ncells() === 0) { |
|
1051 | if (this.ncells() === 0) { | |
1047 | this.insert_code_cell_below(); |
|
1052 | this.insert_code_cell_below(); | |
1048 | }; |
|
1053 | }; | |
1049 | IPython.save_widget.status_save(); |
|
1054 | IPython.save_widget.status_save(); | |
1050 | IPython.save_widget.set_notebook_name(data.metadata.name); |
|
1055 | IPython.save_widget.set_notebook_name(data.metadata.name); | |
1051 | this.dirty = false; |
|
1056 | this.dirty = false; | |
1052 | if (! this.read_only) { |
|
1057 | if (! this.read_only) { | |
1053 | this.start_kernel(); |
|
1058 | this.start_kernel(); | |
1054 | } |
|
1059 | } | |
1055 | // fromJSON always selects the last cell inserted. We need to wait |
|
1060 | // fromJSON always selects the last cell inserted. We need to wait | |
1056 | // until that is done before scrolling to the top. |
|
1061 | // until that is done before scrolling to the top. | |
1057 | setTimeout(function () { |
|
1062 | setTimeout(function () { | |
1058 | IPython.notebook.select(0); |
|
1063 | IPython.notebook.select(0); | |
1059 | IPython.notebook.scroll_to_top(); |
|
1064 | IPython.notebook.scroll_to_top(); | |
1060 | }, 50); |
|
1065 | }, 50); | |
1061 | }; |
|
1066 | }; | |
1062 |
|
1067 | |||
1063 | IPython.Notebook = Notebook; |
|
1068 | IPython.Notebook = Notebook; | |
1064 |
|
1069 | |||
1065 |
|
1070 | |||
1066 | return IPython; |
|
1071 | return IPython; | |
1067 |
|
1072 | |||
1068 | }(IPython)); |
|
1073 | }(IPython)); | |
1069 |
|
1074 |
@@ -1,109 +1,114 | |||||
1 | """ Defines a convenient mix-in class for implementing Qt frontends. |
|
1 | """ Defines a convenient mix-in class for implementing Qt frontends. | |
2 | """ |
|
2 | """ | |
3 |
|
3 | |||
4 | class BaseFrontendMixin(object): |
|
4 | class BaseFrontendMixin(object): | |
5 | """ A mix-in class for implementing Qt frontends. |
|
5 | """ A mix-in class for implementing Qt frontends. | |
6 |
|
6 | |||
7 | To handle messages of a particular type, frontends need only define an |
|
7 | To handle messages of a particular type, frontends need only define an | |
8 | appropriate handler method. For example, to handle 'stream' messaged, define |
|
8 | appropriate handler method. For example, to handle 'stream' messaged, define | |
9 | a '_handle_stream(msg)' method. |
|
9 | a '_handle_stream(msg)' method. | |
10 | """ |
|
10 | """ | |
11 |
|
11 | |||
12 | #--------------------------------------------------------------------------- |
|
12 | #--------------------------------------------------------------------------- | |
13 | # 'BaseFrontendMixin' concrete interface |
|
13 | # 'BaseFrontendMixin' concrete interface | |
14 | #--------------------------------------------------------------------------- |
|
14 | #--------------------------------------------------------------------------- | |
15 |
|
15 | |||
16 | def _get_kernel_manager(self): |
|
16 | def _get_kernel_manager(self): | |
17 | """ Returns the current kernel manager. |
|
17 | """ Returns the current kernel manager. | |
18 | """ |
|
18 | """ | |
19 | return self._kernel_manager |
|
19 | return self._kernel_manager | |
20 |
|
20 | |||
21 | def _set_kernel_manager(self, kernel_manager): |
|
21 | def _set_kernel_manager(self, kernel_manager): | |
22 | """ Disconnect from the current kernel manager (if any) and set a new |
|
22 | """ Disconnect from the current kernel manager (if any) and set a new | |
23 | kernel manager. |
|
23 | kernel manager. | |
24 | """ |
|
24 | """ | |
25 | # Disconnect the old kernel manager, if necessary. |
|
25 | # Disconnect the old kernel manager, if necessary. | |
26 | old_manager = self._kernel_manager |
|
26 | old_manager = self._kernel_manager | |
27 | if old_manager is not None: |
|
27 | if old_manager is not None: | |
28 | old_manager.started_channels.disconnect(self._started_channels) |
|
28 | old_manager.started_channels.disconnect(self._started_channels) | |
29 | old_manager.stopped_channels.disconnect(self._stopped_channels) |
|
29 | old_manager.stopped_channels.disconnect(self._stopped_channels) | |
30 |
|
30 | |||
31 | # Disconnect the old kernel manager's channels. |
|
31 | # Disconnect the old kernel manager's channels. | |
32 | old_manager.sub_channel.message_received.disconnect(self._dispatch) |
|
32 | old_manager.sub_channel.message_received.disconnect(self._dispatch) | |
33 | old_manager.shell_channel.message_received.disconnect(self._dispatch) |
|
33 | old_manager.shell_channel.message_received.disconnect(self._dispatch) | |
34 | old_manager.stdin_channel.message_received.disconnect(self._dispatch) |
|
34 | old_manager.stdin_channel.message_received.disconnect(self._dispatch) | |
35 | old_manager.hb_channel.kernel_died.disconnect( |
|
35 | old_manager.hb_channel.kernel_died.disconnect( | |
36 | self._handle_kernel_died) |
|
36 | self._handle_kernel_died) | |
37 |
|
37 | |||
38 | # Handle the case where the old kernel manager is still listening. |
|
38 | # Handle the case where the old kernel manager is still listening. | |
39 | if old_manager.channels_running: |
|
39 | if old_manager.channels_running: | |
40 | self._stopped_channels() |
|
40 | self._stopped_channels() | |
41 |
|
41 | |||
42 | # Set the new kernel manager. |
|
42 | # Set the new kernel manager. | |
43 | self._kernel_manager = kernel_manager |
|
43 | self._kernel_manager = kernel_manager | |
44 | if kernel_manager is None: |
|
44 | if kernel_manager is None: | |
45 | return |
|
45 | return | |
46 |
|
46 | |||
47 | # Connect the new kernel manager. |
|
47 | # Connect the new kernel manager. | |
48 | kernel_manager.started_channels.connect(self._started_channels) |
|
48 | kernel_manager.started_channels.connect(self._started_channels) | |
49 | kernel_manager.stopped_channels.connect(self._stopped_channels) |
|
49 | kernel_manager.stopped_channels.connect(self._stopped_channels) | |
50 |
|
50 | |||
51 | # Connect the new kernel manager's channels. |
|
51 | # Connect the new kernel manager's channels. | |
52 | kernel_manager.sub_channel.message_received.connect(self._dispatch) |
|
52 | kernel_manager.sub_channel.message_received.connect(self._dispatch) | |
53 | kernel_manager.shell_channel.message_received.connect(self._dispatch) |
|
53 | kernel_manager.shell_channel.message_received.connect(self._dispatch) | |
54 | kernel_manager.stdin_channel.message_received.connect(self._dispatch) |
|
54 | kernel_manager.stdin_channel.message_received.connect(self._dispatch) | |
55 | kernel_manager.hb_channel.kernel_died.connect(self._handle_kernel_died) |
|
55 | kernel_manager.hb_channel.kernel_died.connect(self._handle_kernel_died) | |
56 |
|
56 | |||
57 | # Handle the case where the kernel manager started channels before |
|
57 | # Handle the case where the kernel manager started channels before | |
58 | # we connected. |
|
58 | # we connected. | |
59 | if kernel_manager.channels_running: |
|
59 | if kernel_manager.channels_running: | |
60 | self._started_channels() |
|
60 | self._started_channels() | |
61 |
|
61 | |||
62 | kernel_manager = property(_get_kernel_manager, _set_kernel_manager) |
|
62 | kernel_manager = property(_get_kernel_manager, _set_kernel_manager) | |
63 |
|
63 | |||
64 | #--------------------------------------------------------------------------- |
|
64 | #--------------------------------------------------------------------------- | |
65 | # 'BaseFrontendMixin' abstract interface |
|
65 | # 'BaseFrontendMixin' abstract interface | |
66 | #--------------------------------------------------------------------------- |
|
66 | #--------------------------------------------------------------------------- | |
67 |
|
67 | |||
68 | def _handle_kernel_died(self, since_last_heartbeat): |
|
68 | def _handle_kernel_died(self, since_last_heartbeat): | |
69 | """ This is called when the ``kernel_died`` signal is emitted. |
|
69 | """ This is called when the ``kernel_died`` signal is emitted. | |
70 |
|
70 | |||
71 | This method is called when the kernel heartbeat has not been |
|
71 | This method is called when the kernel heartbeat has not been | |
72 | active for a certain amount of time. The typical action will be to |
|
72 | active for a certain amount of time. The typical action will be to | |
73 | give the user the option of restarting the kernel. |
|
73 | give the user the option of restarting the kernel. | |
74 |
|
74 | |||
75 | Parameters |
|
75 | Parameters | |
76 | ---------- |
|
76 | ---------- | |
77 | since_last_heartbeat : float |
|
77 | since_last_heartbeat : float | |
78 | The time since the heartbeat was last received. |
|
78 | The time since the heartbeat was last received. | |
79 | """ |
|
79 | """ | |
80 |
|
80 | |||
81 | def _started_channels(self): |
|
81 | def _started_channels(self): | |
82 | """ Called when the KernelManager channels have started listening or |
|
82 | """ Called when the KernelManager channels have started listening or | |
83 | when the frontend is assigned an already listening KernelManager. |
|
83 | when the frontend is assigned an already listening KernelManager. | |
84 | """ |
|
84 | """ | |
85 |
|
85 | |||
86 | def _stopped_channels(self): |
|
86 | def _stopped_channels(self): | |
87 | """ Called when the KernelManager channels have stopped listening or |
|
87 | """ Called when the KernelManager channels have stopped listening or | |
88 | when a listening KernelManager is removed from the frontend. |
|
88 | when a listening KernelManager is removed from the frontend. | |
89 | """ |
|
89 | """ | |
90 |
|
90 | |||
91 | #--------------------------------------------------------------------------- |
|
91 | #--------------------------------------------------------------------------- | |
92 | # 'BaseFrontendMixin' protected interface |
|
92 | # 'BaseFrontendMixin' protected interface | |
93 | #--------------------------------------------------------------------------- |
|
93 | #--------------------------------------------------------------------------- | |
94 |
|
94 | |||
95 | def _dispatch(self, msg): |
|
95 | def _dispatch(self, msg): | |
96 | """ Calls the frontend handler associated with the message type of the |
|
96 | """ Calls the frontend handler associated with the message type of the | |
97 | given message. |
|
97 | given message. | |
98 | """ |
|
98 | """ | |
99 | msg_type = msg['header']['msg_type'] |
|
99 | msg_type = msg['header']['msg_type'] | |
100 | handler = getattr(self, '_handle_' + msg_type, None) |
|
100 | handler = getattr(self, '_handle_' + msg_type, None) | |
101 | if handler: |
|
101 | if handler: | |
102 | handler(msg) |
|
102 | handler(msg) | |
103 |
|
103 | |||
104 | def _is_from_this_session(self, msg): |
|
104 | def _is_from_this_session(self, msg): | |
105 | """ Returns whether a reply from the kernel originated from a request |
|
105 | """ Returns whether a reply from the kernel originated from a request | |
106 | from this frontend. |
|
106 | from this frontend. | |
107 | """ |
|
107 | """ | |
108 | session = self._kernel_manager.session.session |
|
108 | session = self._kernel_manager.session.session | |
109 |
re |
|
109 | parent = msg['parent_header'] | |
|
110 | if not parent: | |||
|
111 | # if the message has no parent, assume it is meant for all frontends | |||
|
112 | return True | |||
|
113 | else: | |||
|
114 | return parent.get('session') == session |
@@ -1,322 +1,325 | |||||
1 | # -*- coding: utf-8 -*- |
|
1 | # -*- coding: utf-8 -*- | |
2 | """Pylab (matplotlib) support utilities. |
|
2 | """Pylab (matplotlib) support utilities. | |
3 |
|
3 | |||
4 | Authors |
|
4 | Authors | |
5 | ------- |
|
5 | ------- | |
6 |
|
6 | |||
7 | * Fernando Perez. |
|
7 | * Fernando Perez. | |
8 | * Brian Granger |
|
8 | * Brian Granger | |
9 | """ |
|
9 | """ | |
10 |
|
10 | |||
11 | #----------------------------------------------------------------------------- |
|
11 | #----------------------------------------------------------------------------- | |
12 | # Copyright (C) 2009-2011 The IPython Development Team |
|
12 | # Copyright (C) 2009-2011 The IPython Development Team | |
13 | # |
|
13 | # | |
14 | # Distributed under the terms of the BSD License. The full license is in |
|
14 | # Distributed under the terms of the BSD License. The full license is in | |
15 | # the file COPYING, distributed as part of this software. |
|
15 | # the file COPYING, distributed as part of this software. | |
16 | #----------------------------------------------------------------------------- |
|
16 | #----------------------------------------------------------------------------- | |
17 |
|
17 | |||
18 | #----------------------------------------------------------------------------- |
|
18 | #----------------------------------------------------------------------------- | |
19 | # Imports |
|
19 | # Imports | |
20 | #----------------------------------------------------------------------------- |
|
20 | #----------------------------------------------------------------------------- | |
21 |
|
21 | |||
|
22 | import sys | |||
22 | from io import BytesIO |
|
23 | from io import BytesIO | |
23 |
|
24 | |||
24 | from IPython.utils.decorators import flag_calls |
|
25 | from IPython.utils.decorators import flag_calls | |
25 |
|
26 | |||
26 | # If user specifies a GUI, that dictates the backend, otherwise we read the |
|
27 | # If user specifies a GUI, that dictates the backend, otherwise we read the | |
27 | # user's mpl default from the mpl rc structure |
|
28 | # user's mpl default from the mpl rc structure | |
28 | backends = {'tk': 'TkAgg', |
|
29 | backends = {'tk': 'TkAgg', | |
29 | 'gtk': 'GTKAgg', |
|
30 | 'gtk': 'GTKAgg', | |
30 | 'wx': 'WXAgg', |
|
31 | 'wx': 'WXAgg', | |
31 | 'qt': 'Qt4Agg', # qt3 not supported |
|
32 | 'qt': 'Qt4Agg', # qt3 not supported | |
32 | 'qt4': 'Qt4Agg', |
|
33 | 'qt4': 'Qt4Agg', | |
33 | 'osx': 'MacOSX', |
|
34 | 'osx': 'MacOSX', | |
34 | 'inline' : 'module://IPython.zmq.pylab.backend_inline'} |
|
35 | 'inline' : 'module://IPython.zmq.pylab.backend_inline'} | |
35 |
|
36 | |||
36 | # We also need a reverse backends2guis mapping that will properly choose which |
|
37 | # We also need a reverse backends2guis mapping that will properly choose which | |
37 | # GUI support to activate based on the desired matplotlib backend. For the |
|
38 | # GUI support to activate based on the desired matplotlib backend. For the | |
38 | # most part it's just a reverse of the above dict, but we also need to add a |
|
39 | # most part it's just a reverse of the above dict, but we also need to add a | |
39 | # few others that map to the same GUI manually: |
|
40 | # few others that map to the same GUI manually: | |
40 | backend2gui = dict(zip(backends.values(), backends.keys())) |
|
41 | backend2gui = dict(zip(backends.values(), backends.keys())) | |
41 | # In the reverse mapping, there are a few extra valid matplotlib backends that |
|
42 | # In the reverse mapping, there are a few extra valid matplotlib backends that | |
42 | # map to the same GUI support |
|
43 | # map to the same GUI support | |
43 | backend2gui['GTK'] = backend2gui['GTKCairo'] = 'gtk' |
|
44 | backend2gui['GTK'] = backend2gui['GTKCairo'] = 'gtk' | |
44 | backend2gui['WX'] = 'wx' |
|
45 | backend2gui['WX'] = 'wx' | |
45 | backend2gui['CocoaAgg'] = 'osx' |
|
46 | backend2gui['CocoaAgg'] = 'osx' | |
46 |
|
47 | |||
47 | #----------------------------------------------------------------------------- |
|
48 | #----------------------------------------------------------------------------- | |
48 | # Matplotlib utilities |
|
49 | # Matplotlib utilities | |
49 | #----------------------------------------------------------------------------- |
|
50 | #----------------------------------------------------------------------------- | |
50 |
|
51 | |||
51 |
|
52 | |||
52 | def getfigs(*fig_nums): |
|
53 | def getfigs(*fig_nums): | |
53 | """Get a list of matplotlib figures by figure numbers. |
|
54 | """Get a list of matplotlib figures by figure numbers. | |
54 |
|
55 | |||
55 | If no arguments are given, all available figures are returned. If the |
|
56 | If no arguments are given, all available figures are returned. If the | |
56 | argument list contains references to invalid figures, a warning is printed |
|
57 | argument list contains references to invalid figures, a warning is printed | |
57 | but the function continues pasting further figures. |
|
58 | but the function continues pasting further figures. | |
58 |
|
59 | |||
59 | Parameters |
|
60 | Parameters | |
60 | ---------- |
|
61 | ---------- | |
61 | figs : tuple |
|
62 | figs : tuple | |
62 | A tuple of ints giving the figure numbers of the figures to return. |
|
63 | A tuple of ints giving the figure numbers of the figures to return. | |
63 | """ |
|
64 | """ | |
64 | from matplotlib._pylab_helpers import Gcf |
|
65 | from matplotlib._pylab_helpers import Gcf | |
65 | if not fig_nums: |
|
66 | if not fig_nums: | |
66 | fig_managers = Gcf.get_all_fig_managers() |
|
67 | fig_managers = Gcf.get_all_fig_managers() | |
67 | return [fm.canvas.figure for fm in fig_managers] |
|
68 | return [fm.canvas.figure for fm in fig_managers] | |
68 | else: |
|
69 | else: | |
69 | figs = [] |
|
70 | figs = [] | |
70 | for num in fig_nums: |
|
71 | for num in fig_nums: | |
71 | f = Gcf.figs.get(num) |
|
72 | f = Gcf.figs.get(num) | |
72 | if f is None: |
|
73 | if f is None: | |
73 | print('Warning: figure %s not available.' % num) |
|
74 | print('Warning: figure %s not available.' % num) | |
74 | else: |
|
75 | else: | |
75 | figs.append(f.canvas.figure) |
|
76 | figs.append(f.canvas.figure) | |
76 | return figs |
|
77 | return figs | |
77 |
|
78 | |||
78 |
|
79 | |||
79 | def figsize(sizex, sizey): |
|
80 | def figsize(sizex, sizey): | |
80 | """Set the default figure size to be [sizex, sizey]. |
|
81 | """Set the default figure size to be [sizex, sizey]. | |
81 |
|
82 | |||
82 | This is just an easy to remember, convenience wrapper that sets:: |
|
83 | This is just an easy to remember, convenience wrapper that sets:: | |
83 |
|
84 | |||
84 | matplotlib.rcParams['figure.figsize'] = [sizex, sizey] |
|
85 | matplotlib.rcParams['figure.figsize'] = [sizex, sizey] | |
85 | """ |
|
86 | """ | |
86 | import matplotlib |
|
87 | import matplotlib | |
87 | matplotlib.rcParams['figure.figsize'] = [sizex, sizey] |
|
88 | matplotlib.rcParams['figure.figsize'] = [sizex, sizey] | |
88 |
|
89 | |||
89 |
|
90 | |||
90 | def print_figure(fig, fmt='png'): |
|
91 | def print_figure(fig, fmt='png'): | |
91 | """Convert a figure to svg or png for inline display.""" |
|
92 | """Convert a figure to svg or png for inline display.""" | |
92 | # When there's an empty figure, we shouldn't return anything, otherwise we |
|
93 | # When there's an empty figure, we shouldn't return anything, otherwise we | |
93 | # get big blank areas in the qt console. |
|
94 | # get big blank areas in the qt console. | |
94 | if not fig.axes: |
|
95 | if not fig.axes: | |
95 | return |
|
96 | return | |
96 |
|
97 | |||
97 | fc = fig.get_facecolor() |
|
98 | fc = fig.get_facecolor() | |
98 | ec = fig.get_edgecolor() |
|
99 | ec = fig.get_edgecolor() | |
99 | fig.set_facecolor('white') |
|
100 | fig.set_facecolor('white') | |
100 | fig.set_edgecolor('white') |
|
101 | fig.set_edgecolor('white') | |
101 | try: |
|
102 | try: | |
102 | bytes_io = BytesIO() |
|
103 | bytes_io = BytesIO() | |
103 | fig.canvas.print_figure(bytes_io, format=fmt, bbox_inches='tight') |
|
104 | fig.canvas.print_figure(bytes_io, format=fmt, bbox_inches='tight') | |
104 | data = bytes_io.getvalue() |
|
105 | data = bytes_io.getvalue() | |
105 | finally: |
|
106 | finally: | |
106 | fig.set_facecolor(fc) |
|
107 | fig.set_facecolor(fc) | |
107 | fig.set_edgecolor(ec) |
|
108 | fig.set_edgecolor(ec) | |
108 | return data |
|
109 | return data | |
109 |
|
110 | |||
110 |
|
111 | |||
111 | # We need a little factory function here to create the closure where |
|
112 | # We need a little factory function here to create the closure where | |
112 | # safe_execfile can live. |
|
113 | # safe_execfile can live. | |
113 | def mpl_runner(safe_execfile): |
|
114 | def mpl_runner(safe_execfile): | |
114 | """Factory to return a matplotlib-enabled runner for %run. |
|
115 | """Factory to return a matplotlib-enabled runner for %run. | |
115 |
|
116 | |||
116 | Parameters |
|
117 | Parameters | |
117 | ---------- |
|
118 | ---------- | |
118 | safe_execfile : function |
|
119 | safe_execfile : function | |
119 | This must be a function with the same interface as the |
|
120 | This must be a function with the same interface as the | |
120 | :meth:`safe_execfile` method of IPython. |
|
121 | :meth:`safe_execfile` method of IPython. | |
121 |
|
122 | |||
122 | Returns |
|
123 | Returns | |
123 | ------- |
|
124 | ------- | |
124 | A function suitable for use as the ``runner`` argument of the %run magic |
|
125 | A function suitable for use as the ``runner`` argument of the %run magic | |
125 | function. |
|
126 | function. | |
126 | """ |
|
127 | """ | |
127 |
|
128 | |||
128 | def mpl_execfile(fname,*where,**kw): |
|
129 | def mpl_execfile(fname,*where,**kw): | |
129 | """matplotlib-aware wrapper around safe_execfile. |
|
130 | """matplotlib-aware wrapper around safe_execfile. | |
130 |
|
131 | |||
131 | Its interface is identical to that of the :func:`execfile` builtin. |
|
132 | Its interface is identical to that of the :func:`execfile` builtin. | |
132 |
|
133 | |||
133 | This is ultimately a call to execfile(), but wrapped in safeties to |
|
134 | This is ultimately a call to execfile(), but wrapped in safeties to | |
134 | properly handle interactive rendering.""" |
|
135 | properly handle interactive rendering.""" | |
135 |
|
136 | |||
136 | import matplotlib |
|
137 | import matplotlib | |
137 | import matplotlib.pylab as pylab |
|
138 | import matplotlib.pylab as pylab | |
138 |
|
139 | |||
139 | #print '*** Matplotlib runner ***' # dbg |
|
140 | #print '*** Matplotlib runner ***' # dbg | |
140 | # turn off rendering until end of script |
|
141 | # turn off rendering until end of script | |
141 | is_interactive = matplotlib.rcParams['interactive'] |
|
142 | is_interactive = matplotlib.rcParams['interactive'] | |
142 | matplotlib.interactive(False) |
|
143 | matplotlib.interactive(False) | |
143 | safe_execfile(fname,*where,**kw) |
|
144 | safe_execfile(fname,*where,**kw) | |
144 | matplotlib.interactive(is_interactive) |
|
145 | matplotlib.interactive(is_interactive) | |
145 | # make rendering call now, if the user tried to do it |
|
146 | # make rendering call now, if the user tried to do it | |
146 | if pylab.draw_if_interactive.called: |
|
147 | if pylab.draw_if_interactive.called: | |
147 | pylab.draw() |
|
148 | pylab.draw() | |
148 | pylab.draw_if_interactive.called = False |
|
149 | pylab.draw_if_interactive.called = False | |
149 |
|
150 | |||
150 | return mpl_execfile |
|
151 | return mpl_execfile | |
151 |
|
152 | |||
152 |
|
153 | |||
153 | def select_figure_format(shell, fmt): |
|
154 | def select_figure_format(shell, fmt): | |
154 | """Select figure format for inline backend, either 'png' or 'svg'. |
|
155 | """Select figure format for inline backend, either 'png' or 'svg'. | |
155 |
|
156 | |||
156 | Using this method ensures only one figure format is active at a time. |
|
157 | Using this method ensures only one figure format is active at a time. | |
157 | """ |
|
158 | """ | |
158 | from matplotlib.figure import Figure |
|
159 | from matplotlib.figure import Figure | |
159 | from IPython.zmq.pylab import backend_inline |
|
160 | from IPython.zmq.pylab import backend_inline | |
160 |
|
161 | |||
161 | svg_formatter = shell.display_formatter.formatters['image/svg+xml'] |
|
162 | svg_formatter = shell.display_formatter.formatters['image/svg+xml'] | |
162 | png_formatter = shell.display_formatter.formatters['image/png'] |
|
163 | png_formatter = shell.display_formatter.formatters['image/png'] | |
163 |
|
164 | |||
164 | if fmt=='png': |
|
165 | if fmt=='png': | |
165 | svg_formatter.type_printers.pop(Figure, None) |
|
166 | svg_formatter.type_printers.pop(Figure, None) | |
166 | png_formatter.for_type(Figure, lambda fig: print_figure(fig, 'png')) |
|
167 | png_formatter.for_type(Figure, lambda fig: print_figure(fig, 'png')) | |
167 | elif fmt=='svg': |
|
168 | elif fmt=='svg': | |
168 | png_formatter.type_printers.pop(Figure, None) |
|
169 | png_formatter.type_printers.pop(Figure, None) | |
169 | svg_formatter.for_type(Figure, lambda fig: print_figure(fig, 'svg')) |
|
170 | svg_formatter.for_type(Figure, lambda fig: print_figure(fig, 'svg')) | |
170 | else: |
|
171 | else: | |
171 | raise ValueError("supported formats are: 'png', 'svg', not %r"%fmt) |
|
172 | raise ValueError("supported formats are: 'png', 'svg', not %r"%fmt) | |
172 |
|
173 | |||
173 | # set the format to be used in the backend() |
|
174 | # set the format to be used in the backend() | |
174 | backend_inline._figure_format = fmt |
|
175 | backend_inline._figure_format = fmt | |
175 |
|
176 | |||
176 | #----------------------------------------------------------------------------- |
|
177 | #----------------------------------------------------------------------------- | |
177 | # Code for initializing matplotlib and importing pylab |
|
178 | # Code for initializing matplotlib and importing pylab | |
178 | #----------------------------------------------------------------------------- |
|
179 | #----------------------------------------------------------------------------- | |
179 |
|
180 | |||
180 |
|
181 | |||
181 | def find_gui_and_backend(gui=None): |
|
182 | def find_gui_and_backend(gui=None): | |
182 | """Given a gui string return the gui and mpl backend. |
|
183 | """Given a gui string return the gui and mpl backend. | |
183 |
|
184 | |||
184 | Parameters |
|
185 | Parameters | |
185 | ---------- |
|
186 | ---------- | |
186 | gui : str |
|
187 | gui : str | |
187 | Can be one of ('tk','gtk','wx','qt','qt4','inline'). |
|
188 | Can be one of ('tk','gtk','wx','qt','qt4','inline'). | |
188 |
|
189 | |||
189 | Returns |
|
190 | Returns | |
190 | ------- |
|
191 | ------- | |
191 | A tuple of (gui, backend) where backend is one of ('TkAgg','GTKAgg', |
|
192 | A tuple of (gui, backend) where backend is one of ('TkAgg','GTKAgg', | |
192 | 'WXAgg','Qt4Agg','module://IPython.zmq.pylab.backend_inline'). |
|
193 | 'WXAgg','Qt4Agg','module://IPython.zmq.pylab.backend_inline'). | |
193 | """ |
|
194 | """ | |
194 |
|
195 | |||
195 | import matplotlib |
|
196 | import matplotlib | |
196 |
|
197 | |||
197 | if gui and gui != 'auto': |
|
198 | if gui and gui != 'auto': | |
198 | # select backend based on requested gui |
|
199 | # select backend based on requested gui | |
199 | backend = backends[gui] |
|
200 | backend = backends[gui] | |
200 | else: |
|
201 | else: | |
201 | backend = matplotlib.rcParams['backend'] |
|
202 | backend = matplotlib.rcParams['backend'] | |
202 | # In this case, we need to find what the appropriate gui selection call |
|
203 | # In this case, we need to find what the appropriate gui selection call | |
203 | # should be for IPython, so we can activate inputhook accordingly |
|
204 | # should be for IPython, so we can activate inputhook accordingly | |
204 | gui = backend2gui.get(backend, None) |
|
205 | gui = backend2gui.get(backend, None) | |
205 | return gui, backend |
|
206 | return gui, backend | |
206 |
|
207 | |||
207 |
|
208 | |||
208 | def activate_matplotlib(backend): |
|
209 | def activate_matplotlib(backend): | |
209 | """Activate the given backend and set interactive to True.""" |
|
210 | """Activate the given backend and set interactive to True.""" | |
210 |
|
211 | |||
211 | import matplotlib |
|
212 | import matplotlib | |
212 | if backend.startswith('module://'): |
|
213 | if backend.startswith('module://'): | |
213 | # Work around bug in matplotlib: matplotlib.use converts the |
|
214 | # Work around bug in matplotlib: matplotlib.use converts the | |
214 | # backend_id to lowercase even if a module name is specified! |
|
215 | # backend_id to lowercase even if a module name is specified! | |
215 | matplotlib.rcParams['backend'] = backend |
|
216 | matplotlib.rcParams['backend'] = backend | |
216 | else: |
|
217 | else: | |
217 | matplotlib.use(backend) |
|
218 | matplotlib.use(backend) | |
218 | matplotlib.interactive(True) |
|
219 | matplotlib.interactive(True) | |
219 |
|
220 | |||
220 | # This must be imported last in the matplotlib series, after |
|
221 | # This must be imported last in the matplotlib series, after | |
221 | # backend/interactivity choices have been made |
|
222 | # backend/interactivity choices have been made | |
222 | import matplotlib.pylab as pylab |
|
223 | import matplotlib.pylab as pylab | |
223 |
|
224 | |||
224 | # XXX For now leave this commented out, but depending on discussions with |
|
225 | # XXX For now leave this commented out, but depending on discussions with | |
225 | # mpl-dev, we may be able to allow interactive switching... |
|
226 | # mpl-dev, we may be able to allow interactive switching... | |
226 | #import matplotlib.pyplot |
|
227 | #import matplotlib.pyplot | |
227 | #matplotlib.pyplot.switch_backend(backend) |
|
228 | #matplotlib.pyplot.switch_backend(backend) | |
228 |
|
229 | |||
229 | pylab.show._needmain = False |
|
230 | pylab.show._needmain = False | |
230 | # We need to detect at runtime whether show() is called by the user. |
|
231 | # We need to detect at runtime whether show() is called by the user. | |
231 | # For this, we wrap it into a decorator which adds a 'called' flag. |
|
232 | # For this, we wrap it into a decorator which adds a 'called' flag. | |
232 | pylab.draw_if_interactive = flag_calls(pylab.draw_if_interactive) |
|
233 | pylab.draw_if_interactive = flag_calls(pylab.draw_if_interactive) | |
233 |
|
234 | |||
234 | def import_pylab(user_ns, backend, import_all=True, shell=None): |
|
235 | def import_pylab(user_ns, backend, import_all=True, shell=None): | |
235 | """Import the standard pylab symbols into user_ns.""" |
|
236 | """Import the standard pylab symbols into user_ns.""" | |
236 |
|
237 | |||
237 | # Import numpy as np/pyplot as plt are conventions we're trying to |
|
238 | # Import numpy as np/pyplot as plt are conventions we're trying to | |
238 | # somewhat standardize on. Making them available to users by default |
|
239 | # somewhat standardize on. Making them available to users by default | |
239 | # will greatly help this. |
|
240 | # will greatly help this. | |
240 | s = ("import numpy\n" |
|
241 | s = ("import numpy\n" | |
241 | "import matplotlib\n" |
|
242 | "import matplotlib\n" | |
242 | "from matplotlib import pylab, mlab, pyplot\n" |
|
243 | "from matplotlib import pylab, mlab, pyplot\n" | |
243 | "np = numpy\n" |
|
244 | "np = numpy\n" | |
244 | "plt = pyplot\n" |
|
245 | "plt = pyplot\n" | |
245 | ) |
|
246 | ) | |
246 | exec s in user_ns |
|
247 | exec s in user_ns | |
247 |
|
248 | |||
248 | if shell is not None: |
|
249 | if shell is not None: | |
249 | exec s in shell.user_ns_hidden |
|
250 | exec s in shell.user_ns_hidden | |
250 | # If using our svg payload backend, register the post-execution |
|
251 | # If using our svg payload backend, register the post-execution | |
251 | # function that will pick up the results for display. This can only be |
|
252 | # function that will pick up the results for display. This can only be | |
252 | # done with access to the real shell object. |
|
253 | # done with access to the real shell object. | |
253 | # |
|
254 | # | |
254 | from IPython.zmq.pylab.backend_inline import InlineBackend |
|
255 | from IPython.zmq.pylab.backend_inline import InlineBackend | |
255 |
|
256 | |||
256 | cfg = InlineBackend.instance(config=shell.config) |
|
257 | cfg = InlineBackend.instance(config=shell.config) | |
257 | cfg.shell = shell |
|
258 | cfg.shell = shell | |
258 | if cfg not in shell.configurables: |
|
259 | if cfg not in shell.configurables: | |
259 | shell.configurables.append(cfg) |
|
260 | shell.configurables.append(cfg) | |
260 |
|
261 | |||
261 | if backend == backends['inline']: |
|
262 | if backend == backends['inline']: | |
262 | from IPython.zmq.pylab.backend_inline import flush_figures |
|
263 | from IPython.zmq.pylab.backend_inline import flush_figures | |
263 | from matplotlib import pyplot |
|
264 | from matplotlib import pyplot | |
264 | shell.register_post_execute(flush_figures) |
|
265 | shell.register_post_execute(flush_figures) | |
265 | # load inline_rc |
|
266 | # load inline_rc | |
266 | pyplot.rcParams.update(cfg.rc) |
|
267 | pyplot.rcParams.update(cfg.rc) | |
267 |
|
268 | |||
268 | # Add 'figsize' to pyplot and to the user's namespace |
|
269 | # Add 'figsize' to pyplot and to the user's namespace | |
269 | user_ns['figsize'] = pyplot.figsize = figsize |
|
270 | user_ns['figsize'] = pyplot.figsize = figsize | |
270 | shell.user_ns_hidden['figsize'] = figsize |
|
271 | shell.user_ns_hidden['figsize'] = figsize | |
271 |
|
272 | |||
272 | # Setup the default figure format |
|
273 | # Setup the default figure format | |
273 | fmt = cfg.figure_format |
|
274 | fmt = cfg.figure_format | |
274 | select_figure_format(shell, fmt) |
|
275 | select_figure_format(shell, fmt) | |
275 |
|
276 | |||
276 | # The old pastefig function has been replaced by display |
|
277 | # The old pastefig function has been replaced by display | |
277 | from IPython.core.display import display |
|
278 | from IPython.core.display import display | |
278 | # Add display and display_png to the user's namespace |
|
279 | # Add display and display_png to the user's namespace | |
279 | user_ns['display'] = display |
|
280 | user_ns['display'] = display | |
280 | shell.user_ns_hidden['display'] = display |
|
281 | shell.user_ns_hidden['display'] = display | |
281 | user_ns['getfigs'] = getfigs |
|
282 | user_ns['getfigs'] = getfigs | |
282 | shell.user_ns_hidden['getfigs'] = getfigs |
|
283 | shell.user_ns_hidden['getfigs'] = getfigs | |
283 |
|
284 | |||
284 | if import_all: |
|
285 | if import_all: | |
285 | s = ("from matplotlib.pylab import *\n" |
|
286 | s = ("from matplotlib.pylab import *\n" | |
286 | "from numpy import *\n") |
|
287 | "from numpy import *\n") | |
287 | exec s in user_ns |
|
288 | exec s in user_ns | |
288 | if shell is not None: |
|
289 | if shell is not None: | |
289 | exec s in shell.user_ns_hidden |
|
290 | exec s in shell.user_ns_hidden | |
290 |
|
291 | |||
291 |
|
292 | |||
292 | def pylab_activate(user_ns, gui=None, import_all=True, shell=None): |
|
293 | def pylab_activate(user_ns, gui=None, import_all=True, shell=None): | |
293 | """Activate pylab mode in the user's namespace. |
|
294 | """Activate pylab mode in the user's namespace. | |
294 |
|
295 | |||
295 | Loads and initializes numpy, matplotlib and friends for interactive use. |
|
296 | Loads and initializes numpy, matplotlib and friends for interactive use. | |
296 |
|
297 | |||
297 | Parameters |
|
298 | Parameters | |
298 | ---------- |
|
299 | ---------- | |
299 | user_ns : dict |
|
300 | user_ns : dict | |
300 | Namespace where the imports will occur. |
|
301 | Namespace where the imports will occur. | |
301 |
|
302 | |||
302 | gui : optional, string |
|
303 | gui : optional, string | |
303 | A valid gui name following the conventions of the %gui magic. |
|
304 | A valid gui name following the conventions of the %gui magic. | |
304 |
|
305 | |||
305 | import_all : optional, boolean |
|
306 | import_all : optional, boolean | |
306 | If true, an 'import *' is done from numpy and pylab. |
|
307 | If true, an 'import *' is done from numpy and pylab. | |
307 |
|
308 | |||
308 | Returns |
|
309 | Returns | |
309 | ------- |
|
310 | ------- | |
310 | The actual gui used (if not given as input, it was obtained from matplotlib |
|
311 | The actual gui used (if not given as input, it was obtained from matplotlib | |
311 | itself, and will be needed next to configure IPython's gui integration. |
|
312 | itself, and will be needed next to configure IPython's gui integration. | |
312 | """ |
|
313 | """ | |
313 | gui, backend = find_gui_and_backend(gui) |
|
314 | gui, backend = find_gui_and_backend(gui) | |
314 | activate_matplotlib(backend) |
|
315 | activate_matplotlib(backend) | |
315 | import_pylab(user_ns, backend, import_all, shell) |
|
316 | import_pylab(user_ns, backend, import_all, shell) | |
316 |
|
317 | |||
317 | print """ |
|
318 | print """ | |
318 | Welcome to pylab, a matplotlib-based Python environment [backend: %s]. |
|
319 | Welcome to pylab, a matplotlib-based Python environment [backend: %s]. | |
319 | For more information, type 'help(pylab)'.""" % backend |
|
320 | For more information, type 'help(pylab)'.""" % backend | |
|
321 | # flush stdout, just to be safe | |||
|
322 | sys.stdout.flush() | |||
320 |
|
323 | |||
321 | return gui |
|
324 | return gui | |
322 |
|
325 |
@@ -1,299 +1,303 | |||||
1 | """An Application for launching a kernel |
|
1 | """An Application for launching a kernel | |
2 |
|
2 | |||
3 | Authors |
|
3 | Authors | |
4 | ------- |
|
4 | ------- | |
5 | * MinRK |
|
5 | * MinRK | |
6 | """ |
|
6 | """ | |
7 | #----------------------------------------------------------------------------- |
|
7 | #----------------------------------------------------------------------------- | |
8 | # Copyright (C) 2011 The IPython Development Team |
|
8 | # Copyright (C) 2011 The IPython Development Team | |
9 | # |
|
9 | # | |
10 | # Distributed under the terms of the BSD License. The full license is in |
|
10 | # Distributed under the terms of the BSD License. The full license is in | |
11 | # the file COPYING.txt, distributed as part of this software. |
|
11 | # the file COPYING.txt, distributed as part of this software. | |
12 | #----------------------------------------------------------------------------- |
|
12 | #----------------------------------------------------------------------------- | |
13 |
|
13 | |||
14 | #----------------------------------------------------------------------------- |
|
14 | #----------------------------------------------------------------------------- | |
15 | # Imports |
|
15 | # Imports | |
16 | #----------------------------------------------------------------------------- |
|
16 | #----------------------------------------------------------------------------- | |
17 |
|
17 | |||
18 | # Standard library imports. |
|
18 | # Standard library imports. | |
19 | import json |
|
19 | import json | |
20 | import os |
|
20 | import os | |
21 | import sys |
|
21 | import sys | |
22 |
|
22 | |||
23 | # System library imports. |
|
23 | # System library imports. | |
24 | import zmq |
|
24 | import zmq | |
25 |
|
25 | |||
26 | # IPython imports. |
|
26 | # IPython imports. | |
27 | from IPython.core.ultratb import FormattedTB |
|
27 | from IPython.core.ultratb import FormattedTB | |
28 | from IPython.core.application import ( |
|
28 | from IPython.core.application import ( | |
29 | BaseIPythonApplication, base_flags, base_aliases, catch_config_error |
|
29 | BaseIPythonApplication, base_flags, base_aliases, catch_config_error | |
30 | ) |
|
30 | ) | |
31 | from IPython.utils import io |
|
31 | from IPython.utils import io | |
32 | from IPython.utils.localinterfaces import LOCALHOST |
|
32 | from IPython.utils.localinterfaces import LOCALHOST | |
33 | from IPython.utils.path import filefind |
|
33 | from IPython.utils.path import filefind | |
34 | from IPython.utils.py3compat import str_to_bytes |
|
34 | from IPython.utils.py3compat import str_to_bytes | |
35 | from IPython.utils.traitlets import (Any, Instance, Dict, Unicode, Integer, Bool, |
|
35 | from IPython.utils.traitlets import (Any, Instance, Dict, Unicode, Integer, Bool, | |
36 | DottedObjectName) |
|
36 | DottedObjectName) | |
37 | from IPython.utils.importstring import import_item |
|
37 | from IPython.utils.importstring import import_item | |
38 | # local imports |
|
38 | # local imports | |
39 | from IPython.zmq.entry_point import write_connection_file |
|
39 | from IPython.zmq.entry_point import write_connection_file | |
40 | from IPython.zmq.heartbeat import Heartbeat |
|
40 | from IPython.zmq.heartbeat import Heartbeat | |
41 | from IPython.zmq.parentpoller import ParentPollerUnix, ParentPollerWindows |
|
41 | from IPython.zmq.parentpoller import ParentPollerUnix, ParentPollerWindows | |
42 | from IPython.zmq.session import ( |
|
42 | from IPython.zmq.session import ( | |
43 | Session, session_flags, session_aliases, default_secure, |
|
43 | Session, session_flags, session_aliases, default_secure, | |
44 | ) |
|
44 | ) | |
45 |
|
45 | |||
46 |
|
46 | |||
47 | #----------------------------------------------------------------------------- |
|
47 | #----------------------------------------------------------------------------- | |
48 | # Flags and Aliases |
|
48 | # Flags and Aliases | |
49 | #----------------------------------------------------------------------------- |
|
49 | #----------------------------------------------------------------------------- | |
50 |
|
50 | |||
51 | kernel_aliases = dict(base_aliases) |
|
51 | kernel_aliases = dict(base_aliases) | |
52 | kernel_aliases.update({ |
|
52 | kernel_aliases.update({ | |
53 | 'ip' : 'KernelApp.ip', |
|
53 | 'ip' : 'KernelApp.ip', | |
54 | 'hb' : 'KernelApp.hb_port', |
|
54 | 'hb' : 'KernelApp.hb_port', | |
55 | 'shell' : 'KernelApp.shell_port', |
|
55 | 'shell' : 'KernelApp.shell_port', | |
56 | 'iopub' : 'KernelApp.iopub_port', |
|
56 | 'iopub' : 'KernelApp.iopub_port', | |
57 | 'stdin' : 'KernelApp.stdin_port', |
|
57 | 'stdin' : 'KernelApp.stdin_port', | |
58 | 'f' : 'KernelApp.connection_file', |
|
58 | 'f' : 'KernelApp.connection_file', | |
59 | 'parent': 'KernelApp.parent', |
|
59 | 'parent': 'KernelApp.parent', | |
60 | }) |
|
60 | }) | |
61 | if sys.platform.startswith('win'): |
|
61 | if sys.platform.startswith('win'): | |
62 | kernel_aliases['interrupt'] = 'KernelApp.interrupt' |
|
62 | kernel_aliases['interrupt'] = 'KernelApp.interrupt' | |
63 |
|
63 | |||
64 | kernel_flags = dict(base_flags) |
|
64 | kernel_flags = dict(base_flags) | |
65 | kernel_flags.update({ |
|
65 | kernel_flags.update({ | |
66 | 'no-stdout' : ( |
|
66 | 'no-stdout' : ( | |
67 | {'KernelApp' : {'no_stdout' : True}}, |
|
67 | {'KernelApp' : {'no_stdout' : True}}, | |
68 | "redirect stdout to the null device"), |
|
68 | "redirect stdout to the null device"), | |
69 | 'no-stderr' : ( |
|
69 | 'no-stderr' : ( | |
70 | {'KernelApp' : {'no_stderr' : True}}, |
|
70 | {'KernelApp' : {'no_stderr' : True}}, | |
71 | "redirect stderr to the null device"), |
|
71 | "redirect stderr to the null device"), | |
72 | }) |
|
72 | }) | |
73 |
|
73 | |||
74 | # inherit flags&aliases for Sessions |
|
74 | # inherit flags&aliases for Sessions | |
75 | kernel_aliases.update(session_aliases) |
|
75 | kernel_aliases.update(session_aliases) | |
76 | kernel_flags.update(session_flags) |
|
76 | kernel_flags.update(session_flags) | |
77 |
|
77 | |||
78 |
|
78 | |||
79 |
|
79 | |||
80 | #----------------------------------------------------------------------------- |
|
80 | #----------------------------------------------------------------------------- | |
81 | # Application class for starting a Kernel |
|
81 | # Application class for starting a Kernel | |
82 | #----------------------------------------------------------------------------- |
|
82 | #----------------------------------------------------------------------------- | |
83 |
|
83 | |||
84 | class KernelApp(BaseIPythonApplication): |
|
84 | class KernelApp(BaseIPythonApplication): | |
85 | name='pykernel' |
|
85 | name='pykernel' | |
86 | aliases = Dict(kernel_aliases) |
|
86 | aliases = Dict(kernel_aliases) | |
87 | flags = Dict(kernel_flags) |
|
87 | flags = Dict(kernel_flags) | |
88 | classes = [Session] |
|
88 | classes = [Session] | |
89 | # the kernel class, as an importstring |
|
89 | # the kernel class, as an importstring | |
90 | kernel_class = DottedObjectName('IPython.zmq.pykernel.Kernel') |
|
90 | kernel_class = DottedObjectName('IPython.zmq.pykernel.Kernel') | |
91 | kernel = Any() |
|
91 | kernel = Any() | |
92 | poller = Any() # don't restrict this even though current pollers are all Threads |
|
92 | poller = Any() # don't restrict this even though current pollers are all Threads | |
93 | heartbeat = Instance(Heartbeat) |
|
93 | heartbeat = Instance(Heartbeat) | |
94 | session = Instance('IPython.zmq.session.Session') |
|
94 | session = Instance('IPython.zmq.session.Session') | |
95 | ports = Dict() |
|
95 | ports = Dict() | |
96 |
|
96 | |||
97 | # inherit config file name from parent: |
|
97 | # inherit config file name from parent: | |
98 | parent_appname = Unicode(config=True) |
|
98 | parent_appname = Unicode(config=True) | |
99 | def _parent_appname_changed(self, name, old, new): |
|
99 | def _parent_appname_changed(self, name, old, new): | |
100 | if self.config_file_specified: |
|
100 | if self.config_file_specified: | |
101 | # it was manually specified, ignore |
|
101 | # it was manually specified, ignore | |
102 | return |
|
102 | return | |
103 | self.config_file_name = new.replace('-','_') + u'_config.py' |
|
103 | self.config_file_name = new.replace('-','_') + u'_config.py' | |
104 | # don't let this count as specifying the config file |
|
104 | # don't let this count as specifying the config file | |
105 | self.config_file_specified = False |
|
105 | self.config_file_specified = False | |
106 |
|
106 | |||
107 | # connection info: |
|
107 | # connection info: | |
108 | ip = Unicode(LOCALHOST, config=True, |
|
108 | ip = Unicode(LOCALHOST, config=True, | |
109 | help="Set the IP or interface on which the kernel will listen.") |
|
109 | help="Set the IP or interface on which the kernel will listen.") | |
110 | hb_port = Integer(0, config=True, help="set the heartbeat port [default: random]") |
|
110 | hb_port = Integer(0, config=True, help="set the heartbeat port [default: random]") | |
111 | shell_port = Integer(0, config=True, help="set the shell (XREP) port [default: random]") |
|
111 | shell_port = Integer(0, config=True, help="set the shell (XREP) port [default: random]") | |
112 | iopub_port = Integer(0, config=True, help="set the iopub (PUB) port [default: random]") |
|
112 | iopub_port = Integer(0, config=True, help="set the iopub (PUB) port [default: random]") | |
113 | stdin_port = Integer(0, config=True, help="set the stdin (XREQ) port [default: random]") |
|
113 | stdin_port = Integer(0, config=True, help="set the stdin (XREQ) port [default: random]") | |
114 | connection_file = Unicode('', config=True, |
|
114 | connection_file = Unicode('', config=True, | |
115 | help="""JSON file in which to store connection info [default: kernel-<pid>.json] |
|
115 | help="""JSON file in which to store connection info [default: kernel-<pid>.json] | |
116 |
|
116 | |||
117 | This file will contain the IP, ports, and authentication key needed to connect |
|
117 | This file will contain the IP, ports, and authentication key needed to connect | |
118 | clients to this kernel. By default, this file will be created in the security-dir |
|
118 | clients to this kernel. By default, this file will be created in the security-dir | |
119 | of the current profile, but can be specified by absolute path. |
|
119 | of the current profile, but can be specified by absolute path. | |
120 | """) |
|
120 | """) | |
121 |
|
121 | |||
122 | # streams, etc. |
|
122 | # streams, etc. | |
123 | no_stdout = Bool(False, config=True, help="redirect stdout to the null device") |
|
123 | no_stdout = Bool(False, config=True, help="redirect stdout to the null device") | |
124 | no_stderr = Bool(False, config=True, help="redirect stderr to the null device") |
|
124 | no_stderr = Bool(False, config=True, help="redirect stderr to the null device") | |
125 | outstream_class = DottedObjectName('IPython.zmq.iostream.OutStream', |
|
125 | outstream_class = DottedObjectName('IPython.zmq.iostream.OutStream', | |
126 | config=True, help="The importstring for the OutStream factory") |
|
126 | config=True, help="The importstring for the OutStream factory") | |
127 | displayhook_class = DottedObjectName('IPython.zmq.displayhook.ZMQDisplayHook', |
|
127 | displayhook_class = DottedObjectName('IPython.zmq.displayhook.ZMQDisplayHook', | |
128 | config=True, help="The importstring for the DisplayHook factory") |
|
128 | config=True, help="The importstring for the DisplayHook factory") | |
129 |
|
129 | |||
130 | # polling |
|
130 | # polling | |
131 | parent = Integer(0, config=True, |
|
131 | parent = Integer(0, config=True, | |
132 | help="""kill this process if its parent dies. On Windows, the argument |
|
132 | help="""kill this process if its parent dies. On Windows, the argument | |
133 | specifies the HANDLE of the parent process, otherwise it is simply boolean. |
|
133 | specifies the HANDLE of the parent process, otherwise it is simply boolean. | |
134 | """) |
|
134 | """) | |
135 | interrupt = Integer(0, config=True, |
|
135 | interrupt = Integer(0, config=True, | |
136 | help="""ONLY USED ON WINDOWS |
|
136 | help="""ONLY USED ON WINDOWS | |
137 | Interrupt this process when the parent is signalled. |
|
137 | Interrupt this process when the parent is signalled. | |
138 | """) |
|
138 | """) | |
139 |
|
139 | |||
140 | def init_crash_handler(self): |
|
140 | def init_crash_handler(self): | |
141 | # Install minimal exception handling |
|
141 | # Install minimal exception handling | |
142 | sys.excepthook = FormattedTB(mode='Verbose', color_scheme='NoColor', |
|
142 | sys.excepthook = FormattedTB(mode='Verbose', color_scheme='NoColor', | |
143 | ostream=sys.__stdout__) |
|
143 | ostream=sys.__stdout__) | |
144 |
|
144 | |||
145 | def init_poller(self): |
|
145 | def init_poller(self): | |
146 | if sys.platform == 'win32': |
|
146 | if sys.platform == 'win32': | |
147 | if self.interrupt or self.parent: |
|
147 | if self.interrupt or self.parent: | |
148 | self.poller = ParentPollerWindows(self.interrupt, self.parent) |
|
148 | self.poller = ParentPollerWindows(self.interrupt, self.parent) | |
149 | elif self.parent: |
|
149 | elif self.parent: | |
150 | self.poller = ParentPollerUnix() |
|
150 | self.poller = ParentPollerUnix() | |
151 |
|
151 | |||
152 | def _bind_socket(self, s, port): |
|
152 | def _bind_socket(self, s, port): | |
153 | iface = 'tcp://%s' % self.ip |
|
153 | iface = 'tcp://%s' % self.ip | |
154 | if port <= 0: |
|
154 | if port <= 0: | |
155 | port = s.bind_to_random_port(iface) |
|
155 | port = s.bind_to_random_port(iface) | |
156 | else: |
|
156 | else: | |
157 | s.bind(iface + ':%i'%port) |
|
157 | s.bind(iface + ':%i'%port) | |
158 | return port |
|
158 | return port | |
159 |
|
159 | |||
160 | def load_connection_file(self): |
|
160 | def load_connection_file(self): | |
161 | """load ip/port/hmac config from JSON connection file""" |
|
161 | """load ip/port/hmac config from JSON connection file""" | |
162 | try: |
|
162 | try: | |
163 | fname = filefind(self.connection_file, ['.', self.profile_dir.security_dir]) |
|
163 | fname = filefind(self.connection_file, ['.', self.profile_dir.security_dir]) | |
164 | except IOError: |
|
164 | except IOError: | |
165 | self.log.debug("Connection file not found: %s", self.connection_file) |
|
165 | self.log.debug("Connection file not found: %s", self.connection_file) | |
166 | return |
|
166 | return | |
167 | self.log.debug(u"Loading connection file %s", fname) |
|
167 | self.log.debug(u"Loading connection file %s", fname) | |
168 | with open(fname) as f: |
|
168 | with open(fname) as f: | |
169 | s = f.read() |
|
169 | s = f.read() | |
170 | cfg = json.loads(s) |
|
170 | cfg = json.loads(s) | |
171 | if self.ip == LOCALHOST and 'ip' in cfg: |
|
171 | if self.ip == LOCALHOST and 'ip' in cfg: | |
172 | # not overridden by config or cl_args |
|
172 | # not overridden by config or cl_args | |
173 | self.ip = cfg['ip'] |
|
173 | self.ip = cfg['ip'] | |
174 | for channel in ('hb', 'shell', 'iopub', 'stdin'): |
|
174 | for channel in ('hb', 'shell', 'iopub', 'stdin'): | |
175 | name = channel + '_port' |
|
175 | name = channel + '_port' | |
176 | if getattr(self, name) == 0 and name in cfg: |
|
176 | if getattr(self, name) == 0 and name in cfg: | |
177 | # not overridden by config or cl_args |
|
177 | # not overridden by config or cl_args | |
178 | setattr(self, name, cfg[name]) |
|
178 | setattr(self, name, cfg[name]) | |
179 | if 'key' in cfg: |
|
179 | if 'key' in cfg: | |
180 | self.config.Session.key = str_to_bytes(cfg['key']) |
|
180 | self.config.Session.key = str_to_bytes(cfg['key']) | |
181 |
|
181 | |||
182 | def write_connection_file(self): |
|
182 | def write_connection_file(self): | |
183 | """write connection info to JSON file""" |
|
183 | """write connection info to JSON file""" | |
184 | if os.path.basename(self.connection_file) == self.connection_file: |
|
184 | if os.path.basename(self.connection_file) == self.connection_file: | |
185 | cf = os.path.join(self.profile_dir.security_dir, self.connection_file) |
|
185 | cf = os.path.join(self.profile_dir.security_dir, self.connection_file) | |
186 | else: |
|
186 | else: | |
187 | cf = self.connection_file |
|
187 | cf = self.connection_file | |
188 | write_connection_file(cf, ip=self.ip, key=self.session.key, |
|
188 | write_connection_file(cf, ip=self.ip, key=self.session.key, | |
189 | shell_port=self.shell_port, stdin_port=self.stdin_port, hb_port=self.hb_port, |
|
189 | shell_port=self.shell_port, stdin_port=self.stdin_port, hb_port=self.hb_port, | |
190 | iopub_port=self.iopub_port) |
|
190 | iopub_port=self.iopub_port) | |
191 |
|
191 | |||
192 | def init_connection_file(self): |
|
192 | def init_connection_file(self): | |
193 | if not self.connection_file: |
|
193 | if not self.connection_file: | |
194 | self.connection_file = "kernel-%s.json"%os.getpid() |
|
194 | self.connection_file = "kernel-%s.json"%os.getpid() | |
195 | try: |
|
195 | try: | |
196 | self.load_connection_file() |
|
196 | self.load_connection_file() | |
197 | except Exception: |
|
197 | except Exception: | |
198 | self.log.error("Failed to load connection file: %r", self.connection_file, exc_info=True) |
|
198 | self.log.error("Failed to load connection file: %r", self.connection_file, exc_info=True) | |
199 | self.exit(1) |
|
199 | self.exit(1) | |
200 |
|
200 | |||
201 | def init_sockets(self): |
|
201 | def init_sockets(self): | |
202 | # Create a context, a session, and the kernel sockets. |
|
202 | # Create a context, a session, and the kernel sockets. | |
203 | self.log.info("Starting the kernel at pid: %i", os.getpid()) |
|
203 | self.log.info("Starting the kernel at pid: %i", os.getpid()) | |
204 | context = zmq.Context.instance() |
|
204 | context = zmq.Context.instance() | |
205 | # Uncomment this to try closing the context. |
|
205 | # Uncomment this to try closing the context. | |
206 | # atexit.register(context.term) |
|
206 | # atexit.register(context.term) | |
207 |
|
207 | |||
208 | self.shell_socket = context.socket(zmq.ROUTER) |
|
208 | self.shell_socket = context.socket(zmq.ROUTER) | |
209 | self.shell_port = self._bind_socket(self.shell_socket, self.shell_port) |
|
209 | self.shell_port = self._bind_socket(self.shell_socket, self.shell_port) | |
210 | self.log.debug("shell ROUTER Channel on port: %i"%self.shell_port) |
|
210 | self.log.debug("shell ROUTER Channel on port: %i"%self.shell_port) | |
211 |
|
211 | |||
212 | self.iopub_socket = context.socket(zmq.PUB) |
|
212 | self.iopub_socket = context.socket(zmq.PUB) | |
213 | self.iopub_port = self._bind_socket(self.iopub_socket, self.iopub_port) |
|
213 | self.iopub_port = self._bind_socket(self.iopub_socket, self.iopub_port) | |
214 | self.log.debug("iopub PUB Channel on port: %i"%self.iopub_port) |
|
214 | self.log.debug("iopub PUB Channel on port: %i"%self.iopub_port) | |
215 |
|
215 | |||
216 | self.stdin_socket = context.socket(zmq.ROUTER) |
|
216 | self.stdin_socket = context.socket(zmq.ROUTER) | |
217 | self.stdin_port = self._bind_socket(self.stdin_socket, self.stdin_port) |
|
217 | self.stdin_port = self._bind_socket(self.stdin_socket, self.stdin_port) | |
218 | self.log.debug("stdin ROUTER Channel on port: %i"%self.stdin_port) |
|
218 | self.log.debug("stdin ROUTER Channel on port: %i"%self.stdin_port) | |
219 |
|
219 | |||
220 | self.heartbeat = Heartbeat(context, (self.ip, self.hb_port)) |
|
220 | self.heartbeat = Heartbeat(context, (self.ip, self.hb_port)) | |
221 | self.hb_port = self.heartbeat.port |
|
221 | self.hb_port = self.heartbeat.port | |
222 | self.log.debug("Heartbeat REP Channel on port: %i"%self.hb_port) |
|
222 | self.log.debug("Heartbeat REP Channel on port: %i"%self.hb_port) | |
223 |
|
223 | |||
224 | # Helper to make it easier to connect to an existing kernel. |
|
224 | # Helper to make it easier to connect to an existing kernel. | |
225 | # set log-level to critical, to make sure it is output |
|
225 | # set log-level to critical, to make sure it is output | |
226 | self.log.critical("To connect another client to this kernel, use:") |
|
226 | self.log.critical("To connect another client to this kernel, use:") | |
227 |
|
227 | |||
228 | basename = os.path.basename(self.connection_file) |
|
228 | basename = os.path.basename(self.connection_file) | |
229 | if basename == self.connection_file or \ |
|
229 | if basename == self.connection_file or \ | |
230 | os.path.dirname(self.connection_file) == self.profile_dir.security_dir: |
|
230 | os.path.dirname(self.connection_file) == self.profile_dir.security_dir: | |
231 | # use shortname |
|
231 | # use shortname | |
232 | tail = basename |
|
232 | tail = basename | |
233 | if self.profile != 'default': |
|
233 | if self.profile != 'default': | |
234 | tail += " --profile %s" % self.profile |
|
234 | tail += " --profile %s" % self.profile | |
235 | else: |
|
235 | else: | |
236 | tail = self.connection_file |
|
236 | tail = self.connection_file | |
237 | self.log.critical("--existing %s", tail) |
|
237 | self.log.critical("--existing %s", tail) | |
238 |
|
238 | |||
239 |
|
239 | |||
240 | self.ports = dict(shell=self.shell_port, iopub=self.iopub_port, |
|
240 | self.ports = dict(shell=self.shell_port, iopub=self.iopub_port, | |
241 | stdin=self.stdin_port, hb=self.hb_port) |
|
241 | stdin=self.stdin_port, hb=self.hb_port) | |
242 |
|
242 | |||
243 | def init_session(self): |
|
243 | def init_session(self): | |
244 | """create our session object""" |
|
244 | """create our session object""" | |
245 | default_secure(self.config) |
|
245 | default_secure(self.config) | |
246 | self.session = Session(config=self.config, username=u'kernel') |
|
246 | self.session = Session(config=self.config, username=u'kernel') | |
247 |
|
247 | |||
248 | def init_blackhole(self): |
|
248 | def init_blackhole(self): | |
249 | """redirects stdout/stderr to devnull if necessary""" |
|
249 | """redirects stdout/stderr to devnull if necessary""" | |
250 | if self.no_stdout or self.no_stderr: |
|
250 | if self.no_stdout or self.no_stderr: | |
251 | blackhole = file(os.devnull, 'w') |
|
251 | blackhole = file(os.devnull, 'w') | |
252 | if self.no_stdout: |
|
252 | if self.no_stdout: | |
253 | sys.stdout = sys.__stdout__ = blackhole |
|
253 | sys.stdout = sys.__stdout__ = blackhole | |
254 | if self.no_stderr: |
|
254 | if self.no_stderr: | |
255 | sys.stderr = sys.__stderr__ = blackhole |
|
255 | sys.stderr = sys.__stderr__ = blackhole | |
256 |
|
256 | |||
257 | def init_io(self): |
|
257 | def init_io(self): | |
258 | """Redirect input streams and set a display hook.""" |
|
258 | """Redirect input streams and set a display hook.""" | |
259 | if self.outstream_class: |
|
259 | if self.outstream_class: | |
260 | outstream_factory = import_item(str(self.outstream_class)) |
|
260 | outstream_factory = import_item(str(self.outstream_class)) | |
261 | sys.stdout = outstream_factory(self.session, self.iopub_socket, u'stdout') |
|
261 | sys.stdout = outstream_factory(self.session, self.iopub_socket, u'stdout') | |
262 | sys.stderr = outstream_factory(self.session, self.iopub_socket, u'stderr') |
|
262 | sys.stderr = outstream_factory(self.session, self.iopub_socket, u'stderr') | |
263 | if self.displayhook_class: |
|
263 | if self.displayhook_class: | |
264 | displayhook_factory = import_item(str(self.displayhook_class)) |
|
264 | displayhook_factory = import_item(str(self.displayhook_class)) | |
265 | sys.displayhook = displayhook_factory(self.session, self.iopub_socket) |
|
265 | sys.displayhook = displayhook_factory(self.session, self.iopub_socket) | |
266 |
|
266 | |||
267 | def init_kernel(self): |
|
267 | def init_kernel(self): | |
268 | """Create the Kernel object itself""" |
|
268 | """Create the Kernel object itself""" | |
269 | kernel_factory = import_item(str(self.kernel_class)) |
|
269 | kernel_factory = import_item(str(self.kernel_class)) | |
270 | self.kernel = kernel_factory(config=self.config, session=self.session, |
|
270 | self.kernel = kernel_factory(config=self.config, session=self.session, | |
271 | shell_socket=self.shell_socket, |
|
271 | shell_socket=self.shell_socket, | |
272 | iopub_socket=self.iopub_socket, |
|
272 | iopub_socket=self.iopub_socket, | |
273 | stdin_socket=self.stdin_socket, |
|
273 | stdin_socket=self.stdin_socket, | |
274 | log=self.log |
|
274 | log=self.log | |
275 | ) |
|
275 | ) | |
276 | self.kernel.record_ports(self.ports) |
|
276 | self.kernel.record_ports(self.ports) | |
277 |
|
277 | |||
278 | @catch_config_error |
|
278 | @catch_config_error | |
279 | def initialize(self, argv=None): |
|
279 | def initialize(self, argv=None): | |
280 | super(KernelApp, self).initialize(argv) |
|
280 | super(KernelApp, self).initialize(argv) | |
281 | self.init_blackhole() |
|
281 | self.init_blackhole() | |
282 | self.init_connection_file() |
|
282 | self.init_connection_file() | |
283 | self.init_session() |
|
283 | self.init_session() | |
284 | self.init_poller() |
|
284 | self.init_poller() | |
285 | self.init_sockets() |
|
285 | self.init_sockets() | |
286 | # writing connection file must be *after* init_sockets |
|
286 | # writing connection file must be *after* init_sockets | |
287 | self.write_connection_file() |
|
287 | self.write_connection_file() | |
288 | self.init_io() |
|
288 | self.init_io() | |
289 | self.init_kernel() |
|
289 | self.init_kernel() | |
|
290 | # flush stdout/stderr, so that anything written to these streams during | |||
|
291 | # initialization do not get associated with the first execution request | |||
|
292 | sys.stdout.flush() | |||
|
293 | sys.stderr.flush() | |||
290 |
|
294 | |||
291 | def start(self): |
|
295 | def start(self): | |
292 | self.heartbeat.start() |
|
296 | self.heartbeat.start() | |
293 | if self.poller is not None: |
|
297 | if self.poller is not None: | |
294 | self.poller.start() |
|
298 | self.poller.start() | |
295 | try: |
|
299 | try: | |
296 | self.kernel.start() |
|
300 | self.kernel.start() | |
297 | except KeyboardInterrupt: |
|
301 | except KeyboardInterrupt: | |
298 | pass |
|
302 | pass | |
299 |
|
303 |
General Comments 0
You need to be logged in to leave comments.
Login now