editor.js
228 lines
| 7.9 KiB
| application/javascript
|
JavascriptLexer
Thomas Kluyver
|
r19013 | // Copyright (c) IPython Development Team. | ||
// Distributed under the terms of the Modified BSD License. | ||||
define([ | ||||
'jquery', | ||||
'base/js/utils', | ||||
'codemirror/lib/codemirror', | ||||
'codemirror/mode/meta', | ||||
Min RK
|
r19303 | 'codemirror/addon/comment/comment', | ||
'codemirror/addon/dialog/dialog', | ||||
'codemirror/addon/edit/closebrackets', | ||||
'codemirror/addon/edit/matchbrackets', | ||||
'codemirror/addon/search/searchcursor', | ||||
'codemirror/addon/search/search', | ||||
'codemirror/keymap/emacs', | ||||
'codemirror/keymap/sublime', | ||||
'codemirror/keymap/vim', | ||||
Thomas Kluyver
|
r19013 | ], | ||
function($, | ||||
utils, | ||||
CodeMirror | ||||
) { | ||||
Min RK
|
r19303 | "use strict"; | ||
Min RK
|
r19319 | |||
Thomas Kluyver
|
r19013 | var Editor = function(selector, options) { | ||
Min RK
|
r19303 | var that = this; | ||
Thomas Kluyver
|
r19013 | this.selector = selector; | ||
Bussonnier Matthias
|
r20208 | this.clean = false; | ||
Thomas Kluyver
|
r19013 | this.contents = options.contents; | ||
this.events = options.events; | ||||
this.base_url = options.base_url; | ||||
this.file_path = options.file_path; | ||||
Min RK
|
r19303 | this.config = options.config; | ||
this.codemirror = new CodeMirror($(this.selector)[0]); | ||||
Bussonnier Matthias
|
r20208 | this.codemirror.on('changes', function(cm, changes){ | ||
that._clean_state(); | ||||
}); | ||||
Min RK
|
r19304 | this.generation = -1; | ||
Thomas Kluyver
|
r19013 | |||
Thomas Kluyver
|
r19020 | // It appears we have to set commands on the CodeMirror class, not the | ||
// instance. I'd like to be wrong, but since there should only be one CM | ||||
// instance on the page, this is good enough for now. | ||||
CodeMirror.commands.save = $.proxy(this.save, this); | ||||
Thomas Kluyver
|
r19015 | this.save_enabled = false; | ||
Min RK
|
r19303 | |||
this.config.loaded.then(function () { | ||||
// load codemirror config | ||||
var cfg = that.config.data.Editor || {}; | ||||
var cmopts = $.extend(true, {}, // true = recursive copy | ||||
Editor.default_codemirror_options, | ||||
cfg.codemirror_options || {} | ||||
); | ||||
Min RK
|
r19309 | that._set_codemirror_options(cmopts); | ||
Min RK
|
r19303 | that.events.trigger('config_changed.Editor', {config: that.config}); | ||
Bussonnier Matthias
|
r20208 | that._clean_state(); | ||
Min RK
|
r19303 | }); | ||
Bussonnier Matthias
|
r20208 | this.clean_sel = $('<div/>'); | ||
$('.last_modified').before(this.clean_sel); | ||||
this.clean_sel.addClass('dirty-indicator-dirty'); | ||||
Min RK
|
r19303 | }; | ||
// default CodeMirror options | ||||
Editor.default_codemirror_options = { | ||||
extraKeys: { | ||||
"Tab" : "indentMore", | ||||
}, | ||||
indentUnit: 4, | ||||
theme: "ipython", | ||||
lineNumbers: true, | ||||
Min RK
|
r19341 | lineWrapping: true, | ||
Thomas Kluyver
|
r19013 | }; | ||
Editor.prototype.load = function() { | ||||
Min RK
|
r19303 | /** load the file */ | ||
Thomas Kluyver
|
r19015 | var that = this; | ||
Thomas Kluyver
|
r19013 | var cm = this.codemirror; | ||
Min RK
|
r19303 | return this.contents.get(this.file_path, {type: 'file', format: 'text'}) | ||
Thomas Kluyver
|
r19015 | .then(function(model) { | ||
cm.setValue(model.content); | ||||
Scott Sanderson
|
r19089 | |||
// Setting the file's initial value creates a history entry, | ||||
// which we don't want. | ||||
cm.clearHistory(); | ||||
Min RK
|
r19321 | that._set_mode_for_model(model); | ||
Thomas Kluyver
|
r19015 | that.save_enabled = true; | ||
Min RK
|
r19304 | that.generation = cm.changeGeneration(); | ||
Min RK
|
r19316 | that.events.trigger("file_loaded.Editor", model); | ||
Bussonnier Matthias
|
r20208 | that._clean_state(); | ||
Bussonnier Matthias
|
r20142 | }).catch( | ||
Thomas Kluyver
|
r19015 | function(error) { | ||
Min RK
|
r19337 | that.events.trigger("file_load_failed.Editor", error); | ||
Bussonnier Matthias
|
r20142 | if (((error.xhr||{}).responseJSON||{}).reason === 'bad format') { | ||
Min RK
|
r19337 | window.location = utils.url_path_join( | ||
that.base_url, | ||||
'files', | ||||
that.file_path | ||||
); | ||||
Bussonnier Matthias
|
r20142 | } else { | ||
console.warn('Error while loading: the error was:') | ||||
console.warn(error) | ||||
Min RK
|
r19337 | } | ||
Thomas Kluyver
|
r19015 | cm.setValue("Error! " + error.message + | ||
Bussonnier Matthias
|
r20142 | "\nSaving disabled.\nSee Console for more details."); | ||
cm.setOption('readOnly','nocursor') | ||||
Thomas Kluyver
|
r19015 | that.save_enabled = false; | ||
Thomas Kluyver
|
r19013 | } | ||
Thomas Kluyver
|
r19015 | ); | ||
Thomas Kluyver
|
r19013 | }; | ||
Min RK
|
r19321 | |||
Editor.prototype._set_mode_for_model = function (model) { | ||||
/** Set the CodeMirror mode based on the file model */ | ||||
Scott Sanderson
|
r19787 | |||
Min RK
|
r19321 | // Find and load the highlighting mode, | ||
// first by mime-type, then by file extension | ||||
Scott Sanderson
|
r19788 | |||
Scott Sanderson
|
r19787 | var modeinfo; | ||
Scott Sanderson
|
r19788 | // mimetype is unset on file rename | ||
Scott Sanderson
|
r19787 | if (model.mimetype) { | ||
modeinfo = CodeMirror.findModeByMIME(model.mimetype); | ||||
} | ||||
if (!modeinfo || modeinfo.mode === "null") { | ||||
Min RK
|
r19321 | // find by mime failed, use find by ext | ||
var ext_idx = model.name.lastIndexOf('.'); | ||||
Scott Sanderson
|
r19787 | |||
Min RK
|
r19321 | if (ext_idx > 0) { | ||
// CodeMirror.findModeByExtension wants extension without '.' | ||||
modeinfo = CodeMirror.findModeByExtension(model.name.slice(ext_idx + 1)); | ||||
} | ||||
} | ||||
if (modeinfo) { | ||||
this.set_codemirror_mode(modeinfo); | ||||
} | ||||
}; | ||||
Scott Sanderson
|
r19787 | |||
Min RK
|
r19319 | Editor.prototype.set_codemirror_mode = function (modeinfo) { | ||
/** set the codemirror mode from a modeinfo struct */ | ||||
var that = this; | ||||
utils.requireCodeMirrorMode(modeinfo, function () { | ||||
that.codemirror.setOption('mode', modeinfo.mode); | ||||
that.events.trigger("mode_changed.Editor", modeinfo); | ||||
}); | ||||
}; | ||||
Min RK
|
r19316 | Editor.prototype.get_filename = function () { | ||
return utils.url_path_split(this.file_path)[1]; | ||||
Min RK
|
r19319 | }; | ||
Thomas Kluyver
|
r19013 | |||
Min RK
|
r19316 | Editor.prototype.rename = function (new_name) { | ||
/** rename the file */ | ||||
var that = this; | ||||
var parent = utils.url_path_split(this.file_path)[0]; | ||||
var new_path = utils.url_path_join(parent, new_name); | ||||
return this.contents.rename(this.file_path, new_path).then( | ||||
Min RK
|
r19321 | function (model) { | ||
that.file_path = model.path; | ||||
that.events.trigger('file_renamed.Editor', model); | ||||
that._set_mode_for_model(model); | ||||
Bussonnier Matthias
|
r20208 | that._clean_state(); | ||
Min RK
|
r19316 | } | ||
); | ||||
}; | ||||
Editor.prototype.save = function () { | ||||
Min RK
|
r19303 | /** save the file */ | ||
Thomas Kluyver
|
r19015 | if (!this.save_enabled) { | ||
console.log("Not saving, save disabled"); | ||||
return; | ||||
} | ||||
Thomas Kluyver
|
r19013 | var model = { | ||
Thomas Kluyver
|
r19015 | path: this.file_path, | ||
Thomas Kluyver
|
r19013 | type: 'file', | ||
format: 'text', | ||||
content: this.codemirror.getValue(), | ||||
}; | ||||
var that = this; | ||||
Min RK
|
r19304 | // record change generation for isClean | ||
this.generation = this.codemirror.changeGeneration(); | ||||
Bussonnier Matthias
|
r20149 | that.events.trigger("file_saving.Editor"); | ||
Min RK
|
r19316 | return this.contents.save(this.file_path, model).then(function(data) { | ||
that.events.trigger("file_saved.Editor", data); | ||||
Bussonnier Matthias
|
r20208 | that._clean_state(); | ||
Thomas Kluyver
|
r19013 | }); | ||
}; | ||||
Bussonnier Matthias
|
r20208 | |||
Editor.prototype._clean_state = function(){ | ||||
var clean = this.codemirror.isClean(this.generation); | ||||
if (clean === this.clean){ | ||||
return | ||||
} else { | ||||
this.clean = clean; | ||||
} | ||||
if(clean){ | ||||
this.events.trigger("save_status_clean.Editor"); | ||||
this.clean_sel.attr('class','dirty-indicator-clean').attr('title','No changes to save'); | ||||
} else { | ||||
this.events.trigger("save_status_dirty.Editor"); | ||||
this.clean_sel.attr('class','dirty-indicator-dirty').attr('title','Unsaved changes'); | ||||
} | ||||
}; | ||||
Min RK
|
r19303 | Editor.prototype._set_codemirror_options = function (options) { | ||
// update codemirror options from a dict | ||||
Min RK
|
r19334 | var codemirror = this.codemirror; | ||
$.map(options, function (value, opt) { | ||||
Min RK
|
r19311 | if (value === null) { | ||
value = CodeMirror.defaults[opt]; | ||||
} | ||||
Min RK
|
r19334 | codemirror.setOption(opt, value); | ||
}); | ||||
Bussonnier Matthias
|
r20208 | var that = this; | ||
Min RK
|
r19303 | }; | ||
Editor.prototype.update_codemirror_options = function (options) { | ||||
/** update codemirror options locally and save changes in config */ | ||||
var that = this; | ||||
this._set_codemirror_options(options); | ||||
return this.config.update({ | ||||
Editor: { | ||||
codemirror_options: options | ||||
} | ||||
}).then( | ||||
that.events.trigger('config_changed.Editor', {config: that.config}) | ||||
); | ||||
}; | ||||
Thomas Kluyver
|
r19013 | |||
return {Editor: Editor}; | ||||
}); | ||||