forked from wiktor-obrebski/SublimeJavaCompiler
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathjavacbase.py
190 lines (149 loc) · 6.15 KB
/
javacbase.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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
import sublime, sublime_plugin
import threading, subprocess, functools
import os
try:
from . import edit
except ValueError:
import edit
Edit = edit.Edit
properties = None
def sget(key, default):
global properties
if properties is None:
properties = sublime.load_settings("SublimeJavaCompiler.sublime-settings")
return properties.get(key, default)
def invoke(callback, *args, **kwargs):
# sublime.set_timeout gets used to send things onto the main thread
# most sublime.[something] calls need to be on the main thread
sublime.set_timeout(functools.partial(callback, *args, **kwargs), 0)
class OutputWindow(object):
""" Thread-safe output window
"""
def __init__(self, window, name='javac_wnd'):
self.window = window
self.name = name
self.outputWindow = None
def _getOutputWindow(self):
if (None is self.outputWindow):
self.outputWindow = self.window.get_output_panel(self.name)
self.clear()
return self.outputWindow
def show(self):
def _show():
self._getOutputWindow()
self.window.run_command("show_panel", {"panel": "output." + self.name})
invoke(_show)
def close(self):
def _close():
self.window.run_command("hide_panel", {"panel": "output." + self.name, "cancel": True})
invoke(_close)
def clear(self):
def _clear():
outputWindow = self._getOutputWindow()
outputWindow.set_read_only(False)
with Edit(outputWindow) as edit:
edit.erase(sublime.Region(0, outputWindow.size()))
outputWindow.set_read_only(True)
invoke(_clear)
def write(self, data, new_line=True):
""" Thread-safe writing text to console and adding new line.
"""
def _plain_write(outputWindow, data, new_line=True):
str = data # data.decode("utf-8")
str = str.replace('\r\n', '\n').replace('\r', '\n')
if new_line and not str.endswith('\n'): str += '\n'
outputWindow.set_read_only(False)
with Edit(outputWindow) as edit:
edit.insert(outputWindow.size(), str)
outputWindow.show(outputWindow.size())
outputWindow.set_read_only(True)
text = str(data)
outputWindow = self._getOutputWindow()
invoke(self.show)
invoke(_plain_write, outputWindow, text, new_line)
class CommandBase(sublime_plugin.TextCommand):
def write(self, text):
self.output().write(text)
def output(self):
if not hasattr(self, '_output'):
self._output = OutputWindow(self.view.window())
self._output.show()
return self._output
def call_new_thread_chain(self, orders_list, donemsg="\n------------Done------------"):
""" Calling all cmd in "orders_list" array, after last
close output wnd, if no errors
"""
def _callback(has_errors):
if has_errors: return
if _callback.counter >= len(orders_list):
if donemsg:
self.write(donemsg)
if not has_errors: #and settings.get('hide_output_after_compilation', True):
self.output().close()
return
orders_tuple = orders_list[_callback.counter]()
try:
cmd, working_dir, in_new_window = orders_tuple
except ValueError:
cmd, working_dir = orders_tuple
in_new_window = False
_callback.counter += 1
self.call_new_thread(cmd, _callback, working_dir, in_new_window)
_callback.counter = 0
_callback(False)
def call_new_thread(self, cmd, on_done=None, working_dir=".", in_new_window = False):
thread = JavaCThread(cmd, on_done, self.write, working_dir, in_new_window)
thread.start()
def run(self, edit):
self._run(edit)
class JavaCThread(threading.Thread):
""" Wrapper for calling externall application in thread for this plugin.
first argument is application path, second (on_done) - callback, with one argument (has_errors).
"""
def __init__(self, cmd, on_done = None, out_method = None, working_dir = ".", in_new_window = False, **kwargs):
threading.Thread.__init__(self)
if working_dir is None or working_dir == '':
working_dir = '.'
self.cmd = cmd
self.on_done = on_done
self.working_dir = working_dir
self.kwargs = kwargs
self.out_method = out_method
self.in_new_window = in_new_window
def log(self, text):
if self.out_method is None: return
self.out_method(text)
def run(self):
try:
log = self.log
# Ignore directories that no longer exist
if os.path.isdir(self.working_dir):
# Per http://bugs.python.org/issue8557 shell=True is required to
# get $PATH on Windows. Yay portable code.
shell = os.name == 'nt'
if self.working_dir != '':
os.chdir(self.working_dir)
_creationflags = 0
if self.in_new_window and os.name == 'nt':
_creationflags=subprocess.CREATE_NEW_CONSOLE
proc = subprocess.Popen(self.cmd,
shell=shell,
universal_newlines=True,
stderr=subprocess.STDOUT,
stdout=subprocess.PIPE,
creationflags=_creationflags
)
for line in iter(proc.stdout.readline, ''):
log(line)
proc.wait()
has_errors = proc.returncode != 0
if self.on_done is not None:
invoke(self.on_done, has_errors)
except subprocess.CalledProcessError as e:
if self.on_done is not None:
invoke(self.on_done, e.returncode)
# except OSError as e:
# if e.errno == os.errno.ENOENT:
# self.log('Unable to locate javac in your system.')
# if self.on_done is not None:
# invoke(self.on_done, e.returncode)