completion filtering and sorting - #235

This commit is contained in:
Vadim Lopatin 2017-08-29 17:54:59 +03:00
parent 0f2fc02157
commit 9a6619cd33
2 changed files with 95 additions and 9 deletions

View File

@ -30,9 +30,14 @@ enum DCDResult : int {
FAIL, FAIL,
} }
struct CompletionSymbol {
dstring name;
char kind;
}
alias DocCommentsResultSet = Tuple!(DCDResult, "result", string[], "docComments"); alias DocCommentsResultSet = Tuple!(DCDResult, "result", string[], "docComments");
alias FindDeclarationResultSet = Tuple!(DCDResult, "result", string, "fileName", ulong, "offset"); alias FindDeclarationResultSet = Tuple!(DCDResult, "result", string, "fileName", ulong, "offset");
alias CompletionResultSet = Tuple!(DCDResult, "result", dstring[], "output", char[], "completionKinds"); alias CompletionResultSet = Tuple!(DCDResult, "result", CompletionSymbol[], "output");
class DCDTask { class DCDTask {
@ -301,16 +306,17 @@ class DCDInterface : Thread {
result.result = DCDResult.SUCCESS; result.result = DCDResult.SUCCESS;
result.output.length = response.completions.length; result.output.length = response.completions.length;
result.completionKinds.length = response.completions.length;
int i=0; int i=0;
foreach(s;response.completions) { foreach(s;response.completions) {
char type = 0; char type = 0;
if (i < response.completionKinds.length) if (i < response.completionKinds.length)
type = response.completionKinds[i]; type = response.completionKinds[i];
result.completionKinds[i] = type; result.output[i].kind = type;
result.output[i++] = to!dstring(s); result.output[i].name = to!dstring(s);
i++;
} }
debug(DCD) Log.d("DCD output:\n", response.completions); postProcessCompletions(result.output);
debug(DCD) Log.d("DCD response:\n", response, "\nCompletion result:\n", result.output);
} }
override void postResults() { override void postResults() {
_callback(result); _callback(result);
@ -327,6 +333,87 @@ class DCDInterface : Thread {
} }
int completionTypePriority(char t) {
switch(t) {
case 'c': // - class name
return 10;
case 'i': // - interface name
return 10;
case 's': // - struct name
return 10;
case 'u': // - union name
return 10;
case 'v': // - variable name
return 5;
case 'm': // - member variable name
return 3;
case 'k': // - keyword, built-in version, scope statement
return 20;
case 'f': // - function or method
return 2;
case 'g': // - enum name
return 9;
case 'e': // - enum member
return 8;
case 'P': // - package name
return 30;
case 'M': // - module name
return 20;
case 'a': // - array
return 15;
case 'A': // - associative array
return 15;
case 'l': // - alias name
return 15;
case 't': // - template name
return 14;
case 'T': // - mixin template name
return 14;
default:
return 50;
}
}
int compareCompletionSymbol(ref CompletionSymbol v1, ref CompletionSymbol v2) {
import std.algorithm : cmp;
int p1 = v1.kind.completionTypePriority;
int p2 = v2.kind.completionTypePriority;
if (p1 < p2)
return -1;
if (p1 > p2)
return 1;
return v1.name.cmp(v2.name);
}
bool lessCompletionSymbol(ref CompletionSymbol v1, ref CompletionSymbol v2) {
return compareCompletionSymbol(v1, v2) < 0;
}
void postProcessCompletions(ref CompletionSymbol[] completions) {
import std.algorithm.sorting : sort;
completions.sort!(lessCompletionSymbol);
CompletionSymbol[] res;
bool hasKeywords = false;
bool hasNonKeywords = false;
bool[dstring] found;
foreach(s; completions) {
if (s.kind == 'k')
hasKeywords = true;
else
hasNonKeywords = true;
}
// remove duplicates; remove keywords if non-keyword items are found
foreach(s; completions) {
if (!(s.name in found)) {
found[s.name] = true;
if (s.kind != 'k' || !hasNonKeywords) {
res ~= s;
}
}
}
completions = res;
}
/// to test broken DCD after DUB invocation /// to test broken DCD after DUB invocation
/// run it after DCD ModuleCache is instantiated /// run it after DCD ModuleCache is instantiated

View File

@ -138,7 +138,7 @@ class DEditorTool : EditorTool
dstring[] labels; dstring[] labels;
foreach(index, label; output.output) { foreach(index, label; output.output) {
string iconId; string iconId;
char ch = index < output.completionKinds.length ? output.completionKinds[index] : 0; char ch = label.kind;
switch(ch) { switch(ch) {
case 'c': // - class name case 'c': // - class name
iconId = "symbol-class"; iconId = "symbol-class";
@ -192,12 +192,11 @@ class DEditorTool : EditorTool
iconId = "symbol-mixintemplate"; iconId = "symbol-mixintemplate";
break; break;
default: default:
iconId = "symbol-other";
break; break;
} }
if (!iconId)
iconId = "symbol-other";
icons ~= iconId; icons ~= iconId;
labels ~= label; labels ~= label.name;
} }
callback(labels, icons); callback(labels, icons);
_getCompletionsTask = null; _getCompletionsTask = null;