feat(SublimeText2.EditorPackages): cache packages

This commit is contained in:
Iristyle
2013-04-04 08:55:15 -04:00
parent d65666cdfc
commit c3efdad2c2
274 changed files with 26863 additions and 0 deletions

View File

@@ -0,0 +1,393 @@
# coding: utf8
import sublime
import sublime_plugin
from sublime import Region
import re
# for detecting "real" brackets in BracketeerCommand, and bracket matching in BracketeerBracketMatcher
OPENING_BRACKETS = ['{', '[', '(']
OPENING_BRACKET_LIKE = ['{', '[', '(', '"', "'", u'', '', '«', '']
CLOSING_BRACKETS = ['}', ']', ')']
CLOSING_BRACKET_LIKE = ['}', ']', ')', '"', "'", u'', '', '»', '']
QUOTING_BRACKETS = ['\'', "\""]
class BracketeerCommand(sublime_plugin.TextCommand):
def run(self, edit, **kwargs):
e = self.view.begin_edit('bracketeer')
regions = [region for region in self.view.sel()]
# sort by region.end() DESC
def get_end(region):
return region.end()
regions.sort(key=get_end, reverse=True)
for region in regions:
self.run_each(edit, region, **kwargs)
self.view.end_edit(e)
def complicated_quote_checker(self, insert_braces, region, pressed, after, r_brace):
in_string_scope = self.view.score_selector(region.a, 'string')
in_double_string_scope = in_string_scope and self.view.score_selector(region.a, 'string.quoted.double')
in_single_string_scope = in_string_scope and self.view.score_selector(region.a, 'string.quoted.single')
at_eol = region.a == self.view.line(region.a).b
in_comment_scope = self.view.score_selector(region.a, 'comment')
in_text_scope = self.view.score_selector(region.a, 'text')
in_embedded_scope = self.view.score_selector(region.a, 'source.php') + self.view.score_selector(region.a, 'source.js')
in_text_scope = in_text_scope and not in_embedded_scope
if pressed and pressed in QUOTING_BRACKETS and (in_comment_scope or in_text_scope or in_string_scope):
# if the cursor:
# (a) is preceded by odd numbers of '\'s?
if in_comment_scope:
scope_test = 'comment'
else:
scope_test = 'string'
begin_of_string = region.a
while begin_of_string and self.view.score_selector(begin_of_string - 1, scope_test):
begin_of_string -= 1
check_a = self.view.substr(Region(begin_of_string, region.a))
check_a = len(re.search(r'[\\]*$', check_a).group(0))
check_a = check_a % 2
# (b) is an apostrophe and (inside double quotes or in text or comment scope)
check_b = (in_double_string_scope or in_text_scope or in_comment_scope) and pressed == "'"
# (c) we are at the end of the line and pressed the closing quote
check_c = at_eol and (
in_single_string_scope and pressed == "'"
or
in_double_string_scope and pressed == '"'
)
# then don't insert both, just insert the one.
if check_a or check_b or check_c:
return pressed
def run_each(self, edit, region, braces='{}', pressed=None, unindent=False, select=False, replace=False):
self.view.sel().subtract(region)
if self.view.settings().get('translate_tabs_to_spaces'):
tab = ' ' * self.view.settings().get('tab_size')
else:
tab = "\t"
row, col = self.view.rowcol(region.begin())
indent_point = self.view.text_point(row, 0)
if indent_point < region.begin():
indent = self.view.substr(Region(indent_point, region.begin()))
indent = re.match('[ \t]*', indent).group(0)
else:
indent = ''
line = self.view.substr(self.view.line(region.a))
selection = self.view.substr(region)
# for braces that have newlines ("""), insert the current line's indent
if isinstance(braces, list):
l_brace = braces[0]
r_brace = braces[1]
braces = ''.join(braces)
braces = braces.replace("\n", "\n" + indent)
length = len(l_brace)
else:
braces = braces.replace("\n", "\n" + indent)
length = len(braces) / 2
l_brace = braces[:length]
r_brace = braces[length:]
if region.empty():
after = self.view.substr(Region(region.a, region.a + length))
insert_braces = braces
complicated_check = self.complicated_quote_checker(insert_braces, region, pressed, after, r_brace)
if complicated_check:
insert_braces = complicated_check
elif pressed and after == r_brace and r_brace[-1] == pressed: # and (pressed not in QUOTING_BRACKETS or in_string_scope):
# in this case we pressed the closing character, and that's the character that is to the right
# so do nothing except advance cursor position
insert_braces = False
elif unindent and row > 0 and indent and line == indent:
# indent has the current line's indent
# get previous line's indent:
prev_point = self.view.text_point(row - 1, 0)
prev_line = self.view.line(prev_point)
prev_indent = self.view.substr(prev_line)
prev_indent = re.match('[ \t]*', prev_indent).group(0)
if (not pressed or pressed == l_brace) and len(indent) > len(prev_indent) and indent[len(prev_indent):] == tab:
# move region.a back by 'indent' amount
region = Region(region.a - len(tab), region.b - len(tab))
# and remove the tab
self.view.replace(edit, Region(region.a, region.a + len(tab) - 1), '')
elif pressed and pressed == r_brace:
if len(indent) == len(prev_indent):
# move region.a back by 'indent' amount
region = Region(region.a - len(tab), region.b - len(tab))
# and remove the tab
self.view.replace(edit, Region(region.a, region.a + len(tab) - 1), '')
insert_braces = r_brace
elif pressed and pressed != l_brace:
# we pressed the closing bracket or quote. This *never*
insert_braces = r_brace
if insert_braces:
self.view.insert(edit, region.a, insert_braces)
self.view.sel().add(Region(region.a + length, region.a + length))
elif selection in QUOTING_BRACKETS and pressed in QUOTING_BRACKETS and selection != pressed:
# changing a quote from single <=> double, just insert the quote.
self.view.replace(edit, region, pressed)
self.view.sel().add(Region(region.end(), region.end()))
elif pressed and pressed != l_brace:
b = region.begin() + len(r_brace)
self.view.replace(edit, region, r_brace)
self.view.sel().add(Region(b, b))
else:
substitute = self.view.substr(region)
replacement = l_brace + substitute + r_brace
# if we're inserting "real" brackets, not quotes:
real_brackets = l_brace in OPENING_BRACKETS and r_brace in CLOSING_BRACKETS
# check to see if entire lines are selected, and if so do some smart indenting
bol_is_nl = region.begin() == 0 or self.view.substr(region.begin() - 1) == "\n"
eol_is_nl = region.end() == self.view.size() - 1 or self.view.substr(region.end() - 1) == "\n"
if real_brackets and bol_is_nl and eol_is_nl:
indent = ''
final = ''
m = re.match('([ \t]*)' + tab, self.view.substr(region))
if m:
indent = m.group(1)
final = "\n"
else:
substitute = tab + substitute
replacement = indent + l_brace + "\n" + substitute + indent + r_brace + final
b = region.begin() + len(replacement) - len("\n" + indent + r_brace + final)
else:
b = region.begin() + len(replacement)
if replace and self.view.substr(region.begin() - 1) in OPENING_BRACKET_LIKE and self.view.substr(region.end()) in CLOSING_BRACKET_LIKE:
b -= 1
self.view.replace(edit, Region(region.begin() - 1, region.end() + 1), replacement)
elif replace and self.view.substr(region.begin()) in OPENING_BRACKET_LIKE and self.view.substr(region.end() - 1) in CLOSING_BRACKET_LIKE:
replacement = l_brace + replacement[2:-2] + r_brace
b -= 2
self.view.replace(edit, region, replacement)
l_brace = r_brace = ''
else:
self.view.replace(edit, region, replacement)
if select:
self.view.sel().add(Region(b - len(replacement) + len(l_brace), b - len(r_brace)))
else:
self.view.sel().add(Region(b, b))
class BracketeerIndentCommand(sublime_plugin.TextCommand):
def run(self, edit):
e = self.view.begin_edit('bracketeer')
if self.view.settings().get('translate_tabs_to_spaces'):
tab = ' ' * self.view.settings().get('tab_size')
else:
tab = "\t"
regions = [region for region in self.view.sel()]
# sort by region.end() DESC
def get_end(region):
return region.end()
regions.sort(key=get_end, reverse=True)
for region in regions:
if region.empty():
# insert tab at beginning of line
point = self.view.text_point(self.view.rowcol(region.a)[0], 0)
self.view.insert(edit, point, tab)
else:
# insert tab in front of lines 1:-1
lines = self.view.substr(region).split("\n")
# just one line? indent it
if len(lines) == 1:
substitute = tab + lines[0] + "\n"
else:
default_settings = sublime.load_settings("bracketeer.sublime-settings")
dont_indent_list = default_settings.get('dont_indent')
# lines that start with these strings don't get indented
def dont_indent(line):
return any(dont for dont in dont_indent_list if line[:len(dont)] == dont)
# cursor is at start of line? indent that, too
if len(lines[0]) > 0 and not dont_indent(lines[0]):
substitute = tab
else:
substitute = ''
substitute += lines[0] + "\n"
for line in lines[1:-1]:
if len(line):
if not dont_indent(line):
substitute += tab
substitute += line
substitute += "\n"
substitute += lines[-1]
self.view.replace(edit, region, substitute)
self.view.end_edit(e)
class BracketeerBracketMatcher(sublime_plugin.TextCommand):
def find_brackets(self, region, closing_search_brackets=None):
match_map = {
'}': '{',
']': '[',
')': '(',
}
# find next brace in closing_search_brackets
if not closing_search_brackets:
closing_search_brackets = CLOSING_BRACKETS
elif isinstance(closing_search_brackets, basestring):
closing_search_brackets = [closing_search_brackets]
opening_search_brackets = [match_map[bracket] for bracket in closing_search_brackets]
begin_point = region.begin() - 1
end_point = region.end()
# LaTEX: if selection directly preceeds \right, examine the string that includes \right instead of the actual selection
if self.view.substr( Region(end_point, min(end_point+6,self.view.size())) ) == '\\right': end_point += 6
# /LaTEX
# end_point gets incremented immediately, which skips the first
# character, *unless* the selection is empty, in which case the
# inner contents should be selected before scanning past
if region.empty():
# if the current character is a bracket, and the character to the left is the
# *matching* bracket, don't match the empty contents
c = self.view.substr(end_point)
if c in closing_search_brackets and self.view.substr(end_point - 1) == match_map[c]:
# cursor is between two brackets - select them and return
return Region(begin_point, end_point + 1)
else:
# if the selection is inside two brackets, select them and return
c1 = self.view.substr(begin_point)
c2 = self.view.substr(end_point)
if c2 in closing_search_brackets and c1 == match_map[c2]:
# LaTEX: if \left preceeds selection, select it as well
if self.view.substr(Region(max(begin_point-5,0), begin_point))=='\left': begin_point -= 5
# /LaTEX
return Region(begin_point, end_point + 1)
# scan forward searching for a closing bracket.
started_in_string = bool(self.view.score_selector(end_point, 'string') or self.view.score_selector(begin_point, 'string'))
bracket_count = 0
while True:
c = self.view.substr(end_point)
if started_in_string or not self.view.score_selector(end_point, 'string'):
if bracket_count <= 0 and c in closing_search_brackets:
break
elif c in opening_search_brackets and c in OPENING_BRACKETS:
bracket_count += 1
elif c in closing_search_brackets and c in CLOSING_BRACKETS:
bracket_count -= 1
end_point += 1
if end_point >= self.view.size():
return None
# found a bracket, scan backwards until matching bracket is found.
# matching bracket is determined by counting closing brackets (+1)
# and opening brackets (-1) and when the count is zero and the
# matching opening bracket is found
look_for = match_map[c]
while True:
c = self.view.substr(begin_point)
if started_in_string or not self.view.score_selector(begin_point, 'string'):
if bracket_count == 0 and c == look_for:
break
elif c in opening_search_brackets and c in OPENING_BRACKETS:
bracket_count += 1
elif c in closing_search_brackets and c in CLOSING_BRACKETS:
bracket_count -= 1
begin_point -= 1
if begin_point < 0:
return None
# the current point is to the left of the opening bracket,
# I want it to be to the right.
begin_point += 1
# LaTEX: if selection ends in \right, don't select it
if self.view.substr( Region(max(end_point-6,0), end_point) ) == '\\right': end_point -= 6
# /LaTEX
return Region(begin_point, end_point)
class BracketeerGotoCommand(BracketeerBracketMatcher):
def run(self, edit, **kwargs):
e = self.view.begin_edit('bracketeer')
regions = [region for region in self.view.sel()]
# sort by region.end() DESC
def get_end(region):
return region.end()
regions.sort(key=get_end, reverse=True)
for region in regions:
self.run_each(edit, region, **kwargs)
self.view.end_edit(e)
def run_each(self, edit, region, goto):
cursor = region.b
if goto == "left" and self.view.substr(cursor - 1) == '{':
cursor -= 1
elif goto == "both" and self.view.substr(cursor) == '{':
cursor += 1
elif goto in ["left", "both"] and self.view.substr(cursor - 1) == '}':
cursor -= 1
new_region = self.find_brackets(Region(cursor, cursor), '}')
if new_region:
self.view.sel().subtract(region)
a = new_region.begin()
b = new_region.end()
if self.view.substr(a) in OPENING_BRACKETS:
a += 1
if self.view.substr(b) in CLOSING_BRACKETS:
b += 1
if goto == "left":
new_region = Region(a, a)
self.view.sel().add(new_region)
self.view.show(new_region)
elif goto == "right":
new_region = Region(b, b)
self.view.sel().add(new_region)
self.view.show(new_region)
elif goto == "both":
self.view.sel().add(Region(a, a))
self.view.sel().add(Region(b, b))
self.view.show(new_region.b)
else:
raise ValueError("`goto` should have a value of 'left', 'right', or 'both'), not '" + goto + '"')
class BracketeerSelectCommand(BracketeerBracketMatcher):
def run(self, edit, **kwargs):
e = self.view.begin_edit('bracketeer')
regions = [region for region in self.view.sel()]
# sort by region.end() DESC
def get_end(region):
return region.end()
regions.sort(key=get_end, reverse=True)
for region in regions:
self.run_each(edit, region, **kwargs)
self.view.end_edit(e)
def run_each(self, edit, region):
new_region = self.find_brackets(region)
if new_region:
self.view.sel().subtract(region)
self.view.sel().add(new_region)
self.view.show(new_region.b)