Browse Source

Colorize autocompletion results according to token type.

cl-refactor
yann300 10 years ago
parent
commit
c86860df43
  1. 60
      mix/qml/html/cm/anyword-hint.js
  2. 24
      mix/qml/html/cm/show-hint.css
  3. 610
      mix/qml/html/cm/show-hint.js
  4. 32
      mix/qml/html/cm/solidity.js
  5. 29
      mix/qml/html/cm/solidityToken.js
  6. 1
      mix/qml/html/codeeditor.html
  7. 1
      mix/web.qrc

60
mix/qml/html/cm/anyword-hint.js

@ -20,18 +20,21 @@
var text = editor.getLine(line), m; var text = editor.getLine(line), m;
while (m = re.exec(text)) { while (m = re.exec(text)) {
if (line == cur.line && m[0] === curWord) continue; if (line == cur.line && m[0] === curWord) continue;
if ((!curWord || m[0].lastIndexOf(curWord, 0) == 0) && !Object.prototype.hasOwnProperty.call(seen, m[0])) { if ((!curWord || m[0].lastIndexOf(curWord, 0) === 0) && !Object.prototype.hasOwnProperty.call(seen, m[0])) {
seen[m[0]] = true; seen[m[0]] = true;
list.push(m[0]); list.push({ text: m[0] });
} }
} }
} }
} }
if (editor.getMode().name === "solidity")
for (var key in solidityKeywords())
{ {
if (curWord === false || key.indexOf(curWord, 0) === 0) list = addSolToken(curWord, list, solCurrency(), solCurrency);
list.push(key); list = addSolToken(curWord, list, solKeywords(), solKeywords);
list = addSolToken(curWord, list, solStdContract(), solStdContract);
list = addSolToken(curWord, list, solTime(), solTime);
list = addSolToken(curWord, list, solTypes(), solTypes);
list = addSolToken(curWord, list, solMisc(), solMisc);
} }
return {list: list, from: CodeMirror.Pos(cur.line, start), to: CodeMirror.Pos(cur.line, end)}; return {list: list, from: CodeMirror.Pos(cur.line, start), to: CodeMirror.Pos(cur.line, end)};
@ -39,35 +42,20 @@
})(); })();
solidityKeywords = function(list) function addSolToken(curWord, list, tokens, type)
{ {
var keywords = { "address":true, "indexed":true, "event":true, "delete":true, "break":true, "case":true, "constant":true, "continue":true, "contract":true, "default":true, for (var key in tokens)
"do":true, "else":true, "is":true, "for":true, "function":true, "if":true, "import":true, "mapping":true, "new":true, {
"public":true, "private":true, "return":true, "returns":true, "struct":true, "switch":true, "var":true, "while":true, if (curWord === false || key.indexOf(curWord, 0) === 0)
"int":true, "uint":true, "hash":true, "bool":true, "string":true, "string0":true, "text":true, "real":true, {
"ureal":true, var token = { text: key };
"owned":true, token.render = function(elt, data, cur)
"onlyowner":true, {
"named":true, elt.className = elt.className + " " + type.name.toLowerCase();
"mortal":true, elt.appendChild(document.createTextNode(cur.displayText || cur.text));
"coin":true }
}; list.push(token);
}
for (var i = 1; i <= 32; i++) { }
keywords["int" + i * 8] = true; return list;
keywords["uint" + i * 8] = true;
keywords["hash" + i * 8] = true;
keywords["string" + i] = true;
};
keywords["true"] = true;
keywords["false"] = true;
keywords["null"] = true;
keywords["Config"] = true;
keywords["NameReg"] = true;
keywords["CoinReg"] = true;
return keywords;
} }

24
mix/qml/html/cm/show-hint.css

@ -37,3 +37,27 @@
background: #4a90e2; background: #4a90e2;
color: white; color: white;
} }
.solcurrency {
color: red;
}
.solkeywords {
color: brown;
}
.solstdcontract {
color: blue;
}
.soltime {
color: green;
}
.soltypes {
color: orange;
}
.solMisc {
color: grey;
}

610
mix/qml/html/cm/show-hint.js

