UltiSnips/PySnipEmu.py

386 lines
11 KiB
Python
Raw Normal View History

2009-06-23 14:45:04 +02:00
#!/usr/bin/env python
# encoding: utf-8
import vim
import string
import re
2009-07-01 17:11:33 +02:00
def debug(s):
f = open("/tmp/file.txt","a")
f.write(s+'\n')
f.close()
class TextObject(object):
"""
This base class represents any object in the text
that has a span in any ways
"""
def __init__(self, start, end):
self._start = start
self._end = end
2009-07-01 17:11:33 +02:00
self._delta_rows = 0
self._delta_cols = 0
@property
def start(self):
return self._start
@property
def end(self):
return self._end
2009-07-01 15:20:40 +02:00
def delta_rows():
def fget(self):
return self._delta_rows
def fset(self, value):
self._delta_rows = value
return locals()
delta_rows = property(**delta_rows())
def delta_cols():
def fget(self):
return self._delta_cols
def fset(self, value):
self._delta_cols = value
return locals()
delta_cols = property(**delta_cols())
2009-07-01 17:11:33 +02:00
class Mirror(TextObject):
"""
A Mirror object mirrors a TabStop that is, text is repeated here
"""
2009-07-01 17:11:33 +02:00
def __init__(self, ts, idx, start):
TextObject.__init__(self, (idx,start), (idx,start))
self._tabstop = ts
2009-07-01 15:20:40 +02:00
@property
def tabstop(self):
return self._tabstop
2009-07-01 15:20:40 +02:00
2009-07-01 17:11:33 +02:00
class TabStop(TextObject):
"""
This is the most important TextObject. A TabStop is were the cursor
comes to rest when the user taps through the Snippet.
"""
def __init__(self, no, idx, span, default_text = ""):
2009-07-01 17:11:33 +02:00
TextObject.__init__(self, (idx,span[0]), (idx,span[1]))
2009-07-01 15:20:40 +02:00
self._ct = default_text
def current_text():
def fget(self):
return self._ct
def fset(self, value):
self._ct = value
return locals()
current_text = property(**current_text())
@property
def span(self):
2009-07-01 17:11:33 +02:00
return (self._start[1]+self._delta_cols,self._start[1]+self._delta_cols+len(self._ct))
2009-07-01 17:11:33 +02:00
class SnippetInstance(TextObject):
"""
A Snippet instance is an instance of a Snippet Definition. That is,
when the user expands a snippet, a SnippetInstance is created to
keep track of the corresponding TextObjects. The Snippet itself is
also a TextObject because it has a start an end
"""
2009-07-01 15:20:40 +02:00
def __init__(self,start,end, ts, mirrors):
2009-07-01 17:11:33 +02:00
TextObject.__init__(self, start, end)
self._tabstops = ts
2009-07-01 15:20:40 +02:00
self._mirrors = mirrors
2009-07-01 17:11:33 +02:00
self._text_objects = ts.values() + mirrors
self._selected_tab = None
self._cts = None
for ts in self._tabstops.values():
self._update_mirrors(ts)
2009-07-01 17:11:33 +02:00
def select_next_tab(self, backwards = False):
if self._cts == 0:
if not backwards:
return False
if backwards:
cts_bf = self._cts
if self._cts == 0:
self._cts = max(self._tabstops.keys())
else:
self._cts -= 1
if self._cts <= 0:
self._cts = cts_bf
else:
# All tabs handled?
if self._cts is None:
self._cts = 1
else:
self._cts += 1
if self._cts not in self._tabstops:
self._cts = 0
if 0 not in self._tabstops:
return False
2009-07-01 17:11:33 +02:00
ts = self._tabstops[self._cts]
lineno, col = self._start
2009-07-01 17:11:33 +02:00
newline = lineno + ts.start[0]
if newline == lineno:
newcol = col + ts.span[0]
endcol = col + ts.span[1]
else:
newcol = ts.span[0]
endcol = ts.span[1]
vim.current.window.cursor = newline, newcol
# Select the word
# Depending on the current mode and position, we
# might need to move escape out of the mode and this
# will move our cursor one left
if endcol-newcol > 0:
if newcol != 0 and vim.eval("mode()") == 'i':
move_one_right = "l"
else:
move_one_right = ""
vim.command(r'call feedkeys("\<Esc>%sv%il\<c-g>")'
% (move_one_right, endcol-newcol-1))
2009-07-01 17:11:33 +02:00
self._selected_tab = ts
return True
2009-07-01 15:20:40 +02:00
def _update_mirrors(self,for_ts):
for m in self._mirrors:
if m.tabstop == for_ts:
mirror_line = m.start[0]
line = vim.current.buffer[mirror_line]
line = line[:m.start[1]+m.delta_cols] + \
for_ts.current_text + \
line[m.end[1]+m.delta_cols:]
oldspan = m.end[1]-m.start[1]
# TODO: evil hack!
m._end = (m.start[0],m.start[1]+len(for_ts.current_text))
newspan = m.end[1]-m.start[1]
2009-07-01 15:20:40 +02:00
vim.current.buffer[mirror_line] = line
self._move_to_on_line(newspan-oldspan, m.start[0], m.start[1]+m.delta_cols,cobj=m)
def _move_to_on_line(self,amount, lineno = None, col = None, cobj = None):
if self._cts is None:
return
if lineno is None and col is None:
lineno,col = vim.current.window.cursor
lineno -= 1
cobj = self._tabstops[self._cts]
2009-07-01 17:11:33 +02:00
for m in self._text_objects:
2009-07-01 15:20:40 +02:00
if m.start[0] != lineno:
continue
if m.start[1]+m.delta_cols >= col and m != cobj:
2009-07-01 15:20:40 +02:00
m.delta_cols += amount
def backspace(self,count):
cts = self._tabstops[self._cts]
2009-07-01 17:11:33 +02:00
ll = len(cts.current_text)
2009-07-01 15:20:40 +02:00
2009-07-01 17:11:33 +02:00
cts.current_text = cts.current_text[:-count]
self._move_to_on_line(len(cts.current_text)-ll)
2009-07-01 15:20:40 +02:00
self._update_mirrors(cts)
def chars_entered(self, chars):
cts = self._tabstops[self._cts]
2009-07-01 17:11:33 +02:00
if self._selected_tab is not None:
self._move_to_on_line(len(chars)-len(cts.current_text))
cts.current_text = ""
2009-07-01 17:11:33 +02:00
self._selected_tab = None
else:
self._move_to_on_line(len(chars))
2009-07-01 17:11:33 +02:00
2009-07-01 15:20:40 +02:00
cts.current_text += chars
self._update_mirrors(cts)
2009-06-23 14:45:04 +02:00
class Snippet(object):
2009-07-01 15:20:40 +02:00
_TABSTOP = re.compile(r'''(?xms)
(?:\${(\d+):(.*?)})| # A simple tabstop with default value
(?:\$(\d+)) # A mirror or a tabstop without default value.
''')
2009-06-23 14:45:04 +02:00
def __init__(self,trigger,value):
self._t = trigger
self._v = value
2009-06-23 14:45:04 +02:00
def trigger(self):
return self._t
trigger = property(trigger)
2009-07-01 15:20:40 +02:00
def _handle_tabstop(self, m, val, tabstops, mirrors):
no = int(m.group(1))
def_text = m.group(2)
2009-07-01 15:20:40 +02:00
start, end = m.span()
val = val[:start] + def_text + val[end:]
2009-07-01 15:20:40 +02:00
line_idx = val[:start].count('\n')
line_start = val[:start].rfind('\n') + 1
start_in_line = start - line_start
ts = TabStop(no, line_idx, (start_in_line,start_in_line+len(def_text)), def_text)
tabstops[no] = ts
2009-07-01 15:20:40 +02:00
return val
2009-07-01 15:20:40 +02:00
def _handle_ts_or_mirror(self, m, val, tabstops, mirrors):
no = int(m.group(3))
2009-07-01 15:20:40 +02:00
start, end = m.span()
val = val[:start] + val[end:]
2009-07-01 15:20:40 +02:00
line_idx = val[:start].count('\n')
line_start = val[:start].rfind('\n') + 1
start_in_line = start - line_start
2009-07-01 15:20:40 +02:00
if no in tabstops:
m = Mirror(tabstops[no], line_idx, start_in_line)
mirrors.append(m)
else:
ts = TabStop(no, line_idx, (start_in_line,start_in_line))
tabstops[no] = ts
2009-07-01 15:20:40 +02:00
return val
def _find_tabstops(self, val):
tabstops = {}
mirrors = []
while 1:
m = self._TABSTOP.search(val)
if m is not None:
if m.group(1) is not None: # ${1:hallo}
val = self._handle_tabstop(m,val,tabstops,mirrors)
elif m.group(3) is not None: # $1
val = self._handle_ts_or_mirror(m,val,tabstops,mirrors)
else:
break
return tabstops, mirrors, val
2009-07-01 15:20:40 +02:00
def _replace_tabstops(self):
ts, mirrors, val = self._find_tabstops(self._v)
lines = val.split('\n')
2009-07-01 15:20:40 +02:00
return ts, mirrors, lines
2009-06-23 14:45:04 +02:00
def launch(self, before, after):
2009-06-23 14:45:04 +02:00
lineno,col = vim.current.window.cursor
2009-06-23 14:45:04 +02:00
col -= len(self._t)
2009-07-01 15:20:40 +02:00
ts, mirrors, lines = self._replace_tabstops()
endcol = None
newline = 1
newcol = 0
if not len(ts):
newline = lineno + len(lines) - 1
if len(lines) == 1:
newcol = col + len(lines[-1])
else:
newcol = len(lines[-1])
2009-06-23 14:45:04 +02:00
lines[0] = before + lines[0]
lines[-1] += after
vim.current.buffer[lineno-1:lineno-1+len(lines)] = lines
vim.current.window.cursor = newline, newcol
2009-07-01 15:20:40 +02:00
if len(ts) or len(mirrors):
s = SnippetInstance( (lineno,col), (newline,newcol), ts, mirrors)
s.select_next_tab()
return s
2009-06-23 14:45:04 +02:00
class SnippetManager(object):
def __init__(self):
self.reset()
2009-07-01 15:20:40 +02:00
self._last_cursor_pos = None
def reset(self):
self._snippets = {}
self._current_snippets = []
2009-06-23 14:45:04 +02:00
def add_snippet(self,trigger,value):
self._snippets[trigger] = Snippet(trigger,value)
def try_expand(self, backwards = False):
if len(self._current_snippets):
cs = self._current_snippets[-1]
if not cs.select_next_tab(backwards):
self._current_snippets.pop()
2009-07-01 17:11:33 +02:00
self._last_cursor_pos = vim.current.window.cursor
return
2009-06-23 14:45:04 +02:00
line = vim.current.line
2009-06-23 14:45:04 +02:00
dummy,col = vim.current.window.cursor
if col > 0 and line[col-1] in string.whitespace:
return
# Get the word to the left of the current edit position
before,after = line[:col], line[col:]
word = before.split()[-1]
if word in self._snippets:
s = self._snippets[word].launch(before.rstrip()[:-len(word)], after)
self._last_cursor_pos = vim.current.window.cursor
if s is not None:
self._current_snippets.append(s)
def cursor_moved(self):
2009-07-01 17:11:33 +02:00
2009-07-01 15:20:40 +02:00
cp = vim.current.window.cursor
if len(self._current_snippets) and self._last_cursor_pos is not None:
2009-07-01 15:20:40 +02:00
lineno,col = cp
llineo,lcol = self._last_cursor_pos
# If we moved the line, somethings fishy.
if lineno == self._last_cursor_pos[0]:
cs = self._current_snippets[-1]
if lcol > col: # Some deleting was going on
cs.backspace(lcol-col)
else:
line = vim.current.line
chars = line[lcol:col]
cs.chars_entered(chars)
2009-07-01 17:11:33 +02:00
2009-07-01 15:20:40 +02:00
self._last_cursor_pos = cp
def entered_insert_mode(self):
pass
2009-06-23 14:45:04 +02:00
PySnipSnippets = SnippetManager()