-
-
Notifications
You must be signed in to change notification settings - Fork 37
/
Copy pathopen_include.py
570 lines (469 loc) · 20.5 KB
/
open_include.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
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
import os.path
import re
import threading
import urllib
import time
import sublime
import sublime_plugin
from os.path import expanduser
try:
from .Edit import Edit as Edit
except:
from Edit import Edit as Edit
BINARY = re.compile('\.(psd|ai|cdr|ico|cache|sublime-package|eot|svgz|ttf|woff|zip|tar|gz|rar|bz2|jar|xpi|mov|mpeg|avi|mpg|flv|wmv|mp3|wav|aif|aiff|snd|wma|asf|asx|pcm|pdf|doc|docx|xls|xlsx|ppt|pptx|rtf|sqlite|sqlitedb|fla|swf|exe)$', re.I)
IMAGE = re.compile('\.(apng|png|jpg|gif|jpeg|bmp)$', re.I)
settings = None
os_specific_settings = None
debug = False
ST2 = int(sublime.version()) < 3000
cache = {}
def debug_info(msg):
if debug:
print( msg )
def reset_cache():
global cache
debug_info('-- reset cache --')
cache = {}
cache['os_listdir'] = {}
cache['os_exists'] = {}
cache['os_is_file'] = {}
cache['os_is_dir'] = {}
cache['done'] = {}
cache['checked'] = {}
cache['look_into_folders'] = False
cache['running'] = False
cache['folder'] = False
cache['folder_save'] = True
reset_cache()
def Window():
return sublime.active_window()
def normalize(path):
return os.path.normpath(path.replace('\\', '/').replace('\\', '/'))
def unique(seq):
seen = set()
seen_add = seen.add
return [ x for x in seq if not (x in seen or seen_add(x))]
def os_listdir(path):
global cache
id = normalize(path)
if id not in cache['os_listdir']:
try:
cache['os_listdir'][id] = [os.path.join(path, x).replace('\\', '/') for x in os.listdir(path) if os_is_dir(os.path.join(path, x))]
except:
cache['os_listdir'][id] = []
return cache['os_listdir'][id]
def os_exists(path):
global cache
id = normalize(path)
if id not in cache['os_exists']:
try:
cache['os_exists'][id] = os.path.lexists(path)
except:
cache['os_exists'][id] = False
return cache['os_exists'][id]
def os_is_file(path):
global cache
id = normalize(path)
if id not in cache['os_is_file']:
cache['os_is_file'][id] = os_exists(path) and os.path.isfile(path)
return cache['os_is_file'][id]
def os_is_dir(path):
global cache
id = normalize(path)
if id not in cache['os_is_dir']:
cache['os_is_dir'][id] = os_exists(path) and os.path.isdir(path)
return cache['os_is_dir'][id]
def plugin_loaded():
global settings
global os_specific_settings
settings = sublime.load_settings('Open-Include.sublime-settings')
os_specific_settings = {}
if sublime.platform() == 'windows':
os_specific_settings = sublime.load_settings('Open-Include (Windows).sublime-settings')
elif sublime.platform() == 'osx':
os_specific_settings = sublime.load_settings('Open-Include (OSX).sublime-settings')
else:
os_specific_settings = sublime.load_settings('Open-Include (Linux).sublime-settings')
def get_setting(key, default=None):
return os_specific_settings.get(key, settings.get(key, default))
class OpenInclude(sublime_plugin.TextCommand):
def run(self, edit = None):
if not cache['running']:
cache['running'] = True
if ST2:
OpenIncludeThread().run()
else:
OpenIncludeThread().start()
class OpenIncludeThread(threading.Thread):
def __init__(self):
threading.Thread.__init__(self)
def run(self):
global cache
debug_info('-- running --')
window = sublime.active_window()
view = window.active_view()
something_opened = False
for region in view.sel():
opened = False
debug_info('------------------------------------')
begin = time.time()
# find in files panel
if not opened and 'Find Results.hidden-tmLanguage' in view.settings().get('syntax'):
opened = OpenIncludeFindInFileGoto().run(view);
# selected text
if not opened:
debug_info('\n# selected')
debug_info(view.substr(region))
opened = self.resolve_path(window, view, view.substr(region))
if not opened:
opened = self.try_open_folder(view.substr(region))
# quoted/scope
if not opened and view.score_selector(region.begin(), "parameter.url, string.quoted"):
file_to_open = view.substr(view.extract_scope(region.begin()))
debug_info('\n# quotes')
opened = self.resolve_path(window, view, file_to_open)
if not opened:
opened = self.try_open_folder(file_to_open)
if not opened and get_setting('create_if_not_exists') and view.file_name():
if file_to_open.startswith('http'):
pass
else:
file_to_open = self.resolve_relative(os.path.dirname(view.file_name()), re.sub('"|\'|<|>|\(|\)|\{|\}|\[|\]|;', '', file_to_open))
branch, leaf = os.path.split(file_to_open)
try:
os.makedirs(branch)
except:
pass
self.open(window, file_to_open)
opened = True
# selection expanded to full lines
if not opened:
expanded_lines = view.substr(sublime.Region(view.line(region.begin()).begin(), view.line(region.end()).end()))
debug_info('\n# expanded lines')
opened = self.resolve_path(window, view, expanded_lines)
if not opened:
opened = self.try_open_folder(expanded_lines)
# current line quotes and parenthesis
if not opened:
line = view.substr(view.line(region.begin()))
debug_info('\n# line')
for line in re.split("[(){}\[\]'\"]", line):
line = line.strip()
if line:
opened = self.resolve_path(window, view, line)
if opened:
break
if not opened:
opened = self.try_open_folder(line)
# split by spaces and tabs
if not opened:
words = re.sub("\s+", "\n", expanded_lines)
opened = self.resolve_path(window, view, words)
# word
if not opened:
file_to_open = view.substr(view.word(region)).strip()
debug_info('\n# word')
opened = self.resolve_path(window, view, file_to_open)
if opened:
something_opened = True
# Nothing in a selected region could be opened
if not something_opened:
# run again, and look into every subfolder and it its parent folders
if not cache['look_into_folders']:
cache['look_into_folders'] = True
debug_info('-- running again --')
opened = self.run()
if not opened:
if cache['folder']:
sublime.status_message("Opening Folder: " + normalize(cache['folder']))
window.run_command("open_dir", {"dir": cache['folder']})
reset_cache()
return True
else:
sublime.status_message("Unable to find a file in the current selection")
reset_cache()
return False
else:
reset_cache()
return True
else:
reset_cache()
return True
def expand_paths_with_extensions(self, window, view, paths):
# Special file naming conventions, e.g. '_'+name+'.scss' + current extension
extensions = get_setting('auto_extension', [])
if view.file_name():
file_ext = os.path.splitext(view.file_name())[1]
extensions.append(dict(extension=file_ext))
path_add = []
for path in paths:
if path.startswith('http'):
continue
extension_original = os.path.splitext(path)[1]
for extension in extensions:
if not extension_original:
subs = path.replace('\\', '/').split('/')
subs[-1] = re.sub('"|\'', '', subs[-1]);
subs[-1] = extension.get('prefix', '') + subs[-1] + extension.get('extension', '')
path_add.append(os.path.join(*subs).replace('\\', '/'))
path_add.append(extension.get('prefix', '') + path + extension.get('extension', ''))
return unique(paths + path_add)
def expand_paths_with_sub_and_parent_folders(self, window, view, paths):
paths2 = []
for path in paths:
paths2.append(path)
if not view.file_name():
# folders of opened views
for _view in window.views():
if _view.file_name():
branch, leaf = os.path.split(_view.file_name())
for path in paths:
paths2.append(os.path.join(branch, path).replace('\\', '/'))
else:
# subfolders
branch, leaf = os.path.split(view.file_name())
for dir in os_listdir(branch):
for path in paths:
paths2.append(os.path.join(dir, path).replace('\\', '/'))
# parent folders
branch, leaf = os.path.split(branch)
for dir in os_listdir(branch):
for path in paths:
paths2.append(os.path.join(dir, path).replace('\\', '/'))
# folders of opened views
for _view in window.views():
if _view.file_name():
branch, leaf = os.path.split(_view.file_name())
for path in paths:
paths2.append(os.path.join(branch, path).replace('\\', '/'))
# subfolders of the project folders
for branch in window.folders():
for dir in os_listdir(branch):
for path in paths:
paths2.append(os.path.join(dir, path).replace('\\', '/'))
return unique(paths2)
# resolve the path of these sources and send to try_open
def resolve_path(self, window, view, paths, skip_folders = False):
global cache
if not paths.strip():
return False
debug_info('-- original paths --')
debug_info(paths)
debug_info('-- /original paths --')
try:
paths_decoded = urllib.unquote(paths.encode('utf8'))
paths_decoded = unicode(paths_decoded.decode('utf8'))
paths += '\n' + paths_decoded
except:
try:
paths_decoded = urllib.parse.unquote(paths)
paths += '\n' + paths_decoded
except:
pass
user_home_path = expanduser("~")
paths = re.sub('"|\'|<|>|\(|\)|\{|\}|\[|\]|;', '', paths)
for path in paths.split('\n'):
if not path.startswith('http'):
# remove quotes
paths += '\n' + re.sub(';', '', path)
# remove :row:col
paths += '\n' + re.sub('(\:[0-9]*)+$', '', path).strip()
# replace . for /
paths += '\n' + path.replace('./', '.').replace('.', '/')
# replace :: for /
paths += '\n' + path.replace('::', '/')
# replace ~ for the user's home directory
if get_setting('expand_tilde', True):
paths += '\n' + path.replace('~', user_home_path)
if get_setting('expand_env_var', True):
paths += '\n' + os.path.expandvars(path)
paths = paths.strip().split('\n')
if paths[0].startswith('http'):
return self.try_open(window, paths[0])
if get_setting('use_strict'):
return self.try_open(window, self.resolve_relative(os.path.dirname(view.file_name()), paths[0]))
paths = self.expand_paths_with_extensions(window, view, paths)
if cache['look_into_folders'] and not skip_folders:
paths = self.expand_paths_with_sub_and_parent_folders(window, view, paths)
something_opened = False
folder_structure = [] + ["../" * i for i in range(get_setting('maximum_folder_up', 5))]
if view.file_name():
view_dirname = os.path.dirname(view.file_name())
view_dirname_dirname = os.path.dirname(view_dirname)
paths = unique(paths)
debug_info('-- resolved paths --')
debug_info(paths)
debug_info('-- /resolved paths --')
if len(paths) > 10:
print("-- There are too many paths, probably we shoulnd't open! Path count: %d" % len(paths))
return False
for path in paths:
path = path.strip()
path_normalized = normalize(path)
if path == '' or path_normalized in cache['done']:
continue
cache['done'][path_normalized] = True
cache['folder_save'] = False
# relative to view & view dir name
opened = False
if view.file_name():
for new_path_prefix in folder_structure:
opened = self.create_path_relative_to_folder(window, view, view_dirname, new_path_prefix + path)
if not opened:
opened = self.create_path_relative_to_folder(window, view, view_dirname_dirname, new_path_prefix + path)
if opened:
break
# relative to project folders
if not opened:
for maybe_path in sublime.active_window().folders():
for new_path_prefix in folder_structure:
if self.create_path_relative_to_folder(window, view, maybe_path, new_path_prefix + path):
opened = True
break
if opened:
break
cache['folder_save'] = True
# absolute
if not opened:
opened = self.try_open(window, path)
if opened:
something_opened = True
break
return something_opened
def create_path_relative_to_folder(self, window, view, maybe_path, path):
maybe_path_tpm = self.resolve_relative(maybe_path, path)
return self.try_open(window, maybe_path_tpm)
# try opening the resouce
def try_open(self, window, maybe_path):
global cache
debug_info('Trying to open: ' + maybe_path)
path_normalized = normalize(maybe_path)
if path_normalized in cache['checked']:
return False
cache['checked'][path_normalized] = True
if maybe_path.startswith('http'):
# HTTP URL
if BINARY.search(maybe_path) or get_setting("open_http_in_browser", False):
sublime.status_message("Opening in Browser " + maybe_path)
import webbrowser
webbrowser.open_new_tab(maybe_path)
else:
sublime.status_message("Opening URL " + maybe_path)
# Create thread to download url in background
threading.Thread(target=self.read_url, args=(maybe_path,)).start()
elif os_is_file(maybe_path):
sublime.status_message("Opening File " + normalize(maybe_path))
if IMAGE.search(maybe_path):
self.open(window, maybe_path)
elif BINARY.search(maybe_path):
try:
import desktop
except:
from . import desktop
desktop.open(maybe_path)
else:
# Open within ST
self.open(window, maybe_path)
elif maybe_path and ( os_is_dir(maybe_path) or os_is_dir('\\' + maybe_path) ) and not cache['folder'] and cache['folder_save']:
# Walkaround for UNC path
if maybe_path and maybe_path[0] == '\\':
maybe_path = '\\' + maybe_path
cache['folder'] = maybe_path
return False
else:
return False
return True
# try opening the folder
def try_open_folder(self, text):
if cache['folder'] and re.sub('"|\'', '', text).strip() == cache['folder']:
sublime.status_message("Opening Folder: " + normalize(cache['folder']))
Window().run_command("open_dir", {"dir": cache['folder']})
return True
else:
return False
# util
def resolve_relative(self, absolute, path):
subs = path.replace('\\', '/').split('/')
for sub in subs:
if sub != '':
absolute = os.path.join(absolute, sub)
return absolute.replace('\\', '/')
def read_url(self, url):
try:
if url.startswith('https'):
url = 'http' + url[5:]
import urllib.request
req = urllib.request.urlopen(url)
content = req.read()
encoding = req.headers['content-type'].split('charset=')[-1]
try:
content = str(content, encoding)
except:
try:
content = str(content, 'utf-8')
except:
content = str(content, 'utf8', errors="replace")
content_type = req.headers['content-type'].split(';')[0]
# ST3 is thread-safe, but ST2 is not so we use set_timeout to get to the main thread again
sublime.set_timeout(lambda: self.read_url_on_done(content, content_type), 0)
except:
pass
def read_url_on_done(self, content, content_type):
if content:
window = sublime.active_window()
view = window.new_file()
with Edit(view) as edit:
edit.insert(0, content)
# TODO: convert to a dict and include in settings
if content_type == 'text/html':
view.settings().set('syntax', 'Packages/HTML/HTML.tmLanguage')
elif content_type == 'text/css':
view.settings().set('syntax', 'Packages/CSS/CSS.tmLanguage')
elif content_type == 'text/javascript' or content_type == 'application/javascript' or content_type == 'application/x-javascript':
view.settings().set('syntax', 'Packages/JavaScript/JavaScript.tmLanguage')
elif content_type == 'application/json' or content_type == 'text/json':
view.settings().set('syntax', 'Packages/JavaScript/JSON.tmLanguage')
elif content_type == 'text/xml' or content_type == 'application/xml':
view.settings().set('syntax', 'Packages/XML/XML.tmLanguage')
def open(self, window, path):
if get_setting('in_secondary_colum', False):
window.run_command('set_layout', {"cols": [0.0, 0.5, 1.0], "rows": [0.0, 1.0], "cells": [[0, 0, 1, 1], [1, 0, 2, 1]]})
window.focus_group(1)
window.open_file(path)
class OpenIncludeFindInFileGoto():
def run(self, view):
line_no = self.get_line_no(view)
file_name = self.get_file(view)
if line_no is not None and file_name is not None:
file_loc = "%s:%s" % (file_name, line_no)
self.open(file_loc, sublime.ENCODED_POSITION)
return True
elif file_name is not None:
self.open(file_name)
return True
return False
def get_line_no(self, view):
if len(view.sel()) == 1:
line_text = view.substr(view.line(view.sel()[0]))
match = re.match(r"\s*(\d+).+", line_text)
if match:
return match.group(1)
return None
def get_file(self, view):
if len(view.sel()) == 1:
line = view.line(view.sel()[0])
while line.begin() > 0:
line_text = view.substr(line)
match = re.match(r"^(.+)\:$", line_text)
if match:
if os_exists(match.group(1)):
return match.group(1)
line = view.line(line.begin() - 1)
return None
def open(self, path, position = False):
if get_setting('in_secondary_colum', False):
sublime.active_window().run_command('set_layout', {"cols": [0.0, 0.5, 1.0], "rows": [0.0, 1.0], "cells": [[0, 0, 1, 1], [1, 0, 2, 1]]})
sublime.active_window().focus_group(1)
sublime.active_window().open_file(path, position)
if int(sublime.version()) < 3000:
plugin_loaded()