forked from michaelpnash/sublime-ensime
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathensime_notes.py
146 lines (114 loc) · 3.91 KB
/
ensime_notes.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
import os, sys, stat, functools
import sublime, sublime_plugin
from ensime_server import EnsimeOnly
import ensime_environment
import sexp
ensime_env = ensime_environment.ensime_env
class LangNote:
def __init__(self, lang, msg, fname, severity, start, end, line, col):
self.lang = lang
self.message = msg
self.file_name = fname
self.severity = severity
self.start = start
self.end = end
self.line = line
self.col = col
def lang_note(lang, m):
return LangNote(
lang,
m[":msg"],
m[":file"],
m[":severity"],
m[":beg"],
m[":end"],
m[":line"],
m[":col"])
def erase_error_highlights(view):
view.erase_regions("ensime-error")
view.erase_regions("ensime-error-underline")
def highlight_errors(view, notes):
if notes is None:
print "There were no notes?"
return
print "higlighting errors"
errors = [view.full_line(note.start) for note in notes]
underlines = []
for note in notes:
underlines += [sublime.Region(int(pos)) for pos in range(note.start, note.end)]
view.add_regions(
"ensime-error-underline",
underlines,
"invalid.illegal",
sublime.DRAW_EMPTY_AS_OVERWRITE)
view.add_regions(
"ensime-error",
errors,
"invalid.illegal",
"cross",
sublime.DRAW_OUTLINED)
view_notes = {}
class EnsimeNotes(sublime_plugin.TextCommand, EnsimeOnly):
def run(self, edit, action = "add", lang = "scala", value=None):
if not hasattr(self, "notes"):
self.notes = []
if action == "add":
new_notes = [lang_note(lang, m) for m in value]
self.notes.extend(new_notes)
highlight_errors(self.view, self.notes)
elif action == "clear":
self.notes = []
erase_error_highlights(self.view)
elif action == "display":
nn = self.notes
vw = self.view
vpos = vw.line(vw.sel()[0].begin()).begin()
if len(nn) > 0 and len([a for a in nn if self.view.line(int(a.start)).begin() == vpos]) > 0:
msgs = [note.message for note in self.notes]
self.view.set_status("ensime-typer", "; ".join(set(msgs)))
else:
self.view.erase_status("ensime-typer")
#sublime.set_timeout(functools.partial(self.view.run_command, "ensime_inspect_type_at_point", self.view.id()), 200)
def run_check(view):
view.checked = True
view.run_command("ensime_type_check_file")
class BackgroundTypeChecker(sublime_plugin.EventListener):
def _is_valid_file(self, view):
return bool(not view.file_name() is None and view.file_name().endswith(("scala","java")))
def on_load(self, view):
if self._is_valid_file(view):
run_check(view)
def on_post_save(self, view):
if self._is_valid_file(view):
run_check(view)
def on_selection_modified(self, view):
if self._is_valid_file(view):
view.run_command("ensime_notes", { "action": "display" })
class EnsimeInspectTypeAtPoint(sublime_plugin.TextCommand, EnsimeOnly):
def handle_reply(self, data):
d = data[1][1]
handlers = {
lambda d: d[":name"] == "<notype>": None,
lambda d: ":arrow-type" in d and d[":arrow-type"] == True: self.get_function_string,
lambda d: True: self.default_string
}
keys = [str(v) for i, v in enumerate(d) if i % 2 == 0]
values = [v for i, v in enumerate(d) if i % 2 != 0]
data_dict = dict(zip(keys,values))
msg = None
for cond in handlers:
if cond(data_dict):
msg = handlers[cond](data_dict)
break
self.view.erase_status("ensime-typer")
if msg is not None:
self.view.set_status("ensime-typer", msg)
def run(self, edit):
if self.view.file_name():
cl = ensime_environment.ensime_env.client()
if not cl is None:
cl.inspect_type_at_point(self.view.file_name(), self.view.sel()[0].begin(), self.handle_reply)
def get_function_string(self, d):
return str(d[":name"])
def default_string(self, d):
return "({0}) {1}".format(str(d[":decl-as"]), str(d[":full-name"]))