@ -5,338 +5,338 @@
var ACTIVE_HINT_ELEMENT_CLASS = "CodeMirror-hint-active"; var ACTIVE_HINT_ELEMENT_CLASS = "CodeMirror-hint-active";
CodeMirror.showHint = function(cm, getHints, options) { CodeMirror.showHint = function(cm, getHints, options) {
// We want a single cursor position. // We want a single cursor position.
if (cm.somethingSelected()) return; if (cm.somethingSelected()) return;
if (getHints == null) { if (getHints == null) {
if (options && options.async) return; if (options && options.async) return;
else getHints = CodeMirror.hint.auto; else getHints = CodeMirror.hint.auto;
} }
if (cm.state.completionActive) cm.state.completionActive.close(); if (cm.state.completionActive) cm.state.completionActive.close();
var completion = cm.state.completionActive = new Completion(cm, getHints, options || {}); var completion = cm.state.completionActive = new Completion(cm, getHints, options || {});
CodeMirror.signal(cm, "startCompletion", cm); CodeMirror.signal(cm, "startCompletion", cm);
if (completion.options.async) if (completion.options.async)
getHints(cm, function(hints) { completion.showHints(hints); }, completion.options); getHints(cm, function(hints) { completion.showHints(hints); }, completion.options);
else else
return completion.showHints(getHints(cm, completion.options)); return completion.showHints(getHints(cm, completion.options));
}; };
function Completion(cm, getHints, options) { function Completion(cm, getHints, options) {
this.cm = cm; this.cm = cm;
this.getHints = getHints; this.getHints = getHints;
this.options = options; this.options = options;
this.widget = this.onClose = null; this.widget = this.onClose = null;
} }
Completion.prototype = { Completion.prototype = {
close: function() { close: function() {
if (!this.active()) return; if (!this.active()) return;
this.cm.state.completionActive = null; this.cm.state.completionActive = null;
if (this.widget) this.widget.close(); if (this.widget) this.widget.close();
if (this.onClose) this.onClose(); if (this.onClose) this.onClose();
CodeMirror.signal(this.cm, "endCompletion", this.cm); CodeMirror.signal(this.cm, "endCompletion", this.cm);
}, },
active: function() { active: function() {
return this.cm.state.completionActive == this; return this.cm.state.completionActive == this;
}, },
pick: function(data, i) { pick: function(data, i) {
var completion = data.list[i]; var completion = data.list[i];
if (completion.hint) completion.hint(this.cm, data, completion); if (completion.hint) completion.hint(this.cm, data, completion);
else this.cm.replaceRange(getText(completion), completion.from||data.from, completion.to||data.to); else this.cm.replaceRange(getText(completion), completion.from||data.from, completion.to||data.to);
CodeMirror.signal(data, "pick", completion); CodeMirror.signal(data, "pick", completion);
this.close(); this.close();
}, },
showHints: function(data) { showHints: function(data) {
if (!data || !data.list.length || !this.active()) return this.close(); if (!data || !data.list.length || !this.active()) return this.close();
if (this.options.completeSingle != false && data.list.length == 1) if (this.options.completeSingle != false && data.list.length == 1)
this.pick(data, 0); this.pick(data, 0);
else else
this.showWidget(data); this.showWidget(data);
}, },
showWidget: function(data) { showWidget: function(data) {
this.widget = new Widget(this, data); this.widget = new Widget(this, data);
CodeMirror.signal(data, "shown"); CodeMirror.signal(data, "shown");
var debounce = 0, completion = this, finished; var debounce = 0, completion = this, finished;
var closeOn = this.options.closeCharacters || /[\s()\[\]{};:>,]/; var closeOn = this.options.closeCharacters || /[\s()\[\]{};:>,]/;
var startPos = this.cm.getCursor(), startLen = this.cm.getLine(startPos.line).length; var startPos = this.cm.getCursor(), startLen = this.cm.getLine(startPos.line).length;
var requestAnimationFrame = window.requestAnimationFrame || function(fn) { var requestAnimationFrame = window.requestAnimationFrame || function(fn) {
return setTimeout(fn, 1000/60); return setTimeout(fn, 1000/60);
}; };
var cancelAnimationFrame = window.cancelAnimationFrame || clearTimeout; var cancelAnimationFrame = window.cancelAnimationFrame || clearTimeout;
function done() { function done() {
if (finished) return; if (finished) return;
finished = true; finished = true;
completion.close(); completion.close();
completion.cm.off("cursorActivity", activity); completion.cm.off("cursorActivity", activity);
if (data) CodeMirror.signal(data, "close"); if (data) CodeMirror.signal(data, "close");
} }
function update() { function update() {
if (finished) return; if (finished) return;
CodeMirror.signal(data, "update"); CodeMirror.signal(data, "update");
if (completion.options.async) if (completion.options.async)
completion.getHints(completion.cm, finishUpdate, completion.options); completion.getHints(completion.cm, finishUpdate, completion.options);
else else
finishUpdate(completion.getHints(completion.cm, completion.options)); finishUpdate(completion.getHints(completion.cm, completion.options));
} }
function finishUpdate(data_) { function finishUpdate(data_) {
data = data_; data = data_;
if (finished) return; if (finished) return;
if (!data || !data.list.length) return done(); if (!data || !data.list.length) return done();
if (completion.widget) completion.widget.close(); if (completion.widget) completion.widget.close();
completion.widget = new Widget(completion, data); completion.widget = new Widget(completion, data);
} }
function clearDebounce() { function clearDebounce() {
if (debounce) { if (debounce) {
cancelAnimationFrame(debounce); cancelAnimationFrame(debounce);
debounce = 0; debounce = 0;
} }
} }
function activity() { function activity() {
clearDebounce(); clearDebounce();
var pos = completion.cm.getCursor(), line = completion.cm.getLine(pos.line); var pos = completion.cm.getCursor(), line = completion.cm.getLine(pos.line);
if (pos.line != startPos.line || line.length - pos.ch != startLen - startPos.ch || if (pos.line != startPos.line || line.length - pos.ch != startLen - startPos.ch ||
pos.ch < startPos.ch || completion.cm.somethingSelected() || pos.ch < startPos.ch || completion.cm.somethingSelected() ||
(pos.ch && closeOn.test(line.charAt(pos.ch - 1)))) { (pos.ch && closeOn.test(line.charAt(pos.ch - 1)))) {
completion.close(); completion.close();
} else { } else {
debounce = requestAnimationFrame(update); debounce = requestAnimationFrame(update);
if (completion.widget) completion.widget.close(); if (completion.widget) completion.widget.close();
} }
} }
this.cm.on("cursorActivity", activity); this.cm.on("cursorActivity", activity);
this.onClose = done; this.onClose = done;
} }
}; };
function getText(completion) { function getText(completion) {
if (typeof completion == "string") return completion; if (typeof completion == "string") return completion;
else return completion.text; else return completion.text;
} }
function buildKeyMap(options, handle) { function buildKeyMap(options, handle) {
var baseMap = { var baseMap = {
Up: function() {handle.moveFocus(-1);}, Up: function() {handle.moveFocus(-1);},
Down: function() {handle.moveFocus(1);}, Down: function() {handle.moveFocus(1);},
PageUp: function() {handle.moveFocus(-handle.menuSize() + 1, true);}, PageUp: function() {handle.moveFocus(-handle.menuSize() + 1, true);},
PageDown: function() {handle.moveFocus(handle.menuSize() - 1, true);}, PageDown: function() {handle.moveFocus(handle.menuSize() - 1, true);},
Home: function() {handle.setFocus(0);}, Home: function() {handle.setFocus(0);},
End: function() {handle.setFocus(handle.length - 1);}, End: function() {handle.setFocus(handle.length - 1);},
Enter: handle.pick, Enter: handle.pick,
Tab: handle.pick, Tab: handle.pick,
Esc: handle.close Esc: handle.close
}; };
var ourMap = options.customKeys ? {} : baseMap; var ourMap = options.customKeys ? {} : baseMap;
function addBinding(key, val) { function addBinding(key, val) {
var bound; var bound;
if (typeof val != "string") if (typeof val != "string")
bound = function(cm) { return val(cm, handle); }; bound = function(cm) { return val(cm, handle); };
// This mechanism is deprecated // This mechanism is deprecated
else if (baseMap.hasOwnProperty(val)) else if (baseMap.hasOwnProperty(val))
bound = baseMap[val]; bound = baseMap[val];
else else
bound = val; bound = val;
ourMap[key] = bound; ourMap[key] = bound;
} }
if (options.customKeys) if (options.customKeys)
for (var key in options.customKeys) if (options.customKeys.hasOwnProperty(key)) for (var key in options.customKeys) if (options.customKeys.hasOwnProperty(key))
addBinding(key, options.customKeys[key]); addBinding(key, options.customKeys[key]);
if (options.extraKeys) if (options.extraKeys)
for (var key in options.extraKeys) if (options.extraKeys.hasOwnProperty(key)) for (var key in options.extraKeys) if (options.extraKeys.hasOwnProperty(key))
addBinding(key, options.extraKeys[key]); addBinding(key, options.extraKeys[key]);
return ourMap; return ourMap;
} }
function getHintElement(hintsElement, el) { function getHintElement(hintsElement, el) {
while (el && el != hintsElement) { while (el && el != hintsElement) {
if (el.nodeName.toUpperCase() === "LI" && el.parentNode == hintsElement) return el; if (el.nodeName.toUpperCase() === "LI" && el.parentNode == hintsElement) return el;
el = el.parentNode; el = el.parentNode;
} }
} }
function Widget(completion, data) { function Widget(completion, data) {
this.completion = completion; this.completion = completion;
this.data = data; this.data = data;
var widget = this, cm = completion.cm, options = completion.options; var widget = this, cm = completion.cm, options = completion.options;
var hints = this.hints = document.createElement("ul"); var hints = this.hints = document.createElement("ul");
hints.className = "CodeMirror-hints"; hints.className = "CodeMirror-hints";
this.selectedHint = options.getDefaultSelection ? options.getDefaultSelection(cm,options,data) : 0; this.selectedHint = options.getDefaultSelection ? options.getDefaultSelection(cm,options,data) : 0;
var completions = data.list; var completions = data.list;
for (var i = 0; i < completions.length; ++i) { for (var i = 0; i < completions.length; ++i) {
var elt = hints.appendChild(document.createElement("li")), cur = completions[i]; var elt = hints.appendChild(document.createElement("li")), cur = completions[i];
var className = HINT_ELEMENT_CLASS + (i != this.selectedHint ? "" : " " + ACTIVE_HINT_ELEMENT_CLASS); var className = HINT_ELEMENT_CLASS + (i != this.selectedHint ? "" : " " + ACTIVE_HINT_ELEMENT_CLASS);
if (cur.className != null) className = cur.className + " " + className; if (cur.className != null) className = cur.className + " " + className;
elt.className = className; elt.className = className;
if (cur.render) cur.render(elt, data, cur); if (cur.render) cur.render(elt, data, cur);
else elt.appendChild(document.createTextNode(cur.displayText || getText(cur))); else elt.appendChild(document.createTextNode(cur.displayText || getText(cur)));
elt.hintId = i; elt.hintId = i;
} }
var pos = cm.cursorCoords(options.alignWithWord !== false ? data.from : null); var pos = cm.cursorCoords(options.alignWithWord !== false ? data.from : null);
var left = pos.left, top = pos.bottom, below = true; var left = pos.left, top = pos.bottom, below = true;
hints.style.left = left + "px"; hints.style.left = left + "px";
hints.style.top = top + "px"; hints.style.top = top + "px";
// If we're at the edge of the screen, then we want the menu to appear on the left of the cursor. // If we're at the edge of the screen, then we want the menu to appear on the left of the cursor.
var winW = window.innerWidth || Math.max(document.body.offsetWidth, document.documentElement.offsetWidth); var winW = window.innerWidth || Math.max(document.body.offsetWidth, document.documentElement.offsetWidth);
var winH = window.innerHeight || Math.max(document.body.offsetHeight, document.documentElement.offsetHeight); var winH = window.innerHeight || Math.max(document.body.offsetHeight, document.documentElement.offsetHeight);
(options.container || document.body).appendChild(hints); (options.container || document.body).appendChild(hints);
var box = hints.getBoundingClientRect(), overlapY = box.bottom - winH; var box = hints.getBoundingClientRect(), overlapY = box.bottom - winH;
if (overlapY > 0) { if (overlapY > 0) {
var height = box.bottom - box.top, curTop = box.top - (pos.bottom - pos.top); var height = box.bottom - box.top, curTop = box.top - (pos.bottom - pos.top);
if (curTop - height > 0) { // Fits above cursor if (curTop - height > 0) { // Fits above cursor
hints.style.top = (top = curTop - height) + "px"; hints.style.top = (top = curTop - height) + "px";
below = false; below = false;
} else if (height > winH) { } else if (height > winH) {
hints.style.height = (winH - 5) + "px"; hints.style.height = (winH - 5) + "px";
hints.style.top = (top = pos.bottom - box.top) + "px"; hints.style.top = (top = pos.bottom - box.top) + "px";
var cursor = cm.getCursor(); var cursor = cm.getCursor();
if (data.from.ch != cursor.ch) { if (data.from.ch != cursor.ch) {
pos = cm.cursorCoords(cursor); pos = cm.cursorCoords(cursor);
hints.style.left = (left = pos.left) + "px"; hints.style.left = (left = pos.left) + "px";
box = hints.getBoundingClientRect(); box = hints.getBoundingClientRect();
} }
} }
} }
var overlapX = box.left - winW; var overlapX = box.left - winW;
if (overlapX > 0) { if (overlapX > 0) {
if (box.right - box.left > winW) { if (box.right - box.left > winW) {
hints.style.width = (winW - 5) + "px"; hints.style.width = (winW - 5) + "px";
overlapX -= (box.right - box.left) - winW; overlapX -= (box.right - box.left) - winW;
} }
hints.style.left = (left = pos.left - overlapX) + "px"; hints.style.left = (left = pos.left - overlapX) + "px";
} }
cm.addKeyMap(this.keyMap = buildKeyMap(options, { cm.addKeyMap(this.keyMap = buildKeyMap(options, {
moveFocus: function(n, avoidWrap) { widget.changeActive(widget.selectedHint + n, avoidWrap); }, moveFocus: function(n, avoidWrap) { widget.changeActive(widget.selectedHint + n, avoidWrap); },
setFocus: function(n) { widget.changeActive(n); }, setFocus: function(n) { widget.changeActive(n); },
menuSize: function() { return widget.screenAmount(); }, menuSize: function() { return widget.screenAmount(); },
length: completions.length, length: completions.length,
close: function() { completion.close(); }, close: function() { completion.close(); },
pick: function() { widget.pick(); }, pick: function() { widget.pick(); },
data: data data: data
})); }));
if (options.closeOnUnfocus !== false) { if (options.closeOnUnfocus !== false) {
var closingOnBlur; var closingOnBlur;
cm.on("blur", this.onBlur = function() { closingOnBlur = setTimeout(function() { completion.close(); }, 100); }); cm.on("blur", this.onBlur = function() { closingOnBlur = setTimeout(function() { completion.close(); }, 100); });
cm.on("focus", this.onFocus = function() { clearTimeout(closingOnBlur); }); cm.on("focus", this.onFocus = function() { clearTimeout(closingOnBlur); });
} }
var startScroll = cm.getScrollInfo(); var startScroll = cm.getScrollInfo();
cm.on("scroll", this.onScroll = function() { cm.on("scroll", this.onScroll = function() {
var curScroll = cm.getScrollInfo(), editor = cm.getWrapperElement().getBoundingClientRect(); var curScroll = cm.getScrollInfo(), editor = cm.getWrapperElement().getBoundingClientRect();
var newTop = top + startScroll.top - curScroll.top; var newTop = top + startScroll.top - curScroll.top;
var point = newTop - (window.pageYOffset || (document.documentElement || document.body).scrollTop); var point = newTop - (window.pageYOffset || (document.documentElement || document.body).scrollTop);
if (!below) point += hints.offsetHeight; if (!below) point += hints.offsetHeight;
if (point <= editor.top || point >= editor.bottom) return completion.close(); if (point <= editor.top || point >= editor.bottom) return completion.close();
hints.style.top = newTop + "px"; hints.style.top = newTop + "px";
hints.style.left = (left + startScroll.left - curScroll.left) + "px"; hints.style.left = (left + startScroll.left - curScroll.left) + "px";
}); });
CodeMirror.on(hints, "dblclick", function(e) { CodeMirror.on(hints, "dblclick", function(e) {
var t = getHintElement(hints, e.target || e.srcElement); var t = getHintElement(hints, e.target || e.srcElement);
if (t && t.hintId != null) {widget.changeActive(t.hintId); widget.pick();} if (t && t.hintId != null) {widget.changeActive(t.hintId); widget.pick();}
}); });
CodeMirror.on(hints, "click", function(e) { CodeMirror.on(hints, "click", function(e) {
var t = getHintElement(hints, e.target || e.srcElement); var t = getHintElement(hints, e.target || e.srcElement);
if (t && t.hintId != null) { if (t && t.hintId != null) {
widget.changeActive(t.hintId); widget.changeActive(t.hintId);
if (options.completeOnSingleClick) widget.pick(); if (options.completeOnSingleClick) widget.pick();
} }
}); });
CodeMirror.on(hints, "mousedown", function() { CodeMirror.on(hints, "mousedown", function() {
setTimeout(function(){cm.focus();}, 20); setTimeout(function(){cm.focus();}, 20);
}); });
CodeMirror.signal(data, "select", completions[0], hints.firstChild); CodeMirror.signal(data, "select", completions[0], hints.firstChild);
return true; return true;
} }
Widget.prototype = { Widget.prototype = {
close: function() { close: function() {
if (this.completion.widget != this) return; if (this.completion.widget != this) return;
this.completion.widget = null; this.completion.widget = null;
this.hints.parentNode.removeChild(this.hints); this.hints.parentNode.removeChild(this.hints);
this.completion.cm.removeKeyMap(this.keyMap); this.completion.cm.removeKeyMap(this.keyMap);
var cm = this.completion.cm; var cm = this.completion.cm;
if (this.completion.options.closeOnUnfocus !== false) { if (this.completion.options.closeOnUnfocus !== false) {
cm.off("blur", this.onBlur); cm.off("blur", this.onBlur);
cm.off("focus", this.onFocus); cm.off("focus", this.onFocus);
} }
cm.off("scroll", this.onScroll); cm.off("scroll", this.onScroll);
}, },
pick: function() { pick: function() {
this.completion.pick(this.data, this.selectedHint); this.completion.pick(this.data, this.selectedHint);
}, },
changeActive: function(i, avoidWrap) { changeActive: function(i, avoidWrap) {
if (i >= this.data.list.length) if (i >= this.data.list.length)
i = avoidWrap ? this.data.list.length - 1 : 0; i = avoidWrap ? this.data.list.length - 1 : 0;
else if (i < 0) else if (i < 0)
i = avoidWrap ? 0 : this.data.list.length - 1; i = avoidWrap ? 0 : this.data.list.length - 1;
if (this.selectedHint == i) return; if (this.selectedHint == i) return;
var node = this.hints.childNodes[this.selectedHint]; var node = this.hints.childNodes[this.selectedHint];
node.className = node.className.replace(" " + ACTIVE_HINT_ELEMENT_CLASS, ""); node.className = node.className.replace(" " + ACTIVE_HINT_ELEMENT_CLASS, "");
node = this.hints.childNodes[this.selectedHint = i]; node = this.hints.childNodes[this.selectedHint = i];
node.className += " " + ACTIVE_HINT_ELEMENT_CLASS; node.className += " " + ACTIVE_HINT_ELEMENT_CLASS;
if (node.offsetTop < this.hints.scrollTop) if (node.offsetTop < this.hints.scrollTop)
this.hints.scrollTop = node.offsetTop - 3; this.hints.scrollTop = node.offsetTop - 3;
else if (node.offsetTop + node.offsetHeight > this.hints.scrollTop + this.hints.clientHeight) else if (node.offsetTop + node.offsetHeight > this.hints.scrollTop + this.hints.clientHeight)
this.hints.scrollTop = node.offsetTop + node.offsetHeight - this.hints.clientHeight + 3; this.hints.scrollTop = node.offsetTop + node.offsetHeight - this.hints.clientHeight + 3;
CodeMirror.signal(this.data, "select", this.data.list[this.selectedHint], node); CodeMirror.signal(this.data, "select", this.data.list[this.selectedHint], node);
}, },
screenAmount: function() { screenAmount: function() {
return Math.floor(this.hints.clientHeight / this.hints.firstChild.offsetHeight) || 1; return Math.floor(this.hints.clientHeight / this.hints.firstChild.offsetHeight) || 1;
} }
}; };
CodeMirror.registerHelper("hint", "auto", function(cm, options) { CodeMirror.registerHelper("hint", "auto", function(cm, options) {
var helpers = cm.getHelpers(cm.getCursor(), "hint"), words; var helpers = cm.getHelpers(cm.getCursor(), "hint"), words;
if (helpers.length) { if (helpers.length) {
for (var i = 0; i < helpers.length; i++) { for (var i = 0; i < helpers.length; i++) {
var cur = helpers[i](cm, options); var cur = helpers[i](cm, options);
if (cur && cur.list.length) return cur; if (cur && cur.list.length) return cur;
} }
} else if (words = cm.getHelper(cm.getCursor(), "hintWords")) { } else if (words = cm.getHelper(cm.getCursor(), "hintWords")) {
if (words) return CodeMirror.hint.fromList(cm, {words: words}); if (words) return CodeMirror.hint.fromList(cm, {words: words});
} else if (CodeMirror.hint.anyword) { } else if (CodeMirror.hint.anyword) {
return CodeMirror.hint.anyword(cm, options); return CodeMirror.hint.anyword(cm, options);
} }
}); });
CodeMirror.registerHelper("hint", "fromList", function(cm, options) { CodeMirror.registerHelper("hint", "fromList", function(cm, options) {
var cur = cm.getCursor(), token = cm.getTokenAt(cur); var cur = cm.getCursor(), token = cm.getTokenAt(cur);
var found = []; var found = [];
for (var i = 0; i < options.words.length; i++) { for (var i = 0; i < options.words.length; i++) {
var word = options.words[i]; var word = options.words[i];
if (word.slice(0, token.string.length) == token.string) if (word.slice(0, token.string.length) == token.string)
found.push(word); found.push(word);
} }
if (found.length) return { if (found.length) return {
list: found, list: found,
from: CodeMirror.Pos(cur.line, token.start), from: CodeMirror.Pos(cur.line, token.start),
to: CodeMirror.Pos(cur.line, token.end) to: CodeMirror.Pos(cur.line, token.end)
}; };
}); });
CodeMirror.commands.autocomplete = CodeMirror.showHint; CodeMirror.commands.autocomplete = CodeMirror.showHint;

