lib/ace/mode/folding/vbscript.js (283 lines of code) (raw):
/* ***** BEGIN LICENSE BLOCK *****
* Distributed under the BSD license:
*
* Copyright (c) 2012, Ajax.org B.V.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of Ajax.org B.V. nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL AJAX.ORG B.V. BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* ***** END LICENSE BLOCK ***** */
define(function(require, exports, module) {
"use strict";
var oop = require("../../lib/oop");
var BaseFoldMode = require("./fold_mode").FoldMode;
var Range = require("../../range").Range;
var TokenIterator = require("../../token_iterator").TokenIterator;
var FoldMode = exports.FoldMode = function() {};
oop.inherits(FoldMode, BaseFoldMode);
(function() {
this.indentKeywords = {
"class": 1,
"function": 1,
"sub": 1,
"if": 1,
"select": 1,
"do": 1,
"for": 1,
"while": 1,
"with": 1,
"property": 1,
"else": 1,
"elseif": 1,
"end": -1,
"loop": -1,
"next": -1,
"wend": -1
};
this.foldingStartMarker = /(?:\s|^)(class|function|sub|if|select|do|for|while|with|property|else|elseif)\b/i;
this.foldingStopMarker = /\b(end|loop|next|wend)\b/i;
this.getFoldWidgetRange = function (session, foldStyle, row) {
var line = session.getLine(row);
var isStart = this.foldingStartMarker.test(line);
var isEnd = this.foldingStopMarker.test(line);
if (isStart || isEnd) {
var match = (isEnd) ? this.foldingStopMarker.exec(line) : this.foldingStartMarker.exec(line);
var keyword = match && match[1].toLowerCase();
if (keyword) {
var type = session.getTokenAt(row, match.index + 2).type;
if (type === "keyword.control.asp" || type === "storage.type.function.asp")
return this.vbsBlock(session, row, match.index + 2);
}
}
};
// must return "" if there's no fold, to enable caching
this.getFoldWidget = function(session, foldStyle, row) {
var line = session.getLine(row);
var isStart = this.foldingStartMarker.test(line);
var isEnd = this.foldingStopMarker.test(line);
if (isStart && !isEnd) {
var match = this.foldingStartMarker.exec(line);
var keyword = match && match[1].toLowerCase();
if (keyword) {
var type = session.getTokenAt(row, match.index + 2).type;
if (type == "keyword.control.asp" || type == "storage.type.function.asp") {
if (keyword == "if" && !/then\s*('|$)/i.test(line))
return "";
return "start";
}
}
}
return "";
};
this.vbsBlock = function(session, row, column, tokenRange) {
var stream = new TokenIterator(session, row, column);
var endOpenings = {
"class": 1,
"function": 1,
"sub": 1,
"if": 1,
"select": 1,
"with": 1,
"property": 1,
"else": 1,
"elseif": 1
};
var token = stream.getCurrentToken();
if (!token || (token.type != "keyword.control.asp" && token.type != "storage.type.function.asp"))
return;
var startTokenValue = token.value.toLowerCase();
var val = token.value.toLowerCase();
var stack = [val];
var dir = this.indentKeywords[val];
if (!dir)
return;
var firstRange = stream.getCurrentTokenRange();
switch (val) {
case "property":
case "sub":
case "function":
case "if":
case "select":
case "do":
case "for":
case "class":
case "while":
case "with":
var line = session.getLine(row);
var singleLineCondition = /^\s*If\s+.*\s+Then(?!')\s+(?!')\S/i.test(line);
if (singleLineCondition)
return;
var checkToken = new RegExp("(?:^|\\s)" + val, "i");
var endTest = /^\s*End\s(If|Sub|Select|Function|Class|With|Property)\s*/i.test(line);
if (!checkToken.test(line) && !endTest) {
return;
}
if (endTest) {
var tokenRange = stream.getCurrentTokenRange();
stream.step = stream.stepBackward;
stream.step();
stream.step();
token = stream.getCurrentToken();
if (token) {
val = token.value.toLowerCase();
if (val == "end") {
firstRange = stream.getCurrentTokenRange();
firstRange = new Range(firstRange.start.row, firstRange.start.column, tokenRange.start.row, tokenRange.end.column);
}
}
dir = -1;
}
break;
case "end":
var tokenPos = stream.getCurrentTokenPosition();
firstRange = stream.getCurrentTokenRange();
stream.step = stream.stepForward;
stream.step();
stream.step();
token = stream.getCurrentToken();
if (token) {
val = token.value.toLowerCase();
if (val in endOpenings) {
startTokenValue = val;
var nextTokenPos = stream.getCurrentTokenPosition();
var endColumn = nextTokenPos.column + val.length;
firstRange = new Range(tokenPos.row, tokenPos.column, nextTokenPos.row, endColumn);
}
}
stream.step = stream.stepBackward;
stream.step();
stream.step();
break;
}
var startColumn = dir === -1 ? session.getLine(row - 1).length : session.getLine(row).length;
var startRow = row;
var ranges = [];
ranges.push(firstRange);
stream.step = dir === -1 ? stream.stepBackward : stream.stepForward;
while(token = stream.step()) {
var outputRange = null;
var ignore = false;
if (token.type != "keyword.control.asp" && token.type != "storage.type.function.asp")
continue;
val = token.value.toLowerCase();
var level = dir * this.indentKeywords[val];
switch (val) {
case "property":
case "sub":
case "function":
case "if":
case "select":
case "do":
case "for":
case "class":
case "while":
case "with":
var line = session.getLine(stream.getCurrentTokenRow());
var singleLineCondition = /^\s*If\s+.*\s+Then(?!')\s+(?!')\S/i.test(line);
if (singleLineCondition) {
level = 0;
ignore = true;
}
var checkToken = new RegExp("^\\s* end\\s+" + val, "i");
if (checkToken.test(line)) {
level = 0;
ignore = true;
}
break;
case "elseif":
case "else":
level = 0;
if (startTokenValue != "elseif") {
ignore = true;
}
break;
}
if (level > 0) {
stack.unshift(val);
} else if (level <= 0 && ignore === false) {
stack.shift();
if (!stack.length) {
switch (val) {
case "end":
var tokenPos = stream.getCurrentTokenPosition();
outputRange = stream.getCurrentTokenRange();
stream.step();
stream.step();
token = stream.getCurrentToken();
if (token) {
val = token.value.toLowerCase();
if (val in endOpenings) {
if ((startTokenValue == "else" || startTokenValue == "elseif")) {
if (val !== "if") {
ranges.shift();
}
} else {
if (val != startTokenValue)
ranges.shift();
}
var nextTokenPos = stream.getCurrentTokenPosition();
var endColumn = nextTokenPos.column + val.length;
outputRange = new Range(tokenPos.row, tokenPos.column, nextTokenPos.row, endColumn);
} else {
ranges.shift();
}
} else {
ranges.shift();
}
stream.step = stream.stepBackward;
stream.step();
stream.step();
token = stream.getCurrentToken();
val = token.value.toLowerCase();
break;
case "select":
case "sub":
case "if":
case "function":
case "class":
case "with":
case "property":
if (val != startTokenValue)
ranges.shift();
break;
case "do":
if (startTokenValue != "loop")
ranges.shift();
break;
case "loop":
if (startTokenValue != "do")
ranges.shift();
break;
case "for":
if (startTokenValue != "next")
ranges.shift();
break;
case "next":
if (startTokenValue != "for")
ranges.shift();
break;
case "while":
if (startTokenValue != "wend")
ranges.shift();
break;
case "wend":
if (startTokenValue != "while")
ranges.shift();
break;
}
break;
}
if (level === 0){
stack.unshift(val);
}
}
}
if (!token)
return null;
if (tokenRange) {
if (!outputRange) {
ranges.push(stream.getCurrentTokenRange());
} else {
ranges.push(outputRange);
}
return ranges;
}
var row = stream.getCurrentTokenRow();
if (dir === -1) {
var endColumn = session.getLine(row).length;
return new Range(row, endColumn, startRow - 1, startColumn);
} else
return new Range(startRow, startColumn, row - 1, session.getLine(row - 1).length);
};
}).call(FoldMode.prototype);
});