textcell.js
574 lines
| 16.6 KiB
| application/javascript
|
JavascriptLexer
Brian E. Granger
|
r4609 | //---------------------------------------------------------------------------- | ||
Aron Ahmadia
|
r8565 | // Copyright (C) 2008-2012 The IPython Development Team | ||
Brian E. Granger
|
r4609 | // | ||
// Distributed under the terms of the BSD License. The full license is in | ||||
// the file COPYING, distributed as part of this software. | ||||
//---------------------------------------------------------------------------- | ||||
Brian E. Granger
|
r4349 | |||
//============================================================================ | ||||
Brian Granger
|
r4508 | // TextCell | ||
Brian E. Granger
|
r4349 | //============================================================================ | ||
Matthias BUSSONNIER
|
r10165 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
A module that allow to create different type of Text Cell | ||||
Matthias BUSSONNIER
|
r8739 | @module IPython | ||
@namespace IPython | ||||
Matthias BUSSONNIER
|
r8709 | */ | ||
Brian E. Granger
|
r4352 | var IPython = (function (IPython) { | ||
Matthias BUSSONNIER
|
r11526 | "use strict"; | ||
Brian E. Granger
|
r4349 | |||
Brian Granger
|
r4508 | // TextCell base class | ||
Bradley M. Froehle
|
r8168 | var key = IPython.utils.keycodes; | ||
Brian Granger
|
r4508 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
Matthias BUSSONNIER
|
r8711 | * Construct a new TextCell, codemirror mode is by default 'htmlmixed', and cell type is 'text' | ||
* cell start as not redered. | ||||
* | ||||
Matthias BUSSONNIER
|
r8709 | * @class TextCell | ||
Matthias BUSSONNIER
|
r8711 | * @constructor TextCell | ||
MinRK
|
r11288 | * @extend IPython.Cell | ||
Matthias BUSSONNIER
|
r9537 | * @param {object|undefined} [options] | ||
* @param [options.cm_config] {object} config to pass to CodeMirror, will extend/overwrite default config | ||||
Matthias BUSSONNIER
|
r10165 | * @param [options.placeholder] {string} default string to use when souce in empty for rendering (only use in some TextCell subclass) | ||
Matthias BUSSONNIER
|
r8709 | */ | ||
Matthias BUSSONNIER
|
r9537 | var TextCell = function (options) { | ||
Matthias BUSSONNIER
|
r10165 | // in all TextCell/Cell subclasses | ||
// do not assign most of members here, just pass it down | ||||
// in the options dict potentially overwriting what you wish. | ||||
// they will be assigned in the base class. | ||||
Matthias BUSSONNIER
|
r9537 | |||
Matthias BUSSONNIER
|
r10165 | // we cannot put this as a class key as it has handle to "this". | ||
Matthias BUSSONNIER
|
r9537 | var cm_overwrite_options = { | ||
onKeyEvent: $.proxy(this.handle_codemirror_keyevent,this) | ||||
}; | ||||
Matthias BUSSONNIER
|
r10165 | options = this.mergeopt(TextCell,options,{cm_config:cm_overwrite_options}); | ||
Matthias BUSSONNIER
|
r9537 | |||
Matthias BUSSONNIER
|
r10165 | IPython.Cell.apply(this, [options]); | ||
Matthias BUSSONNIER
|
r9537 | |||
Brian E. Granger
|
r4352 | this.rendered = false; | ||
Brian Granger
|
r4508 | this.cell_type = this.cell_type || 'text'; | ||
Brian E. Granger
|
r4352 | }; | ||
Matthias BUSSONNIER
|
r10165 | TextCell.prototype = new IPython.Cell(); | ||
TextCell.options_default = { | ||||
cm_config : { | ||||
extraKeys: {"Tab": "indentMore","Shift-Tab" : "indentLess"}, | ||||
mode: 'htmlmixed', | ||||
Matthias BUSSONNIER
|
r9537 | lineWrapping : true, | ||
} | ||||
Matthias BUSSONNIER
|
r10165 | }; | ||
Matthias BUSSONNIER
|
r9537 | |||
Brian E. Granger
|
r4352 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* Create the DOM element of the TextCell | ||||
Matthias BUSSONNIER
|
r8709 | * @method create_element | ||
* @private | ||||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.create_element = function () { | ||
Matthias BUSSONNIER
|
r9073 | IPython.Cell.prototype.create_element.apply(this, arguments); | ||
Matthias BUSSONNIER
|
r10217 | var cell = $("<div>").addClass('cell text_cell border-box-sizing'); | ||
Brian E. Granger
|
r4629 | cell.attr('tabindex','2'); | ||
Brian Granger
|
r9144 | |||
Brian E. Granger
|
r9142 | this.celltoolbar = new IPython.CellToolbar(this); | ||
cell.append(this.celltoolbar.element); | ||||
Brian Granger
|
r9144 | |||
Brian Granger
|
r5946 | var input_area = $('<div/>').addClass('text_cell_input border-box-sizing'); | ||
Matthias BUSSONNIER
|
r9537 | this.code_mirror = CodeMirror(input_area.get(0), this.cm_config); | ||
Matthias BUSSONNIER
|
r10165 | |||
Brian E. Granger
|
r4499 | // The tabindex=-1 makes this div focusable. | ||
Brian Granger
|
r5946 | var render_area = $('<div/>').addClass('text_cell_render border-box-sizing'). | ||
Aron Ahmadia
|
r8661 | addClass('rendered_html').attr('tabindex','-1'); | ||
Brian E. Granger
|
r4499 | cell.append(input_area).append(render_area); | ||
Brian E. Granger
|
r4352 | this.element = cell; | ||
}; | ||||
Matthias BUSSONNIER
|
r8711 | /** | ||
* Bind the DOM evet to cell actions | ||||
Matthias BUSSONNIER
|
r8709 | * Need to be called after TextCell.create_element | ||
* @private | ||||
* @method bind_event | ||||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.bind_events = function () { | ||
Brian E. Granger
|
r4352 | IPython.Cell.prototype.bind_events.apply(this); | ||
var that = this; | ||||
this.element.keydown(function (event) { | ||||
Brian Granger
|
r6050 | if (event.which === 13 && !event.shiftKey) { | ||
Brian E. Granger
|
r4352 | if (that.rendered) { | ||
that.edit(); | ||||
Brian Granger
|
r5942 | return false; | ||
Brian Granger
|
r5946 | }; | ||
}; | ||||
}); | ||||
this.element.dblclick(function () { | ||||
that.edit(); | ||||
Brian E. Granger
|
r4352 | }); | ||
}; | ||||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* This method gets called in CodeMirror's onKeyDown/onKeyPress | ||||
Matthias BUSSONNIER
|
r8709 | * handlers and is used to provide custom key handling. | ||
* | ||||
Matthias BUSSONNIER
|
r8711 | * Subclass should override this method to have custom handeling | ||
* | ||||
Matthias BUSSONNIER
|
r8709 | * @method handle_codemirror_keyevent | ||
Matthias BUSSONNIER
|
r8711 | * @param {CodeMirror} editor - The codemirror instance bound to the cell | ||
Matthias BUSSONNIER
|
r8709 | * @param {event} event - | ||
Matthias BUSSONNIER
|
r8711 | * @return {Boolean} `true` if CodeMirror should ignore the event, `false` Otherwise | ||
Matthias BUSSONNIER
|
r8709 | */ | ||
Brian Granger
|
r5942 | TextCell.prototype.handle_codemirror_keyevent = function (editor, event) { | ||
Matthias BUSSONNIER
|
r8709 | |||
Brian Granger
|
r5942 | if (event.keyCode === 13 && (event.shiftKey || event.ctrlKey)) { | ||
// Always ignore shift-enter in CodeMirror as we handle it. | ||||
return true; | ||||
} | ||||
return false; | ||||
}; | ||||
Matthias BUSSONNIER
|
r8709 | /** | ||
Matthias BUSSONNIER
|
r8711 | * Select the current cell and trigger 'focus' | ||
Matthias BUSSONNIER
|
r8709 | * @method select | ||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.select = function () { | ||
Brian E. Granger
|
r4352 | IPython.Cell.prototype.select.apply(this); | ||
Brian Granger
|
r4508 | var output = this.element.find("div.text_cell_render"); | ||
Brian E. Granger
|
r4352 | output.trigger('focus'); | ||
}; | ||||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* unselect the current cell and `render` it | ||||
Matthias BUSSONNIER
|
r8709 | * @method unselect | ||
*/ | ||||
MinRK
|
r5833 | TextCell.prototype.unselect = function() { | ||
// render on selection of another cell | ||||
this.render(); | ||||
IPython.Cell.prototype.unselect.apply(this); | ||||
}; | ||||
Matthias BUSSONNIER
|
r8711 | /** | ||
* | ||||
* put the current cell in edition mode | ||||
Matthias BUSSONNIER
|
r8709 | * @method edit | ||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.edit = function () { | ||
Brian E. Granger
|
r4352 | if (this.rendered === true) { | ||
Brian Granger
|
r4508 | var text_cell = this.element; | ||
Matthias BUSSONNIER
|
r8709 | var output = text_cell.find("div.text_cell_render"); | ||
Brian E. Granger
|
r4352 | output.hide(); | ||
Brian Granger
|
r4508 | text_cell.find('div.text_cell_input').show(); | ||
Brian Granger
|
r5946 | this.code_mirror.refresh(); | ||
Brian E. Granger
|
r4499 | this.code_mirror.focus(); | ||
Brian Granger
|
r5971 | // We used to need an additional refresh() after the focus, but | ||
// it appears that this has been fixed in CM. This bug would show | ||||
// up on FF when a newly loaded markdown cell was edited. | ||||
Brian E. Granger
|
r4352 | this.rendered = false; | ||
Brian Granger
|
r5943 | if (this.get_text() === this.placeholder) { | ||
this.set_text(''); | ||||
Brian Granger
|
r5946 | this.refresh(); | ||
Stefan van der Walt
|
r5479 | } | ||
} | ||||
Brian E. Granger
|
r4349 | }; | ||
Matthias BUSSONNIER
|
r8711 | /** | ||
* Empty, Subclasses must define render. | ||||
Matthias BUSSONNIER
|
r8709 | * @method render | ||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.render = function () {}; | ||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
* setter: {{#crossLink "TextCell/set_text"}}{{/crossLink}} | ||||
* @method get_text | ||||
* @retrun {string} CodeMirror current text value | ||||
*/ | ||||
Brian Granger
|
r5943 | TextCell.prototype.get_text = function() { | ||
Brian E. Granger
|
r4499 | return this.code_mirror.getValue(); | ||
Brian E. Granger
|
r4352 | }; | ||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
* @param {string} text - Codemiror text value | ||||
* @see TextCell#get_text | ||||
* @method set_text | ||||
* */ | ||||
Brian Granger
|
r5943 | TextCell.prototype.set_text = function(text) { | ||
Brian E. Granger
|
r4499 | this.code_mirror.setValue(text); | ||
this.code_mirror.refresh(); | ||||
Brian E. Granger
|
r4352 | }; | ||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
* setter :{{#crossLink "TextCell/set_rendered"}}{{/crossLink}} | ||||
* @method get_rendered | ||||
* @return {html} html of rendered element | ||||
* */ | ||||
Brian E. Granger
|
r4513 | TextCell.prototype.get_rendered = function() { | ||
Brian Granger
|
r4508 | return this.element.find('div.text_cell_render').html(); | ||
}; | ||||
Matthias BUSSONNIER
|
r8709 | /** | ||
* @method set_rendered | ||||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.set_rendered = function(text) { | ||
this.element.find('div.text_cell_render').html(text); | ||||
Brian E. Granger
|
r4352 | }; | ||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
* not deprecated, but implementation wrong | ||||
* @method at_top | ||||
* @deprecated | ||||
* @return {Boolean} true is cell rendered, false otherwise | ||||
* I doubt this is what it is supposed to do | ||||
* this implementation is completly false | ||||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.at_top = function () { | ||
Brian E. Granger
|
r4352 | if (this.rendered) { | ||
return true; | ||||
} else { | ||||
return false; | ||||
} | ||||
}; | ||||
Matthias BUSSONNIER
|
r8709 | /** | ||
* not deprecated, but implementation wrong | ||||
* @method at_bottom | ||||
* @deprecated | ||||
* @return {Boolean} true is cell rendered, false otherwise | ||||
* I doubt this is what it is supposed to do | ||||
* this implementation is completly false | ||||
* */ | ||||
Brian Granger
|
r4508 | TextCell.prototype.at_bottom = function () { | ||
Brian E. Granger
|
r4352 | if (this.rendered) { | ||
return true; | ||||
} else { | ||||
return false; | ||||
} | ||||
}; | ||||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* Create Text cell from JSON | ||||
Matthias BUSSONNIER
|
r8709 | * @param {json} data - JSON serialized text-cell | ||
Matthias BUSSONNIER
|
r8711 | * @method fromJSON | ||
Matthias BUSSONNIER
|
r8709 | */ | ||
Brian Granger
|
r4508 | TextCell.prototype.fromJSON = function (data) { | ||
MinRK
|
r7523 | IPython.Cell.prototype.fromJSON.apply(this, arguments); | ||
Brian Granger
|
r4508 | if (data.cell_type === this.cell_type) { | ||
Brian E. Granger
|
r4499 | if (data.source !== undefined) { | ||
Brian Granger
|
r5943 | this.set_text(data.source); | ||
Paul Ivanov
|
r7587 | // make this value the starting point, so that we can only undo | ||
// to this state, instead of a blank cell | ||||
this.code_mirror.clearHistory(); | ||||
Brian E. Granger
|
r4513 | this.set_rendered(data.rendered || ''); | ||
this.rendered = false; | ||||
this.render(); | ||||
Stefan van der Walt
|
r5479 | } | ||
} | ||||
Brian E. Granger
|
r4513 | }; | ||
Brian E. Granger
|
r4349 | |||
Matthias BUSSONNIER
|
r8711 | /** Generate JSON from cell | ||
* @return {object} cell data serialised to json | ||||
*/ | ||||
Brian Granger
|
r4508 | TextCell.prototype.toJSON = function () { | ||
MinRK
|
r7523 | var data = IPython.Cell.prototype.toJSON.apply(this); | ||
Brian Granger
|
r4508 | data.cell_type = this.cell_type; | ||
Brian Granger
|
r5943 | data.source = this.get_text(); | ||
Brian E. Granger
|
r4484 | return data; | ||
Brian E. Granger
|
r4349 | }; | ||
Brian Granger
|
r4508 | |||
Matthias BUSSONNIER
|
r8709 | /** | ||
Matthias BUSSONNIER
|
r8711 | * @class MarkdownCell | ||
* @constructor MarkdownCell | ||||
MinRK
|
r11288 | * @extends IPython.HTMLCell | ||
Matthias BUSSONNIER
|
r8709 | */ | ||
Matthias BUSSONNIER
|
r10165 | var MarkdownCell = function (options) { | ||
var options = options || {}; | ||||
options = this.mergeopt(MarkdownCell,options); | ||||
TextCell.apply(this, [options]); | ||||
Brian Granger
|
r4508 | this.cell_type = 'markdown'; | ||
}; | ||||
Matthias BUSSONNIER
|
r10165 | MarkdownCell.options_default = { | ||
cm_config: { | ||||
MinRK
|
r10539 | mode: 'gfm' | ||
Matthias BUSSONNIER
|
r10165 | }, | ||
placeholder: "Type *Markdown* and LaTeX: $\\alpha^2$" | ||||
} | ||||
Brian Granger
|
r4508 | |||
MarkdownCell.prototype = new TextCell(); | ||||
Matthias BUSSONNIER
|
r8711 | /** | ||
* @method render | ||||
*/ | ||||
Brian Granger
|
r4508 | MarkdownCell.prototype.render = function () { | ||
if (this.rendered === false) { | ||||
Brian Granger
|
r5943 | var text = this.get_text(); | ||
Jessica B. Hamrick
|
r11843 | var math = null; | ||
Stefan van der Walt
|
r5479 | if (text === "") { text = this.placeholder; } | ||
Jessica B. Hamrick
|
r11843 | var text_and_math = IPython.mathjaxutils.remove_math(text); | ||
text = text_and_math[0]; | ||||
math = text_and_math[1]; | ||||
MinRK
|
r10536 | var html = marked.parser(marked.lexer(text)); | ||
Matthias BUSSONNIER
|
r11524 | html = $(IPython.mathjaxutils.replace_math(html, math)); | ||
MinRK
|
r11211 | // links in markdown cells should open in new tabs | ||
MinRK
|
r11805 | html.find("a[href]").not('[href^="#"]').attr("target", "_blank"); | ||
Aron Ahmadia
|
r8662 | try { | ||
this.set_rendered(html); | ||||
} catch (e) { | ||||
console.log("Error running Javascript in Markdown:"); | ||||
console.log(e); | ||||
this.set_rendered($("<div/>").addClass("js-error").html( | ||||
"Error rendering Markdown!<br/>" + e.toString()) | ||||
); | ||||
MinRK
|
r8052 | } | ||
Aron Ahmadia
|
r8662 | this.element.find('div.text_cell_input').hide(); | ||
this.element.find("div.text_cell_render").show(); | ||||
this.typeset() | ||||
Brian Granger
|
r4508 | this.rendered = true; | ||
Stefan van der Walt
|
r5479 | } | ||
Brian Granger
|
r4508 | }; | ||
MinRK
|
r6248 | // RawCell | ||
Brian Granger
|
r4508 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* @class RawCell | ||||
* @constructor RawCell | ||||
MinRK
|
r11288 | * @extends IPython.TextCell | ||
Matthias BUSSONNIER
|
r8709 | */ | ||
Matthias BUSSONNIER
|
r10165 | var RawCell = function (options) { | ||
options = this.mergeopt(RawCell,options) | ||||
TextCell.apply(this, [options]); | ||||
MinRK
|
r6248 | this.cell_type = 'raw'; | ||
Matthias BUSSONNIER
|
r8202 | |||
Matthias BUSSONNIER
|
r10165 | var that = this | ||
Matthias BUSSONNIER
|
r8202 | this.element.focusout( | ||
function() { that.auto_highlight(); } | ||||
); | ||||
Brian Granger
|
r4508 | }; | ||
Matthias BUSSONNIER
|
r10165 | RawCell.options_default = { | ||
placeholder : "Type plain text and LaTeX: $\\alpha^2$" | ||||
}; | ||||
Brian Granger
|
r4508 | |||
MinRK
|
r6248 | RawCell.prototype = new TextCell(); | ||
Brian Granger
|
r4508 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* Trigger autodetection of highlight scheme for current cell | ||||
* @method auto_highlight | ||||
*/ | ||||
Matthias BUSSONNIER
|
r8202 | RawCell.prototype.auto_highlight = function () { | ||
this._auto_highlight(IPython.config.raw_cell_highlight); | ||||
}; | ||||
Brian Granger
|
r4508 | |||
Matthias BUSSONNIER
|
r8711 | /** @method render **/ | ||
MinRK
|
r6248 | RawCell.prototype.render = function () { | ||
Brian Granger
|
r6017 | this.rendered = true; | ||
this.edit(); | ||||
}; | ||||
Matthias BUSSONNIER
|
r8711 | /** @method handle_codemirror_keyevent **/ | ||
Bradley M. Froehle
|
r8168 | RawCell.prototype.handle_codemirror_keyevent = function (editor, event) { | ||
var that = this; | ||||
if (event.which === key.UPARROW && event.type === 'keydown') { | ||||
// If we are not at the top, let CM handle the up arrow and | ||||
// prevent the global keydown handler from handling it. | ||||
if (!that.at_top()) { | ||||
event.stop(); | ||||
return false; | ||||
} else { | ||||
return true; | ||||
}; | ||||
} else if (event.which === key.DOWNARROW && event.type === 'keydown') { | ||||
// If we are not at the bottom, let CM handle the down arrow and | ||||
// prevent the global keydown handler from handling it. | ||||
if (!that.at_bottom()) { | ||||
event.stop(); | ||||
return false; | ||||
} else { | ||||
return true; | ||||
}; | ||||
}; | ||||
return false; | ||||
}; | ||||
Matthias BUSSONNIER
|
r8711 | /** @method select **/ | ||
MinRK
|
r6248 | RawCell.prototype.select = function () { | ||
Brian Granger
|
r6017 | IPython.Cell.prototype.select.apply(this); | ||
this.code_mirror.refresh(); | ||||
this.code_mirror.focus(); | ||||
}; | ||||
Matthias BUSSONNIER
|
r8711 | /** @method at_top **/ | ||
MinRK
|
r6248 | RawCell.prototype.at_top = function () { | ||
Brian Granger
|
r6017 | var cursor = this.code_mirror.getCursor(); | ||
Bradley M. Froehle
|
r8168 | if (cursor.line === 0 && cursor.ch === 0) { | ||
Brian Granger
|
r6017 | return true; | ||
} else { | ||||
return false; | ||||
Stefan van der Walt
|
r5479 | } | ||
Brian Granger
|
r4508 | }; | ||
Matthias BUSSONNIER
|
r8711 | /** @method at_bottom **/ | ||
MinRK
|
r6248 | RawCell.prototype.at_bottom = function () { | ||
Brian Granger
|
r6017 | var cursor = this.code_mirror.getCursor(); | ||
Bradley M. Froehle
|
r8168 | if (cursor.line === (this.code_mirror.lineCount()-1) && cursor.ch === this.code_mirror.getLine(cursor.line).length) { | ||
Brian Granger
|
r6017 | return true; | ||
} else { | ||||
return false; | ||||
} | ||||
Brian Granger
|
r4508 | }; | ||
Matthias BUSSONNIER
|
r8714 | /** | ||
Matthias BUSSONNIER
|
r8711 | * @class HeadingCell | ||
MinRK
|
r11288 | * @extends IPython.TextCell | ||
Matthias BUSSONNIER
|
r8711 | */ | ||
/** | ||||
* @constructor HeadingCell | ||||
MinRK
|
r11288 | * @extends IPython.TextCell | ||
Matthias BUSSONNIER
|
r8709 | */ | ||
Matthias BUSSONNIER
|
r10165 | var HeadingCell = function (options) { | ||
options = this.mergeopt(HeadingCell,options) | ||||
TextCell.apply(this, [options]); | ||||
Matthias BUSSONNIER
|
r8711 | /** | ||
* heading level of the cell, use getter and setter to access | ||||
* @property level | ||||
*/ | ||||
Brian Granger
|
r6018 | this.level = 1; | ||
Matthias BUSSONNIER
|
r8711 | this.cell_type = 'heading'; | ||
Brian Granger
|
r6018 | }; | ||
Matthias BUSSONNIER
|
r10165 | HeadingCell.options_default = { | ||
placeholder: "Type Heading Here" | ||||
}; | ||||
Brian Granger
|
r6018 | |||
HeadingCell.prototype = new TextCell(); | ||||
Matthias BUSSONNIER
|
r8711 | /** @method fromJSON */ | ||
MinRK
|
r6156 | HeadingCell.prototype.fromJSON = function (data) { | ||
if (data.level != undefined){ | ||||
this.level = data.level; | ||||
} | ||||
Matthias BUSSONNIER
|
r10165 | TextCell.prototype.fromJSON.apply(this, arguments); | ||
MinRK
|
r6156 | }; | ||
Matthias BUSSONNIER
|
r8711 | /** @method toJSON */ | ||
MinRK
|
r6156 | HeadingCell.prototype.toJSON = function () { | ||
Matthias BUSSONNIER
|
r10165 | var data = TextCell.prototype.toJSON.apply(this); | ||
MinRK
|
r6156 | data.level = this.get_level(); | ||
return data; | ||||
}; | ||||
MinRK
|
r13954 | /** | ||
* can the cell be split into two cells | ||||
* @method is_splittable | ||||
**/ | ||||
HeadingCell.prototype.is_splittable = function () { | ||||
return false; | ||||
}; | ||||
/** | ||||
* can the cell be merged with other cells | ||||
* @method is_mergeable | ||||
**/ | ||||
HeadingCell.prototype.is_mergeable = function () { | ||||
return false; | ||||
}; | ||||
MinRK
|
r6156 | |||
Matthias BUSSONNIER
|
r8711 | /** | ||
* Change heading level of cell, and re-render | ||||
* @method set_level | ||||
*/ | ||||
Brian Granger
|
r6019 | HeadingCell.prototype.set_level = function (level) { | ||
this.level = level; | ||||
if (this.rendered) { | ||||
this.rendered = false; | ||||
this.render(); | ||||
}; | ||||
}; | ||||
Matthias BUSSONNIER
|
r8709 | /** The depth of header cell, based on html (h1 to h6) | ||
Matthias BUSSONNIER
|
r8711 | * @method get_level | ||
Matthias BUSSONNIER
|
r8709 | * @return {integer} level - for 1 to 6 | ||
*/ | ||||
Brian Granger
|
r6019 | HeadingCell.prototype.get_level = function () { | ||
return this.level; | ||||
}; | ||||
MinRK
|
r11288 | HeadingCell.prototype.set_rendered = function (html) { | ||
this.element.find("div.text_cell_render").html(html); | ||||
Brian Granger
|
r6019 | }; | ||
Brian Granger
|
r6018 | |||
HeadingCell.prototype.get_rendered = function () { | ||||
var r = this.element.find("div.text_cell_render"); | ||||
return r.children().first().html(); | ||||
Brian Granger
|
r6019 | }; | ||
Brian Granger
|
r6018 | |||
HeadingCell.prototype.render = function () { | ||||
if (this.rendered === false) { | ||||
var text = this.get_text(); | ||||
Jessica B. Hamrick
|
r11843 | var math = null; | ||
MinRK
|
r11290 | // Markdown headings must be a single line | ||
text = text.replace(/\n/g, ' '); | ||||
Brian Granger
|
r6018 | if (text === "") { text = this.placeholder; } | ||
MinRK
|
r11288 | text = Array(this.level + 1).join("#") + " " + text; | ||
Matthias BUSSONNIER
|
r11526 | var text_and_math = IPython.mathjaxutils.remove_math(text); | ||
Jessica B. Hamrick
|
r11843 | text = text_and_math[0]; | ||
math = text_and_math[1]; | ||||
MinRK
|
r11288 | var html = marked.parser(marked.lexer(text)); | ||
Matthias BUSSONNIER
|
r11526 | var h = $(IPython.mathjaxutils.replace_math(html, math)); | ||
MinRK
|
r11291 | // add id and linkback anchor | ||
var hash = h.text().replace(/ /g, '-'); | ||||
h.attr('id', hash); | ||||
h.append( | ||||
MinRK
|
r11288 | $('<a/>') | ||
.addClass('anchor-link') | ||||
.attr('href', '#' + hash) | ||||
.text('ΒΆ') | ||||
); | ||||
this.set_rendered(h); | ||||
Brian Granger
|
r6018 | this.typeset(); | ||
this.element.find('div.text_cell_input').hide(); | ||||
this.element.find("div.text_cell_render").show(); | ||||
this.rendered = true; | ||||
Brian Granger
|
r6019 | }; | ||
Brian Granger
|
r6018 | }; | ||
Brian Granger
|
r4508 | IPython.TextCell = TextCell; | ||
IPython.MarkdownCell = MarkdownCell; | ||||
MinRK
|
r6248 | IPython.RawCell = RawCell; | ||
Brian Granger
|
r6019 | IPython.HeadingCell = HeadingCell; | ||
Brian Granger
|
r4508 | |||
Brian E. Granger
|
r4349 | |||
Brian E. Granger
|
r4352 | return IPython; | ||
Brian E. Granger
|
r4349 | |||
Brian E. Granger
|
r4352 | }(IPython)); | ||
Brian E. Granger
|
r4349 | |||