32
mix/qml/html/cm/solidity.js

@ -12,32 +12,10 @@
CodeMirror.defineMode("solidity", function(config) { CodeMirror.defineMode("solidity", function(config) {
var indentUnit = config.indentUnit; var indentUnit = config.indentUnit;
var keywords = { "address":true, "indexed":true, "event":true, "delete":true, "break":true, "case":true, "constant":true, "continue":true, "contract":true, "default":true, var types = solTypes();
"do":true, "else":true, "is":true, "for":true, "function":true, "if":true, "import":true, "mapping":true, "new":true, var stdContract = solStdContract();
"public":true, "private":true, "return":true, "returns":true, "struct":true, "switch":true, "var":true, "while":true, var keywords = solKeywords();
"int":true, "uint":true, "hash":true, "bool":true, "string":true, "string0":true, "text":true, "real":true, var atoms = solMisc();
"ureal":true,
"owned":true,
"onlyowner":true,
"named":true,
"mortal":true,
"coin":true,
};
for (var i = 1; i <= 32; i++) {
keywords["int" + i * 8] = true;
keywords["uint" + i * 8] = true;
keywords["hash" + i * 8] = true;
keywords["string" + i] = true;
};
var atoms = {
"true":true, "false":true, "null":true,
"Config":true,
"NameReg":true,
"CoinReg":true,
};
var isOperatorChar = /[+\-*&^%:=<>!|\/]/; var isOperatorChar = /[+\-*&^%:=<>!|\/]/;
@ -84,6 +62,8 @@ CodeMirror.defineMode("solidity", function(config) {
return "keyword"; return "keyword";
} }
if (atoms.propertyIsEnumerable(cur)) return "atom"; if (atoms.propertyIsEnumerable(cur)) return "atom";
if (types.propertyIsEnumerable(cur)) return "variable-2";
if (stdContract.propertyIsEnumerable(cur)) return "variable-3";
return "variable"; return "variable";
} }

29
mix/qml/html/cm/solidityToken.js

@ -0,0 +1,29 @@
function solCurrency()
{
return { "wei": true, "szabo": true, "finney": true, "ether": true };
}
function solKeywords()
{
return { "break": true, "case": true, "constant": true, "continue": true, "contract": true, "default": true, "do": true, "else": true, "event": true, "external": true, "is": true, "indexed": true, "for": true, "function": true, "if": true, "import": true, "mapping": true, "modifier": true, "new": true, "public": true, "private": true, "internal": true, "return": true, "returns": true, "struct": true, "switch": true, "var": true, "while": true, "enum": true };
}
function solStdContract()
{
return { "Config": true, "NameReg": true, "CoinReg": true, "owned": true, "onlyowner": true, "named": true, "mortal": true, "coin": true };
}
function solTime()
{
return { "seconds": true, "minutes": true, "hours": true, "days": true, "weeks": true, "years": true, "after": true };
}
function solTypes()
{
return { "int": true, "int8": true, "int16": true, "int24": true, "int32": true, "int40": true, "int48": true, "int56": true, "int64": true, "int72": true, "int80": true, "int88": true, "int96": true, "int104": true, "int112": true, "int120": true, "int128": true, "int136": true, "int144": true, "int152": true, "int160": true, "int168": true, "int178": true, "int184": true, "int192": true, "int200": true, "int208": true, "int216": true, "int224": true, "int232": true, "int240": true, "int248": true, "int256": true, "uint": true, "uint8": true, "uint16": true, "uint24": true, "uint32": true, "uint40": true, "uint48": true, "uint56": true, "uint64": true, "uint72": true, "uint80": true, "uint88": true, "uint96": true, "uint104": true, "uint112": true, "uint120": true, "uint128": true, "uint136": true, "uint144": true, "uint152": true, "uint160": true, "uint168": true, "uint178": true, "uint184": true, "uint192": true, "uint200": true, "uint208": true, "uint216": true, "uint224": true, "uint232": true, "uint240": true, "uint248": true, "uint256": true, "bytes0": true, "bytes1": true, "bytes2": true, "bytes3": true, "bytes4": true, "bytes5": true, "bytes6": true, "bytes7": true, "bytes8": true, "bytes9": true, "bytes10": true, "bytes11": true, "bytes12": true, "bytes13": true, "bytes14": true, "bytes15": true, "bytes16": true, "bytes17": true, "bytes18": true, "bytes19": true, "bytes20": true, "bytes21": true, "bytes22": true, "bytes23": true, "bytes24": true, "bytes25": true, "bytes26": true, "bytes27": true, "bytes28": true, "bytes29": true, "bytes30": true, "bytes31": true, "bytes32": true, "bytes": true, "byte": true, "address": true, "bool": true, "string": true, "real": true, "ureal": true };
}
function solMisc()
{
return { "true": true, "false": true, "null": true, "funtion": true, "contract": true };
}

1
mix/qml/html/codeeditor.html

@ -10,6 +10,7 @@
<script src="cm/css.js"></script> <script src="cm/css.js"></script>
<script src="cm/xml.js"></script> <script src="cm/xml.js"></script>
<script src="cm/htmlmixed.js"></script> <script src="cm/htmlmixed.js"></script>
<script src="cm/solidityToken.js"></script>
<script src="cm/solidity.js"></script> <script src="cm/solidity.js"></script>
<script src="cm/fullscreen.js"></script> <script src="cm/fullscreen.js"></script>
<script src="cm/active-line.js"></script> <script src="cm/active-line.js"></script>

1
mix/web.qrc

@ -25,5 +25,6 @@
<file>qml/html/cm/show-hint.js</file> <file>qml/html/cm/show-hint.js</file>
<file>qml/html/cm/show-hint.css</file> <file>qml/html/cm/show-hint.css</file>
<file>qml/html/cm/closebrackets.js</file> <file>qml/html/cm/closebrackets.js</file>
<file>qml/html/cm/solidityToken.js</file>
</qresource> </qresource>
</RCC> </RCC>

Loading…
Cancel
Save