Ответ 1
Вот начало решения:
$(function() {
var editor = ace.edit("editor1")
, session = editor.getSession()
, Range = require("ace/range").Range
, range = new Range(1, 4, 1, 10)
, markerId = session.addMarker(range, "readonly-highlight");
session.setMode("ace/mode/javascript");
editor.keyBinding.addKeyboardHandler({
handleKeyboard : function(data, hash, keyString, keyCode, event) {
if (hash === -1 || (keyCode <= 40 && keyCode >= 37)) return false;
if (intersects(range)) {
return {command:"null", passEvent:false};
}
}
});
before(editor, 'onPaste', preventReadonly);
before(editor, 'onCut', preventReadonly);
range.start = session.doc.createAnchor(range.start);
range.end = session.doc.createAnchor(range.end);
range.end.$insertRight = true;
function before(obj, method, wrapper) {
var orig = obj[method];
obj[method] = function() {
var args = Array.prototype.slice.call(arguments);
return wrapper.call(this, function(){
return orig.apply(obj, args);
}, args);
}
return obj[method];
}
function intersects(range) {
return editor.getSelectionRange().intersects(range);
}
function preventReadonly(next, args) {
if (intersects(range)) return;
next();
}
});
видеть, как он работает в этой скрипте: http://jsfiddle.net/bzwheeler/btsxgena/
Основными работами являются:
- создать начальные и конечные анкеры, которые отслеживают местоположение части "только для чтения", когда изменяется документ вокруг него.
- создать диапазон для инкапсуляции якорей
- добавьте пользовательский манипулятор, чтобы проверить, повлияет ли текущее надвигающееся нажатие на диапазон readonly и отменит его, если это так.
- добавить пользовательские дескрипторы палитры/вырезания для защиты от контекстного меню и действий вырезать/вставить меню браузера.