outputarea.js
1002 lines
| 34.2 KiB
| application/javascript
|
JavascriptLexer
MinRK
|
r16568 | // Copyright (c) IPython Development Team. | ||
// Distributed under the terms of the Modified BSD License. | ||||
Brian Granger
|
r7177 | |||
Jonathan Frederic
|
r17198 | define([ | ||
'base/js/namespace', | ||||
Jonathan Frederic
|
r17215 | 'jqueryui', | ||
Jonathan Frederic
|
r17198 | 'base/js/utils', | ||
'base/js/security', | ||||
'base/js/keyboard', | ||||
'notebook/js/mathjaxutils', | ||||
Ben Duffield
|
r17389 | 'components/marked/lib/marked', | ||
], function(IPython, $, utils, security, keyboard, mathjaxutils, marked) { | ||||
Brian Granger
|
r7177 | "use strict"; | ||
Matthias BUSSONNIER
|
r10776 | /** | ||
* @class OutputArea | ||||
* | ||||
* @constructor | ||||
*/ | ||||
Jonathan Frederic
|
r17214 | var OutputArea = function (options) { | ||
this.selector = options.selector; | ||||
this.events = options.events; | ||||
this.keyboard_manager = options.keyboard_manager; | ||||
this.wrapper = $(options.selector); | ||||
Brian Granger
|
r7177 | this.outputs = []; | ||
this.collapsed = false; | ||||
MinRK
|
r7362 | this.scrolled = false; | ||
MinRK
|
r14856 | this.trusted = true; | ||
Jonathan Frederic
|
r12592 | this.clear_queued = null; | ||
Jonathan Frederic
|
r17214 | if (options.prompt_area === undefined) { | ||
Brian Granger
|
r7177 | this.prompt_area = true; | ||
Brian Granger
|
r7181 | } else { | ||
Jonathan Frederic
|
r17214 | this.prompt_area = options.prompt_area; | ||
Matthias BUSSONNIER
|
r9555 | } | ||
MinRK
|
r7362 | this.create_elements(); | ||
Brian Granger
|
r7177 | this.style(); | ||
MinRK
|
r7362 | this.bind_events(); | ||
}; | ||||
Matthias BUSSONNIER
|
r14886 | |||
/** | ||||
* Class prototypes | ||||
**/ | ||||
MinRK
|
r7362 | OutputArea.prototype.create_elements = function () { | ||
this.element = $("<div/>"); | ||||
this.collapse_button = $("<div/>"); | ||||
this.prompt_overlay = $("<div/>"); | ||||
this.wrapper.append(this.prompt_overlay); | ||||
this.wrapper.append(this.element); | ||||
this.wrapper.append(this.collapse_button); | ||||
Brian Granger
|
r7177 | }; | ||
OutputArea.prototype.style = function () { | ||||
MinRK
|
r7362 | this.collapse_button.hide(); | ||
this.prompt_overlay.hide(); | ||||
this.wrapper.addClass('output_wrapper'); | ||||
Matthias BUSSONNIER
|
r13332 | this.element.addClass('output'); | ||
MinRK
|
r7362 | |||
Jonathan Frederic
|
r16913 | this.collapse_button.addClass("btn btn-default output_collapsed"); | ||
Harry Moreno
|
r9903 | this.collapse_button.attr('title', 'click to expand output'); | ||
Matthias BUSSONNIER
|
r14634 | this.collapse_button.text('. . .'); | ||
MinRK
|
r7362 | |||
this.prompt_overlay.addClass('out_prompt_overlay prompt'); | ||||
Harry Moreno
|
r9903 | this.prompt_overlay.attr('title', 'click to expand output; double click to hide output'); | ||
MinRK
|
r7362 | |||
Brian Granger
|
r7177 | this.collapse(); | ||
}; | ||||
Matthias BUSSONNIER
|
r10776 | /** | ||
* Should the OutputArea scroll? | ||||
* Returns whether the height (in lines) exceeds a threshold. | ||||
* | ||||
* @private | ||||
* @method _should_scroll | ||||
* @param [lines=100]{Integer} | ||||
* @return {Bool} | ||||
* | ||||
*/ | ||||
MinRK
|
r7362 | OutputArea.prototype._should_scroll = function (lines) { | ||
Matthias BUSSONNIER
|
r10776 | if (lines <=0 ){ return } | ||
MinRK
|
r7362 | if (!lines) { | ||
MinRK
|
r7728 | lines = 100; | ||
MinRK
|
r7362 | } | ||
// line-height from http://stackoverflow.com/questions/1185151 | ||||
var fontSize = this.element.css('font-size'); | ||||
var lineHeight = Math.floor(parseInt(fontSize.replace('px','')) * 1.5); | ||||
return (this.element.height() > lines * lineHeight); | ||||
}; | ||||
OutputArea.prototype.bind_events = function () { | ||||
var that = this; | ||||
this.prompt_overlay.dblclick(function () { that.toggle_output(); }); | ||||
this.prompt_overlay.click(function () { that.toggle_scroll(); }); | ||||
this.element.resize(function () { | ||||
MinRK
|
r7733 | // FIXME: Firefox on Linux misbehaves, so automatic scrolling is disabled | ||
Jonathan Frederic
|
r17198 | if ( utils.browser[0] === "Firefox" ) { | ||
MinRK
|
r7733 | return; | ||
} | ||||
MinRK
|
r7362 | // maybe scroll output, | ||
// if it's grown large enough and hasn't already been scrolled. | ||||
Matthias BUSSONNIER
|
r10776 | if ( !that.scrolled && that._should_scroll(OutputArea.auto_scroll_threshold)) { | ||
MinRK
|
r7362 | that.scroll_area(); | ||
} | ||||
}); | ||||
this.collapse_button.click(function () { | ||||
that.expand(); | ||||
}); | ||||
}; | ||||
Brian Granger
|
r7177 | OutputArea.prototype.collapse = function () { | ||
if (!this.collapsed) { | ||||
this.element.hide(); | ||||
MinRK
|
r7362 | this.prompt_overlay.hide(); | ||
if (this.element.html()){ | ||||
this.collapse_button.show(); | ||||
} | ||||
Brian Granger
|
r7177 | this.collapsed = true; | ||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
OutputArea.prototype.expand = function () { | ||||
if (this.collapsed) { | ||||
MinRK
|
r7362 | this.collapse_button.hide(); | ||
Brian Granger
|
r7177 | this.element.show(); | ||
MinRK
|
r7362 | this.prompt_overlay.show(); | ||
Brian Granger
|
r7177 | this.collapsed = false; | ||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
OutputArea.prototype.toggle_output = function () { | ||||
if (this.collapsed) { | ||||
this.expand(); | ||||
} else { | ||||
this.collapse(); | ||||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
MinRK
|
r7362 | OutputArea.prototype.scroll_area = function () { | ||
this.element.addClass('output_scroll'); | ||||
MinRK
|
r7423 | this.prompt_overlay.attr('title', 'click to unscroll output; double click to hide'); | ||
MinRK
|
r7362 | this.scrolled = true; | ||
}; | ||||
OutputArea.prototype.unscroll_area = function () { | ||||
this.element.removeClass('output_scroll'); | ||||
MinRK
|
r7423 | this.prompt_overlay.attr('title', 'click to scroll output; double click to hide'); | ||
MinRK
|
r7362 | this.scrolled = false; | ||
}; | ||||
Matthias BUSSONNIER
|
r10776 | /** | ||
* | ||||
Matthias BUSSONNIER
|
r10778 | * Scroll OutputArea if height supperior than a threshold (in lines). | ||
Matthias BUSSONNIER
|
r10776 | * | ||
Matthias BUSSONNIER
|
r10778 | * Threshold is a maximum number of lines. If unspecified, defaults to | ||
Matthias BUSSONNIER
|
r10782 | * OutputArea.minimum_scroll_threshold. | ||
Matthias BUSSONNIER
|
r10778 | * | ||
Matthias BUSSONNIER
|
r10793 | * Negative threshold will prevent the OutputArea from ever scrolling. | ||
Matthias BUSSONNIER
|
r10776 | * | ||
* @method scroll_if_long | ||||
Matthias BUSSONNIER
|
r10793 | * | ||
* @param [lines=20]{Number} Default to 20 if not set, | ||||
* behavior undefined for value of `0`. | ||||
Matthias BUSSONNIER
|
r10776 | * | ||
**/ | ||||
MinRK
|
r7362 | OutputArea.prototype.scroll_if_long = function (lines) { | ||
Matthias BUSSONNIER
|
r10782 | var n = lines | OutputArea.minimum_scroll_threshold; | ||
Matthias BUSSONNIER
|
r10776 | if(n <= 0){ | ||
return | ||||
} | ||||
if (this._should_scroll(n)) { | ||||
MinRK
|
r7362 | // only allow scrolling long-enough output | ||
this.scroll_area(); | ||||
Matthias BUSSONNIER
|
r9555 | } | ||
MinRK
|
r7362 | }; | ||
OutputArea.prototype.toggle_scroll = function () { | ||||
if (this.scrolled) { | ||||
this.unscroll_area(); | ||||
} else { | ||||
// only allow scrolling long-enough output | ||||
Matthias BUSSONNIER
|
r10776 | this.scroll_if_long(); | ||
Matthias BUSSONNIER
|
r9555 | } | ||
MinRK
|
r7362 | }; | ||
Brian Granger
|
r7177 | // typeset with MathJax if MathJax is available | ||
OutputArea.prototype.typeset = function () { | ||||
if (window.MathJax){ | ||||
MathJax.Hub.Queue(["Typeset",MathJax.Hub]); | ||||
} | ||||
}; | ||||
MinRK
|
r13207 | OutputArea.prototype.handle_output = function (msg) { | ||
Brian Granger
|
r7177 | var json = {}; | ||
MinRK
|
r13207 | var msg_type = json.output_type = msg.header.msg_type; | ||
var content = msg.content; | ||||
Brian Granger
|
r7177 | if (msg_type === "stream") { | ||
json.text = content.data; | ||||
json.stream = content.name; | ||||
} else if (msg_type === "display_data") { | ||||
Paul Ivanov
|
r14131 | json = content.data; | ||
json.output_type = msg_type; | ||||
json.metadata = content.metadata; | ||||
MinRK
|
r16568 | } else if (msg_type === "execute_result") { | ||
Paul Ivanov
|
r14131 | json = content.data; | ||
MinRK
|
r16589 | json.output_type = msg_type; | ||
Paul Ivanov
|
r14131 | json.metadata = content.metadata; | ||
Brian Granger
|
r7177 | json.prompt_number = content.execution_count; | ||
MinRK
|
r16569 | } else if (msg_type === "error") { | ||
MinRK
|
r16589 | json.ename = content.ename; | ||
json.evalue = content.evalue; | ||||
json.traceback = content.traceback; | ||||
} else { | ||||
console.log("unhandled output message", msg); | ||||
return; | ||||
Matthias BUSSONNIER
|
r9555 | } | ||
Paul Ivanov
|
r14134 | this.append_output(json); | ||
Brian Granger
|
r7177 | }; | ||
Paul Ivanov
|
r14118 | |||
MinRK
|
r14157 | |||
Paul Ivanov
|
r14124 | OutputArea.prototype.rename_keys = function (data, key_map) { | ||
Paul Ivanov
|
r14127 | var remapped = {}; | ||
MinRK
|
r14115 | for (var key in data) { | ||
Paul Ivanov
|
r14124 | var new_key = key_map[key] || key; | ||
MinRK
|
r14157 | remapped[new_key] = data[key]; | ||
Paul Ivanov
|
r14112 | } | ||
Paul Ivanov
|
r14127 | return remapped; | ||
Brian Granger
|
r7177 | }; | ||
MinRK
|
r14158 | |||
OutputArea.output_types = [ | ||||
'application/javascript', | ||||
'text/html', | ||||
Andrew Jesaitis
|
r16364 | 'text/markdown', | ||
MinRK
|
r14158 | 'text/latex', | ||
'image/svg+xml', | ||||
'image/png', | ||||
'image/jpeg', | ||||
Brian E. Granger
|
r15127 | 'application/pdf', | ||
MinRK
|
r14158 | 'text/plain' | ||
]; | ||||
OutputArea.prototype.validate_output = function (json) { | ||||
// scrub invalid outputs | ||||
// TODO: right now everything is a string, but JSON really shouldn't be. | ||||
// nbformat 4 will fix that. | ||||
$.map(OutputArea.output_types, function(key){ | ||||
if (json[key] !== undefined && typeof json[key] !== 'string') { | ||||
console.log("Invalid type for " + key, json[key]); | ||||
delete json[key]; | ||||
} | ||||
}); | ||||
return json; | ||||
}; | ||||
MinRK
|
r14856 | |||
Paul Ivanov
|
r14134 | OutputArea.prototype.append_output = function (json) { | ||
Brian Granger
|
r7177 | this.expand(); | ||
Jonathan Frederic
|
r16191 | |||
// validate output data types | ||||
json = this.validate_output(json); | ||||
Jonathan Frederic
|
r12592 | // Clear the output if clear is queued. | ||
Jonathan Frederic
|
r12817 | var needs_height_reset = false; | ||
Jonathan Frederic
|
r12592 | if (this.clear_queued) { | ||
this.clear_output(false); | ||||
Jonathan Frederic
|
r12817 | needs_height_reset = true; | ||
Jonathan Frederic
|
r12592 | } | ||
jon
|
r16187 | |||
MinRK
|
r17305 | var record_output = true; | ||
MinRK
|
r16589 | if (json.output_type === 'execute_result') { | ||
MinRK
|
r16568 | this.append_execute_result(json); | ||
MinRK
|
r16589 | } else if (json.output_type === 'error') { | ||
MinRK
|
r16569 | this.append_error(json); | ||
Brian Granger
|
r7177 | } else if (json.output_type === 'stream') { | ||
MinRK
|
r17305 | // append_stream might have merged the output with earlier stream output | ||
record_output = this.append_stream(json); | ||||
Matthias BUSSONNIER
|
r9555 | } | ||
jon
|
r16187 | |||
Jonathan Frederic
|
r16191 | // We must release the animation fixed height in a callback since Gecko | ||
// (FireFox) doesn't render the image immediately as the data is | ||||
// available. | ||||
var that = this; | ||||
var handle_appended = function ($el) { | ||||
// Only reset the height to automatic if the height is currently | ||||
// fixed (done by wait=True flag on clear_output). | ||||
if (needs_height_reset) { | ||||
that.element.height(''); | ||||
} | ||||
that.element.trigger('resize'); | ||||
}; | ||||
jon
|
r16187 | if (json.output_type === 'display_data') { | ||
this.append_display_data(json, handle_appended); | ||||
Jonathan Frederic
|
r16191 | } else { | ||
jon
|
r16187 | handle_appended(); | ||
} | ||||
MinRK
|
r14856 | |||
MinRK
|
r17305 | if (record_output) { | ||
this.outputs.push(json); | ||||
} | ||||
Brian Granger
|
r7177 | }; | ||
OutputArea.prototype.create_output_area = function () { | ||||
Matthias BUSSONNIER
|
r10216 | var oa = $("<div/>").addClass("output_area"); | ||
Brian Granger
|
r7177 | if (this.prompt_area) { | ||
oa.append($('<div/>').addClass('prompt')); | ||||
} | ||||
return oa; | ||||
}; | ||||
Pablo de Oliveira
|
r13412 | |||
Paul Ivanov
|
r14116 | function _get_metadata_key(metadata, key, mime) { | ||
var mime_md = metadata[mime]; | ||||
// mime-specific higher priority | ||||
if (mime_md && mime_md[key] !== undefined) { | ||||
return mime_md[key]; | ||||
} | ||||
// fallback on global | ||||
return metadata[key]; | ||||
} | ||||
OutputArea.prototype.create_output_subarea = function(md, classes, mime) { | ||||
Pablo de Oliveira
|
r13412 | var subarea = $('<div/>').addClass('output_subarea').addClass(classes); | ||
Paul Ivanov
|
r14116 | if (_get_metadata_key(md, 'isolated', mime)) { | ||
Pablo de Oliveira
|
r13412 | // Create an iframe to isolate the subarea from the rest of the | ||
// document | ||||
Pablo de Oliveira
|
r13418 | var iframe = $('<iframe/>').addClass('box-flex1'); | ||
Pablo de Oliveira
|
r13419 | iframe.css({'height':1, 'width':'100%', 'display':'block'}); | ||
Pablo de Oliveira
|
r13412 | iframe.attr('frameborder', 0); | ||
iframe.attr('scrolling', 'auto'); | ||||
// Once the iframe is loaded, the subarea is dynamically inserted | ||||
iframe.on('load', function() { | ||||
// Workaround needed by Firefox, to properly render svg inside | ||||
// iframes, see http://stackoverflow.com/questions/10177190/ | ||||
// svg-dynamically-added-to-iframe-does-not-render-correctly | ||||
this.contentDocument.open(); | ||||
// Insert the subarea into the iframe | ||||
// We must directly write the html. When using Jquery's append | ||||
// method, javascript is evaluated in the parent document and | ||||
Jonathan Frederic
|
r15376 | // not in the iframe document. At this point, subarea doesn't | ||
// contain any user content. | ||||
Pablo de Oliveira
|
r13412 | this.contentDocument.write(subarea.html()); | ||
this.contentDocument.close(); | ||||
var body = this.contentDocument.body; | ||||
Pablo de Oliveira
|
r13418 | // Adjust the iframe height automatically | ||
Pablo de Oliveira
|
r13412 | iframe.height(body.scrollHeight + 'px'); | ||
}); | ||||
// Elements should be appended to the inner subarea and not to the | ||||
// iframe | ||||
iframe.append = function(that) { | ||||
subarea.append(that); | ||||
}; | ||||
return iframe; | ||||
} else { | ||||
return subarea; | ||||
} | ||||
} | ||||
Brian E. Granger
|
r13792 | OutputArea.prototype._append_javascript_error = function (err, element) { | ||
MinRK
|
r12332 | // display a message when a javascript error occurs in display output | ||
var msg = "Javascript error adding output!" | ||||
Brian E. Granger
|
r13792 | if ( element === undefined ) return; | ||
Jonathan Frederic
|
r15376 | element | ||
.append($('<div/>').text(msg).addClass('js-error')) | ||||
.append($('<div/>').text(err.toString()).addClass('js-error')) | ||||
.append($('<div/>').text('See your browser Javascript console for more details.').addClass('js-error')); | ||||
MinRK
|
r12332 | }; | ||
OutputArea.prototype._safe_append = function (toinsert) { | ||||
// safely append an item to the document | ||||
// this is an object created by user code, | ||||
// and may have errors, which should not be raised | ||||
// under any circumstances. | ||||
try { | ||||
this.element.append(toinsert); | ||||
} catch(err) { | ||||
console.log(err); | ||||
Brian E. Granger
|
r13792 | // Create an actual output_area and output_subarea, which creates | ||
// the prompt area and the proper indentation. | ||||
var toinsert = this.create_output_area(); | ||||
var subarea = $('<div/>').addClass('output_subarea'); | ||||
toinsert.append(subarea); | ||||
this._append_javascript_error(err, subarea); | ||||
this.element.append(toinsert); | ||||
MinRK
|
r12332 | } | ||
}; | ||||
Brian Granger
|
r7177 | |||
MinRK
|
r16568 | OutputArea.prototype.append_execute_result = function (json) { | ||
Brian Granger
|
r7177 | var n = json.prompt_number || ' '; | ||
var toinsert = this.create_output_area(); | ||||
if (this.prompt_area) { | ||||
Matthias BUSSONNIER
|
r14634 | toinsert.find('div.prompt').addClass('output_prompt').text('Out[' + n + ']:'); | ||
Brian Granger
|
r7177 | } | ||
Jonathan Frederic
|
r15729 | var inserted = this.append_mime_type(json, toinsert); | ||
if (inserted) { | ||||
MinRK
|
r16589 | inserted.addClass('output_result'); | ||
Jonathan Frederic
|
r15729 | } | ||
MinRK
|
r12332 | this._safe_append(toinsert); | ||
Brian Granger
|
r7177 | // If we just output latex, typeset it. | ||
Andrew Jesaitis
|
r16364 | if ((json['text/latex'] !== undefined) || | ||
(json['text/html'] !== undefined) || | ||||
(json['text/markdown'] !== undefined)) { | ||||
Brian Granger
|
r7177 | this.typeset(); | ||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
MinRK
|
r16569 | OutputArea.prototype.append_error = function (json) { | ||
Brian Granger
|
r7177 | var tb = json.traceback; | ||
if (tb !== undefined && tb.length > 0) { | ||||
var s = ''; | ||||
var len = tb.length; | ||||
for (var i=0; i<len; i++) { | ||||
s = s + tb[i] + '\n'; | ||||
} | ||||
s = s + '\n'; | ||||
var toinsert = this.create_output_area(); | ||||
MinRK
|
r15783 | var append_text = OutputArea.append_map['text/plain']; | ||
if (append_text) { | ||||
MinRK
|
r16589 | append_text.apply(this, [s, {}, toinsert]).addClass('output_error'); | ||
MinRK
|
r15783 | } | ||
MinRK
|
r12332 | this._safe_append(toinsert); | ||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
OutputArea.prototype.append_stream = function (json) { | ||||
// temporary fix: if stream undefined (json file written prior to this patch), | ||||
// default to most likely stdout: | ||||
MinRK
|
r15783 | if (json.stream === undefined){ | ||
Brian Granger
|
r7177 | json.stream = 'stdout'; | ||
} | ||||
Michael Droettboom
|
r7350 | var text = json.text; | ||
Brian Granger
|
r7177 | var subclass = "output_"+json.stream; | ||
if (this.outputs.length > 0){ | ||||
// have at least one output to consider | ||||
var last = this.outputs[this.outputs.length-1]; | ||||
if (last.output_type == 'stream' && json.stream == last.stream){ | ||||
// latest output was in the same stream, | ||||
// so append directly into its pre tag | ||||
// escape ANSI & HTML specials: | ||||
MinRK
|
r17305 | last.text = utils.fixCarriageReturn(last.text + json.text); | ||
Michael Droettboom
|
r7350 | var pre = this.element.find('div.'+subclass).last().find('pre'); | ||
MinRK
|
r17305 | var html = utils.fixConsole(last.text); | ||
Jonathan Frederic
|
r15406 | // The only user content injected with this HTML call is | ||
Jonathan Frederic
|
r15376 | // escaped by the fixConsole() method. | ||
Michael Droettboom
|
r7339 | pre.html(html); | ||
MinRK
|
r17305 | // return false signals that we merged this output with the previous one, | ||
// and the new output shouldn't be recorded. | ||||
return false; | ||||
Brian Granger
|
r7177 | } | ||
} | ||||
Michael Droettboom
|
r7339 | |||
if (!text.replace("\r", "")) { | ||||
// text is nothing (empty string, \r, etc.) | ||||
// so don't append any elements, which might add undesirable space | ||||
MinRK
|
r17305 | // return true to indicate the output should be recorded. | ||
return true; | ||||
Michael Droettboom
|
r7339 | } | ||
Brian Granger
|
r7177 | // If we got here, attach a new div | ||
var toinsert = this.create_output_area(); | ||||
MinRK
|
r15783 | var append_text = OutputArea.append_map['text/plain']; | ||
if (append_text) { | ||||
append_text.apply(this, [text, {}, toinsert]).addClass("output_stream " + subclass); | ||||
} | ||||
MinRK
|
r12332 | this._safe_append(toinsert); | ||
MinRK
|
r17305 | return true; | ||
Brian Granger
|
r7177 | }; | ||
jon
|
r16187 | OutputArea.prototype.append_display_data = function (json, handle_inserted) { | ||
Brian Granger
|
r7177 | var toinsert = this.create_output_area(); | ||
jon
|
r16187 | if (this.append_mime_type(json, toinsert, handle_inserted)) { | ||
Jonathan Frederic
|
r13454 | this._safe_append(toinsert); | ||
// If we just output latex, typeset it. | ||||
Andrew Jesaitis
|
r16364 | if ((json['text/latex'] !== undefined) || | ||
(json['text/html'] !== undefined) || | ||||
(json['text/markdown'] !== undefined)) { | ||||
Jonathan Frederic
|
r13454 | this.typeset(); | ||
} | ||||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
MinRK
|
r14856 | OutputArea.safe_outputs = { | ||
'text/plain' : true, | ||||
MinRK
|
r15643 | 'text/latex' : true, | ||
MinRK
|
r14856 | 'image/png' : true, | ||
'image/jpeg' : true | ||||
}; | ||||
jon
|
r16187 | OutputArea.prototype.append_mime_type = function (json, element, handle_inserted) { | ||
MinRK
|
r16226 | for (var i=0; i < OutputArea.display_order.length; i++) { | ||
var type = OutputArea.display_order[i]; | ||||
Paul Ivanov
|
r14135 | var append = OutputArea.append_map[type]; | ||
if ((json[type] !== undefined) && append) { | ||||
MinRK
|
r15644 | var value = json[type]; | ||
MinRK
|
r14856 | if (!this.trusted && !OutputArea.safe_outputs[type]) { | ||
MinRK
|
r15644 | // not trusted, sanitize HTML | ||
Brian E. Granger
|
r15634 | if (type==='text/html' || type==='text/svg') { | ||
Jonathan Frederic
|
r17198 | value = security.sanitize_html(value); | ||
MinRK
|
r15644 | } else { | ||
MinRK
|
r15658 | // don't display if we don't know how to sanitize it | ||
MinRK
|
r15661 | console.log("Ignoring untrusted " + type + " output."); | ||
Jonathan Frederic
|
r15728 | continue; | ||
MinRK
|
r14856 | } | ||
} | ||||
Paul Ivanov
|
r14140 | var md = json.metadata || {}; | ||
jon
|
r16187 | var toinsert = append.apply(this, [value, md, element, handle_inserted]); | ||
// Since only the png and jpeg mime types call the inserted | ||||
Jonathan Frederic
|
r16191 | // callback, if the mime type is something other we must call the | ||
jon
|
r16187 | // inserted callback only when the element is actually inserted | ||
// into the DOM. Use a timeout of 0 to do this. | ||||
Jonathan Frederic
|
r16189 | if (['image/png', 'image/jpeg'].indexOf(type) < 0 && handle_inserted !== undefined) { | ||
jon
|
r16187 | setTimeout(handle_inserted, 0); | ||
} | ||||
Jonathan Frederic
|
r17198 | this.events.trigger('output_appended.OutputArea', [type, value, md, toinsert]); | ||
Jonathan Frederic
|
r15729 | return toinsert; | ||
Matthias BUSSONNIER
|
r9540 | } | ||
} | ||||
Jonathan Frederic
|
r15729 | return null; | ||
Brian Granger
|
r7177 | }; | ||
MinRK
|
r15762 | var append_html = function (html, md, element) { | ||
Paul Ivanov
|
r14140 | var type = 'text/html'; | ||
Paul Ivanov
|
r14116 | var toinsert = this.create_output_subarea(md, "output_html rendered_html", type); | ||
Jonathan Frederic
|
r17198 | this.keyboard_manager.register_events(toinsert); | ||
Brian Granger
|
r7177 | toinsert.append(html); | ||
element.append(toinsert); | ||||
Matthias BUSSONNIER
|
r14883 | return toinsert; | ||
Brian Granger
|
r7177 | }; | ||
Andrew Jesaitis
|
r16364 | var append_markdown = function(markdown, md, element) { | ||
var type = 'text/markdown'; | ||||
var toinsert = this.create_output_subarea(md, "output_markdown", type); | ||||
Jonathan Frederic
|
r17198 | var text_and_math = mathjaxutils.remove_math(markdown); | ||
Andrew Jesaitis
|
r16364 | var text = text_and_math[0]; | ||
var math = text_and_math[1]; | ||||
var html = marked.parser(marked.lexer(text)); | ||||
Jonathan Frederic
|
r17198 | html = mathjaxutils.replace_math(html, math); | ||
Andrew Jesaitis
|
r16364 | toinsert.append(html); | ||
element.append(toinsert); | ||||
return toinsert; | ||||
}; | ||||
MinRK
|
r15762 | var append_javascript = function (js, md, element) { | ||
Brian Granger
|
r7177 | // We just eval the JS code, element appears in the local scope. | ||
Paul Ivanov
|
r14140 | var type = 'application/javascript'; | ||
Matthias BUSSONNIER
|
r14884 | var toinsert = this.create_output_subarea(md, "output_javascript", type); | ||
Jonathan Frederic
|
r17198 | this.keyboard_manager.register_events(toinsert); | ||
Matthias BUSSONNIER
|
r14884 | element.append(toinsert); | ||
Jonathan Frederic
|
r15992 | |||
// Fix for ipython/issues/5293, make sure `element` is the area which | ||||
// output can be inserted into at the time of JS execution. | ||||
element = toinsert; | ||||
Matthias BUSSONNIER
|
r8071 | try { | ||
eval(js); | ||||
} catch(err) { | ||||
Brian E. Granger
|
r13792 | console.log(err); | ||
Matthias BUSSONNIER
|
r14884 | this._append_javascript_error(err, toinsert); | ||
Matthias BUSSONNIER
|
r8071 | } | ||
Matthias BUSSONNIER
|
r14884 | return toinsert; | ||
Matthias BUSSONNIER
|
r9555 | }; | ||
Brian Granger
|
r7177 | |||
MinRK
|
r15762 | var append_text = function (data, md, element) { | ||
Paul Ivanov
|
r14140 | var type = 'text/plain'; | ||
Paul Ivanov
|
r14116 | var toinsert = this.create_output_subarea(md, "output_text", type); | ||
Brian Granger
|
r7177 | // escape ANSI & HTML specials in plaintext: | ||
data = utils.fixConsole(data); | ||||
Michael Droettboom
|
r7339 | data = utils.fixCarriageReturn(data); | ||
Erik M. Bray
|
r8528 | data = utils.autoLinkUrls(data); | ||
Jonathan Frederic
|
r15406 | // The only user content injected with this HTML call is | ||
Jonathan Frederic
|
r15376 | // escaped by the fixConsole() method. | ||
Brian Granger
|
r7177 | toinsert.append($("<pre/>").html(data)); | ||
element.append(toinsert); | ||||
Matthias BUSSONNIER
|
r14883 | return toinsert; | ||
Brian Granger
|
r7177 | }; | ||
Jonathan Frederic
|
r16203 | var append_svg = function (svg_html, md, element) { | ||
Paul Ivanov
|
r14140 | var type = 'image/svg+xml'; | ||
Paul Ivanov
|
r14116 | var toinsert = this.create_output_subarea(md, "output_svg", type); | ||
Jonathan Frederic
|
r16203 | |||
// Get the svg element from within the HTML. | ||||
var svg = $('<div />').html(svg_html).find('svg'); | ||||
var svg_area = $('<div />'); | ||||
var width = svg.attr('width'); | ||||
var height = svg.attr('height'); | ||||
svg | ||||
.width('100%') | ||||
.height('100%'); | ||||
svg_area | ||||
.width(width) | ||||
.height(height); | ||||
// The jQuery resize handlers don't seem to work on the svg element. | ||||
// When the svg renders completely, measure it's size and set the parent | ||||
// div to that size. Then set the svg to 100% the size of the parent | ||||
// div and make the parent div resizable. | ||||
this._dblclick_to_reset_size(svg_area, true, false); | ||||
svg_area.append(svg); | ||||
toinsert.append(svg_area); | ||||
Brian Granger
|
r7355 | element.append(toinsert); | ||
Jonathan Frederic
|
r16203 | |||
Matthias BUSSONNIER
|
r14883 | return toinsert; | ||
Brian Granger
|
r7177 | }; | ||
Jonathan Frederic
|
r16203 | OutputArea.prototype._dblclick_to_reset_size = function (img, immediately, resize_parent) { | ||
// Add a resize handler to an element | ||||
// | ||||
// img: jQuery element | ||||
// immediately: bool=False | ||||
// Wait for the element to load before creating the handle. | ||||
// resize_parent: bool=True | ||||
// Should the parent of the element be resized when the element is | ||||
// reset (by double click). | ||||
var callback = function (){ | ||||
MinRK
|
r7601 | var h0 = img.height(); | ||
var w0 = img.width(); | ||||
if (!(h0 && w0)) { | ||||
MinRK
|
r15551 | // zero size, don't make it resizable | ||
Matthias BUSSONNIER
|
r9555 | return; | ||
MinRK
|
r7601 | } | ||
MinRK
|
r7599 | img.resizable({ | ||
aspectRatio: true, | ||||
MinRK
|
r7601 | autoHide: true | ||
}); | ||||
img.dblclick(function () { | ||||
// resize wrapper & image together for some reason: | ||||
img.height(h0); | ||||
img.width(w0); | ||||
Jonathan Frederic
|
r16203 | if (resize_parent === undefined || resize_parent) { | ||
img.parent().height(h0); | ||||
img.parent().width(w0); | ||||
} | ||||
MinRK
|
r7599 | }); | ||
Jonathan Frederic
|
r16203 | }; | ||
if (immediately) { | ||||
callback(); | ||||
} else { | ||||
img.on("load", callback); | ||||
} | ||||
Matthias BUSSONNIER
|
r9555 | }; | ||
MinRK
|
r14727 | |||
var set_width_height = function (img, md, mime) { | ||||
// set width and height of an img element from metadata | ||||
var height = _get_metadata_key(md, 'height', mime); | ||||
MinRK
|
r14807 | if (height !== undefined) img.attr('height', height); | ||
MinRK
|
r14727 | var width = _get_metadata_key(md, 'width', mime); | ||
MinRK
|
r14807 | if (width !== undefined) img.attr('width', width); | ||
MinRK
|
r14727 | }; | ||
jon
|
r16187 | var append_png = function (png, md, element, handle_inserted) { | ||
Paul Ivanov
|
r14140 | var type = 'image/png'; | ||
Paul Ivanov
|
r14116 | var toinsert = this.create_output_subarea(md, "output_png", type); | ||
Jonathan Frederic
|
r16189 | var img = $("<img/>"); | ||
if (handle_inserted !== undefined) { | ||||
Jonathan Frederic
|
r16191 | img.on('load', function(){ | ||
handle_inserted(img); | ||||
}); | ||||
jon
|
r16187 | } | ||
Jonathan Frederic
|
r16190 | img[0].src = 'data:image/png;base64,'+ png; | ||
MinRK
|
r14807 | set_width_height(img, md, 'image/png'); | ||
MinRK
|
r7601 | this._dblclick_to_reset_size(img); | ||
Brian Granger
|
r7352 | toinsert.append(img); | ||
Brian Granger
|
r7177 | element.append(toinsert); | ||
Matthias BUSSONNIER
|
r14883 | return toinsert; | ||
Brian Granger
|
r7177 | }; | ||
jon
|
r16187 | var append_jpeg = function (jpeg, md, element, handle_inserted) { | ||
Paul Ivanov
|
r14140 | var type = 'image/jpeg'; | ||
Paul Ivanov
|
r14116 | var toinsert = this.create_output_subarea(md, "output_jpeg", type); | ||
Jonathan Frederic
|
r16189 | var img = $("<img/>"); | ||
if (handle_inserted !== undefined) { | ||||
Jonathan Frederic
|
r16191 | img.on('load', function(){ | ||
handle_inserted(img); | ||||
}); | ||||
jon
|
r16187 | } | ||
Jonathan Frederic
|
r16190 | img[0].src = 'data:image/jpeg;base64,'+ jpeg; | ||
MinRK
|
r14807 | set_width_height(img, md, 'image/jpeg'); | ||
MinRK
|
r7601 | this._dblclick_to_reset_size(img); | ||
Brian Granger
|
r7352 | toinsert.append(img); | ||
Brian Granger
|
r7177 | element.append(toinsert); | ||
Matthias BUSSONNIER
|
r14883 | return toinsert; | ||
Brian Granger
|
r7177 | }; | ||
MinRK
|
r15762 | var append_pdf = function (pdf, md, element) { | ||
Brian E. Granger
|
r15127 | var type = 'application/pdf'; | ||
var toinsert = this.create_output_subarea(md, "output_pdf", type); | ||||
var a = $('<a/>').attr('href', 'data:application/pdf;base64,'+pdf); | ||||
a.attr('target', '_blank'); | ||||
a.text('View PDF') | ||||
toinsert.append(a); | ||||
element.append(toinsert); | ||||
return toinsert; | ||||
} | ||||
MinRK
|
r15762 | var append_latex = function (latex, md, element) { | ||
Brian Granger
|
r7177 | // This method cannot do the typesetting because the latex first has to | ||
// be on the page. | ||||
Paul Ivanov
|
r14140 | var type = 'text/latex'; | ||
Paul Ivanov
|
r14116 | var toinsert = this.create_output_subarea(md, "output_latex", type); | ||
Brian Granger
|
r7177 | toinsert.append(latex); | ||
element.append(toinsert); | ||||
Matthias BUSSONNIER
|
r14883 | return toinsert; | ||
Brian Granger
|
r7177 | }; | ||
Matthias BUSSONNIER
|
r10776 | |||
Paul Ivanov
|
r14132 | |||
MinRK
|
r13207 | OutputArea.prototype.append_raw_input = function (msg) { | ||
MinRK
|
r10370 | var that = this; | ||
MinRK
|
r10368 | this.expand(); | ||
MinRK
|
r13207 | var content = msg.content; | ||
MinRK
|
r10368 | var area = this.create_output_area(); | ||
MinRK
|
r11612 | |||
// disable any other raw_inputs, if they are left around | ||||
MinRK
|
r15825 | $("div.output_subarea.raw_input_container").remove(); | ||
MinRK
|
r11612 | |||
MinRK
|
r16573 | var input_type = content.password ? 'password' : 'text'; | ||
MinRK
|
r10368 | area.append( | ||
$("<div/>") | ||||
MinRK
|
r15825 | .addClass("box-flex1 output_subarea raw_input_container") | ||
MinRK
|
r10368 | .append( | ||
MinRK
|
r10370 | $("<span/>") | ||
MinRK
|
r15825 | .addClass("raw_input_prompt") | ||
MinRK
|
r10370 | .text(content.prompt) | ||
) | ||||
.append( | ||||
$("<input/>") | ||||
.addClass("raw_input") | ||||
MinRK
|
r16573 | .attr('type', input_type) | ||
MinRK
|
r10934 | .attr("size", 47) | ||
MinRK
|
r10370 | .keydown(function (event, ui) { | ||
// make sure we submit on enter, | ||||
// and don't re-execute the *cell* on shift-enter | ||||
Jonathan Frederic
|
r17198 | if (event.which === keyboard.keycodes.enter) { | ||
MinRK
|
r10370 | that._submit_raw_input(); | ||
return false; | ||||
} | ||||
}) | ||||
MinRK
|
r10368 | ) | ||
MinRK
|
r10370 | ); | ||
Brian E. Granger
|
r14046 | |||
MinRK
|
r10368 | this.element.append(area); | ||
Brian E. Granger
|
r14046 | var raw_input = area.find('input.raw_input'); | ||
// Register events that enable/disable the keyboard manager while raw | ||||
// input is focused. | ||||
Jonathan Frederic
|
r17198 | this.keyboard_manager.register_events(raw_input); | ||
Brian E. Granger
|
r14046 | // Note, the following line used to read raw_input.focus().focus(). | ||
// This seemed to be needed otherwise only the cell would be focused. | ||||
// But with the modal UI, this seems to work fine with one call to focus(). | ||||
raw_input.focus(); | ||||
MinRK
|
r10368 | } | ||
Brian E. Granger
|
r14046 | |||
MinRK
|
r10368 | OutputArea.prototype._submit_raw_input = function (evt) { | ||
MinRK
|
r15825 | var container = this.element.find("div.raw_input_container"); | ||
var theprompt = container.find("span.raw_input_prompt"); | ||||
MinRK
|
r10368 | var theinput = container.find("input.raw_input"); | ||
MinRK
|
r10934 | var value = theinput.val(); | ||
MinRK
|
r16573 | var echo = value; | ||
// don't echo if it's a password | ||||
if (theinput.attr('type') == 'password') { | ||||
echo = '········'; | ||||
} | ||||
MinRK
|
r10368 | var content = { | ||
output_type : 'stream', | ||||
MinRK
|
r17062 | stream : 'stdout', | ||
MinRK
|
r16573 | text : theprompt.text() + echo + '\n' | ||
MinRK
|
r10368 | } | ||
// remove form container | ||||
container.parent().remove(); | ||||
// replace with plaintext version in stdout | ||||
this.append_output(content, false); | ||||
Jonathan Frederic
|
r17198 | this.events.trigger('send_input_reply.Kernel', value); | ||
MinRK
|
r10368 | } | ||
Brian Granger
|
r7177 | |||
MinRK
|
r13207 | OutputArea.prototype.handle_clear_output = function (msg) { | ||
MinRK
|
r14977 | // msg spec v4 had stdout, stderr, display keys | ||
// v4.1 replaced these with just wait | ||||
// The default behavior is the same (stdout=stderr=display=True, wait=False), | ||||
// so v4 messages will still be properly handled, | ||||
// except for the rarely used clearing less than all output. | ||||
this.clear_output(msg.content.wait || false); | ||||
Matthias BUSSONNIER
|
r9555 | }; | ||
Brian Granger
|
r7177 | |||
Jonathan Frederic
|
r12592 | OutputArea.prototype.clear_output = function(wait) { | ||
if (wait) { | ||||
Jonathan Frederic
|
r12582 | |||
Jonathan Frederic
|
r12592 | // If a clear is queued, clear before adding another to the queue. | ||
if (this.clear_queued) { | ||||
this.clear_output(false); | ||||
}; | ||||
this.clear_queued = true; | ||||
} else { | ||||
Jonathan Frederic
|
r12606 | // Fix the output div's height if the clear_output is waiting for | ||
// new output (it is being used in an animation). | ||||
if (this.clear_queued) { | ||||
var height = this.element.height(); | ||||
this.element.height(height); | ||||
this.clear_queued = false; | ||||
} | ||||
Jonathan Frederic
|
r16191 | // Clear all | ||
// Remove load event handlers from img tags because we don't want | ||||
// them to fire if the image is never added to the page. | ||||
this.element.find('img').off('load'); | ||||
Jonathan Frederic
|
r12592 | this.element.html(""); | ||
this.outputs = []; | ||||
MinRK
|
r14856 | this.trusted = true; | ||
Jonathan Frederic
|
r12592 | this.unscroll_area(); | ||
return; | ||||
}; | ||||
Brian Granger
|
r7177 | }; | ||
// JSON serialization | ||||
OutputArea.prototype.fromJSON = function (outputs) { | ||||
var len = outputs.length; | ||||
Paul Ivanov
|
r14127 | var data; | ||
Paul Ivanov
|
r14134 | |||
Brian Granger
|
r7177 | for (var i=0; i<len; i++) { | ||
Paul Ivanov
|
r14130 | data = outputs[i]; | ||
var msg_type = data.output_type; | ||||
MinRK
|
r16589 | if (msg_type == "pyout") { | ||
// pyout message has been renamed to execute_result, | ||||
// but the nbformat has not been updated, | ||||
// so transform back to pyout for json. | ||||
msg_type = data.output_type = "execute_result"; | ||||
} else if (msg_type == "pyerr") { | ||||
// pyerr message has been renamed to error, | ||||
// but the nbformat has not been updated, | ||||
// so transform back to pyerr for json. | ||||
msg_type = data.output_type = "error"; | ||||
} | ||||
if (msg_type === "display_data" || msg_type === "execute_result") { | ||||
Paul Ivanov
|
r14123 | // convert short keys to mime keys | ||
Paul Ivanov
|
r14136 | // TODO: remove mapping of short keys when we update to nbformat 4 | ||
Paul Ivanov
|
r14130 | data = this.rename_keys(data, OutputArea.mime_map_r); | ||
data.metadata = this.rename_keys(data.metadata, OutputArea.mime_map_r); | ||||
MinRK
|
r16667 | // msg spec JSON is an object, nbformat v3 JSON is a JSON string | ||
if (data["application/json"] !== undefined && typeof data["application/json"] === 'string') { | ||||
data["application/json"] = JSON.parse(data["application/json"]); | ||||
} | ||||
Paul Ivanov
|
r14120 | } | ||
Paul Ivanov
|
r14134 | this.append_output(data); | ||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | }; | ||
OutputArea.prototype.toJSON = function () { | ||||
var outputs = []; | ||||
var len = this.outputs.length; | ||||
Paul Ivanov
|
r14127 | var data; | ||
Brian Granger
|
r7177 | for (var i=0; i<len; i++) { | ||
Paul Ivanov
|
r14130 | data = this.outputs[i]; | ||
var msg_type = data.output_type; | ||||
MinRK
|
r16589 | if (msg_type === "display_data" || msg_type === "execute_result") { | ||
Paul Ivanov
|
r14123 | // convert mime keys to short keys | ||
Paul Ivanov
|
r14130 | data = this.rename_keys(data, OutputArea.mime_map); | ||
Paul Ivanov
|
r14136 | data.metadata = this.rename_keys(data.metadata, OutputArea.mime_map); | ||
MinRK
|
r16667 | // msg spec JSON is an object, nbformat v3 JSON is a JSON string | ||
if (data.json !== undefined && typeof data.json !== 'string') { | ||||
data.json = JSON.stringify(data.json); | ||||
} | ||||
Paul Ivanov
|
r14123 | } | ||
MinRK
|
r16589 | if (msg_type == "execute_result") { | ||
// pyout message has been renamed to execute_result, | ||||
// but the nbformat has not been updated, | ||||
// so transform back to pyout for json. | ||||
data.output_type = "pyout"; | ||||
} else if (msg_type == "error") { | ||||
// pyerr message has been renamed to error, | ||||
// but the nbformat has not been updated, | ||||
// so transform back to pyerr for json. | ||||
data.output_type = "pyerr"; | ||||
} | ||||
Paul Ivanov
|
r14123 | outputs[i] = data; | ||
Matthias BUSSONNIER
|
r9555 | } | ||
Brian Granger
|
r7177 | return outputs; | ||
}; | ||||
Matthias BUSSONNIER
|
r14985 | /** | ||
* Class properties | ||||
**/ | ||||
/** | ||||
* Threshold to trigger autoscroll when the OutputArea is resized, | ||||
* typically when new outputs are added. | ||||
* | ||||
* Behavior is undefined if autoscroll is lower than minimum_scroll_threshold, | ||||
* unless it is < 0, in which case autoscroll will never be triggered | ||||
* | ||||
* @property auto_scroll_threshold | ||||
* @type Number | ||||
* @default 100 | ||||
* | ||||
**/ | ||||
OutputArea.auto_scroll_threshold = 100; | ||||
/** | ||||
* Lower limit (in lines) for OutputArea to be made scrollable. OutputAreas | ||||
* shorter than this are never scrolled. | ||||
* | ||||
* @property minimum_scroll_threshold | ||||
* @type Number | ||||
* @default 20 | ||||
* | ||||
**/ | ||||
OutputArea.minimum_scroll_threshold = 20; | ||||
OutputArea.mime_map = { | ||||
"text/plain" : "text", | ||||
"text/html" : "html", | ||||
"image/svg+xml" : "svg", | ||||
"image/png" : "png", | ||||
"image/jpeg" : "jpeg", | ||||
"text/latex" : "latex", | ||||
"application/json" : "json", | ||||
"application/javascript" : "javascript", | ||||
}; | ||||
OutputArea.mime_map_r = { | ||||
"text" : "text/plain", | ||||
"html" : "text/html", | ||||
"svg" : "image/svg+xml", | ||||
"png" : "image/png", | ||||
"jpeg" : "image/jpeg", | ||||
"latex" : "text/latex", | ||||
"json" : "application/json", | ||||
"javascript" : "application/javascript", | ||||
}; | ||||
OutputArea.display_order = [ | ||||
'application/javascript', | ||||
'text/html', | ||||
Andrew Jesaitis
|
r16364 | 'text/markdown', | ||
Matthias BUSSONNIER
|
r14985 | 'text/latex', | ||
'image/svg+xml', | ||||
'image/png', | ||||
'image/jpeg', | ||||
Brian E. Granger
|
r15127 | 'application/pdf', | ||
Matthias BUSSONNIER
|
r14985 | 'text/plain' | ||
]; | ||||
OutputArea.append_map = { | ||||
MinRK
|
r15762 | "text/plain" : append_text, | ||
"text/html" : append_html, | ||||
Andrew Jesaitis
|
r16364 | "text/markdown": append_markdown, | ||
MinRK
|
r15762 | "image/svg+xml" : append_svg, | ||
"image/png" : append_png, | ||||
"image/jpeg" : append_jpeg, | ||||
"text/latex" : append_latex, | ||||
"application/javascript" : append_javascript, | ||||
"application/pdf" : append_pdf | ||||
Matthias BUSSONNIER
|
r14985 | }; | ||
Brian Granger
|
r7177 | |||
Jonathan Frederic
|
r17198 | // For backwards compatability. | ||
Brian Granger
|
r7177 | IPython.OutputArea = OutputArea; | ||
Jonathan Frederic
|
r17201 | return {'OutputArea': OutputArea}; | ||
Jonathan Frederic
|
r17198 | }); | ||