
Previously, we were at an ACE editor published between 1.1.8 and 1.1.9. This caused multiple issues and was especially a problem for the upcoming pair programming feature. Further, updating ace is a long-time priority, see https://github.com/openHPI/codeocean/issues/250. Now, we are not yet updating to the latest version, but rather to the next minor version. This already contains breaking changes, and we are currently interested to keep the number of changes as low as possible. Further updating ACE might be still a future task. The new ACE version 1.2.0 is taken from this tag: https://github.com/ajaxorg/ace-builds/releases/tag/v1.2.0. We are using the src build (not minified, not in the noconflict version), since the same was used before as well. Further, we need to change our migration for storing editor events. Since the table is not yet used (in production), we also update the enum.
668 lines
25 KiB
JavaScript
668 lines
25 KiB
JavaScript
define("ace/mode/logiql_highlight_rules",["require","exports","module","ace/lib/oop","ace/mode/text_highlight_rules"], function(require, exports, module) {
|
|
"use strict";
|
|
|
|
var oop = require("../lib/oop");
|
|
var TextHighlightRules = require("./text_highlight_rules").TextHighlightRules;
|
|
|
|
var LogiQLHighlightRules = function() {
|
|
|
|
this.$rules = { start:
|
|
[ { token: 'comment.block',
|
|
regex: '/\\*',
|
|
push:
|
|
[ { token: 'comment.block', regex: '\\*/', next: 'pop' },
|
|
{ defaultToken: 'comment.block' } ],
|
|
},
|
|
{ token: 'comment.single',
|
|
regex: '//.*',
|
|
},
|
|
{ token: 'constant.numeric',
|
|
regex: '\\d+(?:\\.\\d+)?(?:[eE][+-]?\\d+)?[fd]?',
|
|
},
|
|
{ token: 'string',
|
|
regex: '"',
|
|
push:
|
|
[ { token: 'string', regex: '"', next: 'pop' },
|
|
{ defaultToken: 'string' } ],
|
|
},
|
|
{ token: 'constant.language',
|
|
regex: '\\b(true|false)\\b',
|
|
},
|
|
{ token: 'entity.name.type.logicblox',
|
|
regex: '`[a-zA-Z_:]+(\\d|\\a)*\\b',
|
|
},
|
|
{ token: 'keyword.start', regex: '->', comment: 'Constraint' },
|
|
{ token: 'keyword.start', regex: '-->', comment: 'Level 1 Constraint'},
|
|
{ token: 'keyword.start', regex: '<-', comment: 'Rule' },
|
|
{ token: 'keyword.start', regex: '<--', comment: 'Level 1 Rule' },
|
|
{ token: 'keyword.end', regex: '\\.', comment: 'Terminator' },
|
|
{ token: 'keyword.other', regex: '!', comment: 'Negation' },
|
|
{ token: 'keyword.other', regex: ',', comment: 'Conjunction' },
|
|
{ token: 'keyword.other', regex: ';', comment: 'Disjunction' },
|
|
{ token: 'keyword.operator', regex: '<=|>=|!=|<|>', comment: 'Equality'},
|
|
{ token: 'keyword.other', regex: '@', comment: 'Equality' },
|
|
{ token: 'keyword.operator', regex: '\\+|-|\\*|/', comment: 'Arithmetic operations'},
|
|
{ token: 'keyword', regex: '::', comment: 'Colon colon' },
|
|
{ token: 'support.function',
|
|
regex: '\\b(agg\\s*<<)',
|
|
push:
|
|
[ { include: '$self' },
|
|
{ token: 'support.function',
|
|
regex: '>>',
|
|
next: 'pop' } ],
|
|
},
|
|
{ token: 'storage.modifier',
|
|
regex: '\\b(lang:[\\w:]*)',
|
|
},
|
|
{ token: [ 'storage.type', 'text' ],
|
|
regex: '(export|sealed|clauses|block|alias|alias_all)(\\s*\\()(?=`)',
|
|
},
|
|
{ token: 'entity.name',
|
|
regex: '[a-zA-Z_][a-zA-Z_0-9:]*(@prev|@init|@final)?(?=(\\(|\\[))',
|
|
},
|
|
{ token: 'variable.parameter',
|
|
regex: '([a-zA-Z][a-zA-Z_0-9]*|_)\\s*(?=(,|\\.|<-|->|\\)|\\]|=))',
|
|
} ] }
|
|
|
|
this.normalizeRules();
|
|
};
|
|
|
|
oop.inherits(LogiQLHighlightRules, TextHighlightRules);
|
|
|
|
exports.LogiQLHighlightRules = LogiQLHighlightRules;
|
|
});
|
|
|
|
define("ace/mode/folding/coffee",["require","exports","module","ace/lib/oop","ace/mode/folding/fold_mode","ace/range"], function(require, exports, module) {
|
|
"use strict";
|
|
|
|
var oop = require("../../lib/oop");
|
|
var BaseFoldMode = require("./fold_mode").FoldMode;
|
|
var Range = require("../../range").Range;
|
|
|
|
var FoldMode = exports.FoldMode = function() {};
|
|
oop.inherits(FoldMode, BaseFoldMode);
|
|
|
|
(function() {
|
|
|
|
this.getFoldWidgetRange = function(session, foldStyle, row) {
|
|
var range = this.indentationBlock(session, row);
|
|
if (range)
|
|
return range;
|
|
|
|
var re = /\S/;
|
|
var line = session.getLine(row);
|
|
var startLevel = line.search(re);
|
|
if (startLevel == -1 || line[startLevel] != "#")
|
|
return;
|
|
|
|
var startColumn = line.length;
|
|
var maxRow = session.getLength();
|
|
var startRow = row;
|
|
var endRow = row;
|
|
|
|
while (++row < maxRow) {
|
|
line = session.getLine(row);
|
|
var level = line.search(re);
|
|
|
|
if (level == -1)
|
|
continue;
|
|
|
|
if (line[level] != "#")
|
|
break;
|
|
|
|
endRow = row;
|
|
}
|
|
|
|
if (endRow > startRow) {
|
|
var endColumn = session.getLine(endRow).length;
|
|
return new Range(startRow, startColumn, endRow, endColumn);
|
|
}
|
|
};
|
|
this.getFoldWidget = function(session, foldStyle, row) {
|
|
var line = session.getLine(row);
|
|
var indent = line.search(/\S/);
|
|
var next = session.getLine(row + 1);
|
|
var prev = session.getLine(row - 1);
|
|
var prevIndent = prev.search(/\S/);
|
|
var nextIndent = next.search(/\S/);
|
|
|
|
if (indent == -1) {
|
|
session.foldWidgets[row - 1] = prevIndent!= -1 && prevIndent < nextIndent ? "start" : "";
|
|
return "";
|
|
}
|
|
if (prevIndent == -1) {
|
|
if (indent == nextIndent && line[indent] == "#" && next[indent] == "#") {
|
|
session.foldWidgets[row - 1] = "";
|
|
session.foldWidgets[row + 1] = "";
|
|
return "start";
|
|
}
|
|
} else if (prevIndent == indent && line[indent] == "#" && prev[indent] == "#") {
|
|
if (session.getLine(row - 2).search(/\S/) == -1) {
|
|
session.foldWidgets[row - 1] = "start";
|
|
session.foldWidgets[row + 1] = "";
|
|
return "";
|
|
}
|
|
}
|
|
|
|
if (prevIndent!= -1 && prevIndent < indent)
|
|
session.foldWidgets[row - 1] = "start";
|
|
else
|
|
session.foldWidgets[row - 1] = "";
|
|
|
|
if (indent < nextIndent)
|
|
return "start";
|
|
else
|
|
return "";
|
|
};
|
|
|
|
}).call(FoldMode.prototype);
|
|
|
|
});
|
|
|
|
define("ace/mode/behaviour/cstyle",["require","exports","module","ace/lib/oop","ace/mode/behaviour","ace/token_iterator","ace/lib/lang"], function(require, exports, module) {
|
|
"use strict";
|
|
|
|
var oop = require("../../lib/oop");
|
|
var Behaviour = require("../behaviour").Behaviour;
|
|
var TokenIterator = require("../../token_iterator").TokenIterator;
|
|
var lang = require("../../lib/lang");
|
|
|
|
var SAFE_INSERT_IN_TOKENS =
|
|
["text", "paren.rparen", "punctuation.operator"];
|
|
var SAFE_INSERT_BEFORE_TOKENS =
|
|
["text", "paren.rparen", "punctuation.operator", "comment"];
|
|
|
|
var context;
|
|
var contextCache = {};
|
|
var initContext = function(editor) {
|
|
var id = -1;
|
|
if (editor.multiSelect) {
|
|
id = editor.selection.index;
|
|
if (contextCache.rangeCount != editor.multiSelect.rangeCount)
|
|
contextCache = {rangeCount: editor.multiSelect.rangeCount};
|
|
}
|
|
if (contextCache[id])
|
|
return context = contextCache[id];
|
|
context = contextCache[id] = {
|
|
autoInsertedBrackets: 0,
|
|
autoInsertedRow: -1,
|
|
autoInsertedLineEnd: "",
|
|
maybeInsertedBrackets: 0,
|
|
maybeInsertedRow: -1,
|
|
maybeInsertedLineStart: "",
|
|
maybeInsertedLineEnd: ""
|
|
};
|
|
};
|
|
|
|
var getWrapped = function(selection, selected, opening, closing) {
|
|
var rowDiff = selection.end.row - selection.start.row;
|
|
return {
|
|
text: opening + selected + closing,
|
|
selection: [
|
|
0,
|
|
selection.start.column + 1,
|
|
rowDiff,
|
|
selection.end.column + (rowDiff ? 0 : 1)
|
|
]
|
|
};
|
|
};
|
|
|
|
var CstyleBehaviour = function() {
|
|
this.add("braces", "insertion", function(state, action, editor, session, text) {
|
|
var cursor = editor.getCursorPosition();
|
|
var line = session.doc.getLine(cursor.row);
|
|
if (text == '{') {
|
|
initContext(editor);
|
|
var selection = editor.getSelectionRange();
|
|
var selected = session.doc.getTextRange(selection);
|
|
if (selected !== "" && selected !== "{" && editor.getWrapBehavioursEnabled()) {
|
|
return getWrapped(selection, selected, '{', '}');
|
|
} else if (CstyleBehaviour.isSaneInsertion(editor, session)) {
|
|
if (/[\]\}\)]/.test(line[cursor.column]) || editor.inMultiSelectMode) {
|
|
CstyleBehaviour.recordAutoInsert(editor, session, "}");
|
|
return {
|
|
text: '{}',
|
|
selection: [1, 1]
|
|
};
|
|
} else {
|
|
CstyleBehaviour.recordMaybeInsert(editor, session, "{");
|
|
return {
|
|
text: '{',
|
|
selection: [1, 1]
|
|
};
|
|
}
|
|
}
|
|
} else if (text == '}') {
|
|
initContext(editor);
|
|
var rightChar = line.substring(cursor.column, cursor.column + 1);
|
|
if (rightChar == '}') {
|
|
var matching = session.$findOpeningBracket('}', {column: cursor.column + 1, row: cursor.row});
|
|
if (matching !== null && CstyleBehaviour.isAutoInsertedClosing(cursor, line, text)) {
|
|
CstyleBehaviour.popAutoInsertedClosing();
|
|
return {
|
|
text: '',
|
|
selection: [1, 1]
|
|
};
|
|
}
|
|
}
|
|
} else if (text == "\n" || text == "\r\n") {
|
|
initContext(editor);
|
|
var closing = "";
|
|
if (CstyleBehaviour.isMaybeInsertedClosing(cursor, line)) {
|
|
closing = lang.stringRepeat("}", context.maybeInsertedBrackets);
|
|
CstyleBehaviour.clearMaybeInsertedClosing();
|
|
}
|
|
var rightChar = line.substring(cursor.column, cursor.column + 1);
|
|
if (rightChar === '}') {
|
|
var openBracePos = session.findMatchingBracket({row: cursor.row, column: cursor.column+1}, '}');
|
|
if (!openBracePos)
|
|
return null;
|
|
var next_indent = this.$getIndent(session.getLine(openBracePos.row));
|
|
} else if (closing) {
|
|
var next_indent = this.$getIndent(line);
|
|
} else {
|
|
CstyleBehaviour.clearMaybeInsertedClosing();
|
|
return;
|
|
}
|
|
var indent = next_indent + session.getTabString();
|
|
|
|
return {
|
|
text: '\n' + indent + '\n' + next_indent + closing,
|
|
selection: [1, indent.length, 1, indent.length]
|
|
};
|
|
} else {
|
|
CstyleBehaviour.clearMaybeInsertedClosing();
|
|
}
|
|
});
|
|
|
|
this.add("braces", "deletion", function(state, action, editor, session, range) {
|
|
var selected = session.doc.getTextRange(range);
|
|
if (!range.isMultiLine() && selected == '{') {
|
|
initContext(editor);
|
|
var line = session.doc.getLine(range.start.row);
|
|
var rightChar = line.substring(range.end.column, range.end.column + 1);
|
|
if (rightChar == '}') {
|
|
range.end.column++;
|
|
return range;
|
|
} else {
|
|
context.maybeInsertedBrackets--;
|
|
}
|
|
}
|
|
});
|
|
|
|
this.add("parens", "insertion", function(state, action, editor, session, text) {
|
|
if (text == '(') {
|
|
initContext(editor);
|
|
var selection = editor.getSelectionRange();
|
|
var selected = session.doc.getTextRange(selection);
|
|
if (selected !== "" && editor.getWrapBehavioursEnabled()) {
|
|
return getWrapped(selection, selected, '(', ')');
|
|
} else if (CstyleBehaviour.isSaneInsertion(editor, session)) {
|
|
CstyleBehaviour.recordAutoInsert(editor, session, ")");
|
|
return {
|
|
text: '()',
|
|
selection: [1, 1]
|
|
};
|
|
}
|
|
} else if (text == ')') {
|
|
initContext(editor);
|
|
var cursor = editor.getCursorPosition();
|
|
var line = session.doc.getLine(cursor.row);
|
|
var rightChar = line.substring(cursor.column, cursor.column + 1);
|
|
if (rightChar == ')') {
|
|
var matching = session.$findOpeningBracket(')', {column: cursor.column + 1, row: cursor.row});
|
|
if (matching !== null && CstyleBehaviour.isAutoInsertedClosing(cursor, line, text)) {
|
|
CstyleBehaviour.popAutoInsertedClosing();
|
|
return {
|
|
text: '',
|
|
selection: [1, 1]
|
|
};
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
this.add("parens", "deletion", function(state, action, editor, session, range) {
|
|
var selected = session.doc.getTextRange(range);
|
|
if (!range.isMultiLine() && selected == '(') {
|
|
initContext(editor);
|
|
var line = session.doc.getLine(range.start.row);
|
|
var rightChar = line.substring(range.start.column + 1, range.start.column + 2);
|
|
if (rightChar == ')') {
|
|
range.end.column++;
|
|
return range;
|
|
}
|
|
}
|
|
});
|
|
|
|
this.add("brackets", "insertion", function(state, action, editor, session, text) {
|
|
if (text == '[') {
|
|
initContext(editor);
|
|
var selection = editor.getSelectionRange();
|
|
var selected = session.doc.getTextRange(selection);
|
|
if (selected !== "" && editor.getWrapBehavioursEnabled()) {
|
|
return getWrapped(selection, selected, '[', ']');
|
|
} else if (CstyleBehaviour.isSaneInsertion(editor, session)) {
|
|
CstyleBehaviour.recordAutoInsert(editor, session, "]");
|
|
return {
|
|
text: '[]',
|
|
selection: [1, 1]
|
|
};
|
|
}
|
|
} else if (text == ']') {
|
|
initContext(editor);
|
|
var cursor = editor.getCursorPosition();
|
|
var line = session.doc.getLine(cursor.row);
|
|
var rightChar = line.substring(cursor.column, cursor.column + 1);
|
|
if (rightChar == ']') {
|
|
var matching = session.$findOpeningBracket(']', {column: cursor.column + 1, row: cursor.row});
|
|
if (matching !== null && CstyleBehaviour.isAutoInsertedClosing(cursor, line, text)) {
|
|
CstyleBehaviour.popAutoInsertedClosing();
|
|
return {
|
|
text: '',
|
|
selection: [1, 1]
|
|
};
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
this.add("brackets", "deletion", function(state, action, editor, session, range) {
|
|
var selected = session.doc.getTextRange(range);
|
|
if (!range.isMultiLine() && selected == '[') {
|
|
initContext(editor);
|
|
var line = session.doc.getLine(range.start.row);
|
|
var rightChar = line.substring(range.start.column + 1, range.start.column + 2);
|
|
if (rightChar == ']') {
|
|
range.end.column++;
|
|
return range;
|
|
}
|
|
}
|
|
});
|
|
|
|
this.add("string_dquotes", "insertion", function(state, action, editor, session, text) {
|
|
if (text == '"' || text == "'") {
|
|
initContext(editor);
|
|
var quote = text;
|
|
var selection = editor.getSelectionRange();
|
|
var selected = session.doc.getTextRange(selection);
|
|
if (selected !== "" && selected !== "'" && selected != '"' && editor.getWrapBehavioursEnabled()) {
|
|
return getWrapped(selection, selected, quote, quote);
|
|
} else if (!selected) {
|
|
var cursor = editor.getCursorPosition();
|
|
var line = session.doc.getLine(cursor.row);
|
|
var leftChar = line.substring(cursor.column-1, cursor.column);
|
|
var rightChar = line.substring(cursor.column, cursor.column + 1);
|
|
|
|
var token = session.getTokenAt(cursor.row, cursor.column);
|
|
var rightToken = session.getTokenAt(cursor.row, cursor.column + 1);
|
|
if (leftChar == "\\" && token && /escape/.test(token.type))
|
|
return null;
|
|
|
|
var stringBefore = token && /string|escape/.test(token.type);
|
|
var stringAfter = !rightToken || /string|escape/.test(rightToken.type);
|
|
|
|
var pair;
|
|
if (rightChar == quote) {
|
|
pair = stringBefore !== stringAfter;
|
|
} else {
|
|
if (stringBefore && !stringAfter)
|
|
return null; // wrap string with different quote
|
|
if (stringBefore && stringAfter)
|
|
return null; // do not pair quotes inside strings
|
|
var wordRe = session.$mode.tokenRe;
|
|
wordRe.lastIndex = 0;
|
|
var isWordBefore = wordRe.test(leftChar);
|
|
wordRe.lastIndex = 0;
|
|
var isWordAfter = wordRe.test(leftChar);
|
|
if (isWordBefore || isWordAfter)
|
|
return null; // before or after alphanumeric
|
|
if (rightChar && !/[\s;,.})\]\\]/.test(rightChar))
|
|
return null; // there is rightChar and it isn't closing
|
|
pair = true;
|
|
}
|
|
return {
|
|
text: pair ? quote + quote : "",
|
|
selection: [1,1]
|
|
};
|
|
}
|
|
}
|
|
});
|
|
|
|
this.add("string_dquotes", "deletion", function(state, action, editor, session, range) {
|
|
var selected = session.doc.getTextRange(range);
|
|
if (!range.isMultiLine() && (selected == '"' || selected == "'")) {
|
|
initContext(editor);
|
|
var line = session.doc.getLine(range.start.row);
|
|
var rightChar = line.substring(range.start.column + 1, range.start.column + 2);
|
|
if (rightChar == selected) {
|
|
range.end.column++;
|
|
return range;
|
|
}
|
|
}
|
|
});
|
|
|
|
};
|
|
|
|
|
|
CstyleBehaviour.isSaneInsertion = function(editor, session) {
|
|
var cursor = editor.getCursorPosition();
|
|
var iterator = new TokenIterator(session, cursor.row, cursor.column);
|
|
if (!this.$matchTokenType(iterator.getCurrentToken() || "text", SAFE_INSERT_IN_TOKENS)) {
|
|
var iterator2 = new TokenIterator(session, cursor.row, cursor.column + 1);
|
|
if (!this.$matchTokenType(iterator2.getCurrentToken() || "text", SAFE_INSERT_IN_TOKENS))
|
|
return false;
|
|
}
|
|
iterator.stepForward();
|
|
return iterator.getCurrentTokenRow() !== cursor.row ||
|
|
this.$matchTokenType(iterator.getCurrentToken() || "text", SAFE_INSERT_BEFORE_TOKENS);
|
|
};
|
|
|
|
CstyleBehaviour.$matchTokenType = function(token, types) {
|
|
return types.indexOf(token.type || token) > -1;
|
|
};
|
|
|
|
CstyleBehaviour.recordAutoInsert = function(editor, session, bracket) {
|
|
var cursor = editor.getCursorPosition();
|
|
var line = session.doc.getLine(cursor.row);
|
|
if (!this.isAutoInsertedClosing(cursor, line, context.autoInsertedLineEnd[0]))
|
|
context.autoInsertedBrackets = 0;
|
|
context.autoInsertedRow = cursor.row;
|
|
context.autoInsertedLineEnd = bracket + line.substr(cursor.column);
|
|
context.autoInsertedBrackets++;
|
|
};
|
|
|
|
CstyleBehaviour.recordMaybeInsert = function(editor, session, bracket) {
|
|
var cursor = editor.getCursorPosition();
|
|
var line = session.doc.getLine(cursor.row);
|
|
if (!this.isMaybeInsertedClosing(cursor, line))
|
|
context.maybeInsertedBrackets = 0;
|
|
context.maybeInsertedRow = cursor.row;
|
|
context.maybeInsertedLineStart = line.substr(0, cursor.column) + bracket;
|
|
context.maybeInsertedLineEnd = line.substr(cursor.column);
|
|
context.maybeInsertedBrackets++;
|
|
};
|
|
|
|
CstyleBehaviour.isAutoInsertedClosing = function(cursor, line, bracket) {
|
|
return context.autoInsertedBrackets > 0 &&
|
|
cursor.row === context.autoInsertedRow &&
|
|
bracket === context.autoInsertedLineEnd[0] &&
|
|
line.substr(cursor.column) === context.autoInsertedLineEnd;
|
|
};
|
|
|
|
CstyleBehaviour.isMaybeInsertedClosing = function(cursor, line) {
|
|
return context.maybeInsertedBrackets > 0 &&
|
|
cursor.row === context.maybeInsertedRow &&
|
|
line.substr(cursor.column) === context.maybeInsertedLineEnd &&
|
|
line.substr(0, cursor.column) == context.maybeInsertedLineStart;
|
|
};
|
|
|
|
CstyleBehaviour.popAutoInsertedClosing = function() {
|
|
context.autoInsertedLineEnd = context.autoInsertedLineEnd.substr(1);
|
|
context.autoInsertedBrackets--;
|
|
};
|
|
|
|
CstyleBehaviour.clearMaybeInsertedClosing = function() {
|
|
if (context) {
|
|
context.maybeInsertedBrackets = 0;
|
|
context.maybeInsertedRow = -1;
|
|
}
|
|
};
|
|
|
|
|
|
|
|
oop.inherits(CstyleBehaviour, Behaviour);
|
|
|
|
exports.CstyleBehaviour = CstyleBehaviour;
|
|
});
|
|
|
|
define("ace/mode/matching_brace_outdent",["require","exports","module","ace/range"], function(require, exports, module) {
|
|
"use strict";
|
|
|
|
var Range = require("../range").Range;
|
|
|
|
var MatchingBraceOutdent = function() {};
|
|
|
|
(function() {
|
|
|
|
this.checkOutdent = function(line, input) {
|
|
if (! /^\s+$/.test(line))
|
|
return false;
|
|
|
|
return /^\s*\}/.test(input);
|
|
};
|
|
|
|
this.autoOutdent = function(doc, row) {
|
|
var line = doc.getLine(row);
|
|
var match = line.match(/^(\s*\})/);
|
|
|
|
if (!match) return 0;
|
|
|
|
var column = match[1].length;
|
|
var openBracePos = doc.findMatchingBracket({row: row, column: column});
|
|
|
|
if (!openBracePos || openBracePos.row == row) return 0;
|
|
|
|
var indent = this.$getIndent(doc.getLine(openBracePos.row));
|
|
doc.replace(new Range(row, 0, row, column-1), indent);
|
|
};
|
|
|
|
this.$getIndent = function(line) {
|
|
return line.match(/^\s*/)[0];
|
|
};
|
|
|
|
}).call(MatchingBraceOutdent.prototype);
|
|
|
|
exports.MatchingBraceOutdent = MatchingBraceOutdent;
|
|
});
|
|
|
|
define("ace/mode/logiql",["require","exports","module","ace/lib/oop","ace/mode/text","ace/mode/logiql_highlight_rules","ace/mode/folding/coffee","ace/token_iterator","ace/range","ace/mode/behaviour/cstyle","ace/mode/matching_brace_outdent"], function(require, exports, module) {
|
|
"use strict";
|
|
|
|
var oop = require("../lib/oop");
|
|
var TextMode = require("./text").Mode;
|
|
var LogiQLHighlightRules = require("./logiql_highlight_rules").LogiQLHighlightRules;
|
|
var FoldMode = require("./folding/coffee").FoldMode;
|
|
var TokenIterator = require("../token_iterator").TokenIterator;
|
|
var Range = require("../range").Range;
|
|
var CstyleBehaviour = require("./behaviour/cstyle").CstyleBehaviour;
|
|
var MatchingBraceOutdent = require("./matching_brace_outdent").MatchingBraceOutdent;
|
|
|
|
var Mode = function() {
|
|
this.HighlightRules = LogiQLHighlightRules;
|
|
this.foldingRules = new FoldMode();
|
|
this.$outdent = new MatchingBraceOutdent();
|
|
this.$behaviour = new CstyleBehaviour();
|
|
};
|
|
oop.inherits(Mode, TextMode);
|
|
|
|
(function() {
|
|
this.lineCommentStart = "//";
|
|
this.blockComment = {start: "/*", end: "*/"};
|
|
|
|
this.getNextLineIndent = function(state, line, tab) {
|
|
var indent = this.$getIndent(line);
|
|
|
|
var tokenizedLine = this.getTokenizer().getLineTokens(line, state);
|
|
var tokens = tokenizedLine.tokens;
|
|
var endState = tokenizedLine.state;
|
|
if (/comment|string/.test(endState))
|
|
return indent;
|
|
if (tokens.length && tokens[tokens.length - 1].type == "comment.single")
|
|
return indent;
|
|
|
|
var match = line.match();
|
|
if (/(-->|<--|<-|->|{)\s*$/.test(line))
|
|
indent += tab;
|
|
return indent;
|
|
};
|
|
|
|
this.checkOutdent = function(state, line, input) {
|
|
if (this.$outdent.checkOutdent(line, input))
|
|
return true;
|
|
|
|
if (input !== "\n" && input !== "\r\n")
|
|
return false;
|
|
|
|
if (!/^\s+/.test(line))
|
|
return false;
|
|
|
|
return true;
|
|
};
|
|
|
|
this.autoOutdent = function(state, doc, row) {
|
|
if (this.$outdent.autoOutdent(doc, row))
|
|
return;
|
|
var prevLine = doc.getLine(row);
|
|
var match = prevLine.match(/^\s+/);
|
|
var column = prevLine.lastIndexOf(".") + 1;
|
|
if (!match || !row || !column) return 0;
|
|
|
|
var line = doc.getLine(row + 1);
|
|
var startRange = this.getMatching(doc, {row: row, column: column});
|
|
if (!startRange || startRange.start.row == row) return 0;
|
|
|
|
column = match[0].length;
|
|
var indent = this.$getIndent(doc.getLine(startRange.start.row));
|
|
doc.replace(new Range(row + 1, 0, row + 1, column), indent);
|
|
};
|
|
|
|
this.getMatching = function(session, row, column) {
|
|
if (row == undefined)
|
|
row = session.selection.lead
|
|
if (typeof row == "object") {
|
|
column = row.column;
|
|
row = row.row;
|
|
}
|
|
|
|
var startToken = session.getTokenAt(row, column);
|
|
var KW_START = "keyword.start", KW_END = "keyword.end";
|
|
var tok;
|
|
if (!startToken)
|
|
return;
|
|
if (startToken.type == KW_START) {
|
|
var it = new TokenIterator(session, row, column);
|
|
it.step = it.stepForward;
|
|
} else if (startToken.type == KW_END) {
|
|
var it = new TokenIterator(session, row, column);
|
|
it.step = it.stepBackward;
|
|
} else
|
|
return;
|
|
|
|
while (tok = it.step()) {
|
|
if (tok.type == KW_START || tok.type == KW_END)
|
|
break;
|
|
}
|
|
if (!tok || tok.type == startToken.type)
|
|
return;
|
|
|
|
var col = it.getCurrentTokenColumn();
|
|
var row = it.getCurrentTokenRow();
|
|
return new Range(row, col, row, col + tok.value.length);
|
|
};
|
|
this.$id = "ace/mode/logiql";
|
|
}).call(Mode.prototype);
|
|
|
|
exports.Mode = Mode;
|
|
});
|