531 lines
20 KiB
Python
531 lines
20 KiB
Python
from collections import defaultdict
|
|
import math, os, string
|
|
import color, method
|
|
from lex import Lexer
|
|
from point import Point
|
|
from render import RenderString
|
|
|
|
DEBUG = False
|
|
|
|
class ActionError(Exception):
|
|
pass
|
|
|
|
class Handler(object):
|
|
def __init__(self):
|
|
self.prefixes = set(["C-x", "C-c", "C-u"])
|
|
self.last_sequence = ''
|
|
self.curr_tokens = []
|
|
self.bindings = {}
|
|
|
|
# handle adding and removing actions
|
|
def add_action(self, action):
|
|
if self.window is None:
|
|
return
|
|
elif action.name in self.window.application.methods:
|
|
return
|
|
else:
|
|
self.window.application.methods[action.name] = action
|
|
def del_action(self, name):
|
|
if self.window is None:
|
|
return
|
|
for binding in self.bindings.keys():
|
|
if self.bindings[binding] == name:
|
|
del self.bindings[binding]
|
|
def add_binding(self, name, sequence):
|
|
if self.window is None:
|
|
#return
|
|
raise Exception("No window available")
|
|
elif not hasattr(self.window, 'application'):
|
|
raise Exception("No application available")
|
|
elif name not in self.window.application.methods:
|
|
raise Exception("No action called %r found" % name)
|
|
else:
|
|
self.bindings[sequence] = name
|
|
def add_bindings(self, name, sequences):
|
|
if self.window is None:
|
|
return
|
|
for sequence in sequences:
|
|
self.add_binding(name, sequence)
|
|
def del_binding(self, sequence):
|
|
if self.window is None:
|
|
return
|
|
del self.bindings[sequence]
|
|
def add_action_and_bindings(self, action, sequences):
|
|
if self.window is None:
|
|
return
|
|
self.add_action(action)
|
|
for sequence in sequences:
|
|
self.add_binding(action.name, sequence)
|
|
|
|
def handle_token(self, t):
|
|
'''self.handle_token(token): returns None, or the action to
|
|
take. raises an exception on unknown input'''
|
|
self.curr_tokens.append(t)
|
|
sequence = " ".join(self.curr_tokens)
|
|
if sequence in self.bindings:
|
|
act = self.window.application.methods[self.bindings[sequence]]
|
|
self.last_sequence = sequence
|
|
self.curr_tokens = []
|
|
return act
|
|
elif t in self.prefixes:
|
|
for binding in self.bindings:
|
|
if binding.startswith(sequence):
|
|
return None
|
|
self.curr_tokens = []
|
|
self.last_sequence = sequence
|
|
raise ActionError("no action defined for %r" % (sequence))
|
|
|
|
class Fundamental(Handler):
|
|
'''This is the default mode'''
|
|
modename = "Fundamental"
|
|
paths = []
|
|
basenames = []
|
|
extensions = []
|
|
detection = []
|
|
savetabs = False
|
|
tabwidth = 4
|
|
tabbercls = None
|
|
grammar = None
|
|
lexer = None
|
|
tabber = None
|
|
context = None
|
|
colors = {}
|
|
word_letters = None
|
|
commentc = None
|
|
|
|
# config settings installed/modified by the mode
|
|
config = {}
|
|
dconfig = {}
|
|
lconfig = {}
|
|
sconfig = {}
|
|
|
|
actions = []
|
|
_bindings = {}
|
|
completers = {}
|
|
#format = "%(flag)s %(bname)-18s (%(mname)s) %(indent)s %(cursor)s/%(mark)s %(perc)s"
|
|
format = "%(flag)s %(bname)-18s (%(mname)s) %(indent)s %(cursor)s %(perc)s %(vc-info)s"
|
|
|
|
header_size = 3
|
|
header_fg = 'default'
|
|
header_bg = 'blue'
|
|
|
|
## margins
|
|
def _get_header(self): return self.get_setting('header')
|
|
def _set_header(self, value): return self.set_setting('header', value)
|
|
header = property(_get_header, _set_header)
|
|
def _get_footer(self): return self.get_setting('footer')
|
|
def _set_footer(self, value): return self.set_setting('footer', value)
|
|
footer = property(_get_footer, _set_footer)
|
|
def _get_lmargin(self): return self.get_setting('lmargin')
|
|
def _set_lmargin(self, value): return self.set_setting('lmargin', value)
|
|
lmargin = property(_get_lmargin, _set_lmargin)
|
|
def _get_rmargin(self): return self.get_setting('rmargin')
|
|
def _set_rmargin(self, value): return self.set_setting('rmargin', value)
|
|
rmargin = property(_get_rmargin, _set_rmargin)
|
|
|
|
def install(cls, app):
|
|
app.setmode(cls.modename.lower(), cls, paths=cls.paths,
|
|
bases=cls.basenames, exts=cls.extensions,
|
|
detection=cls.detection)
|
|
for (key, val) in cls.colors.iteritems():
|
|
if key in app.token_colors:
|
|
s = 'token_colors: name %r conflict in %r' % (key, cls.modename)
|
|
raise Exception, s
|
|
else:
|
|
app.token_colors[key] = val
|
|
|
|
# install configuration stuff
|
|
for (key, val) in cls.config.iteritems():
|
|
assert key not in app.config, "config conflict: %r" % key
|
|
app.config[key] = val
|
|
for (key, val) in cls.lconfig.iteritems():
|
|
app.config.setdefault(key, [])
|
|
for val2 in val:
|
|
app.config[key].append(val2)
|
|
for (key, val) in cls.sconfig.iteritems():
|
|
app.config.setdefault(key, set())
|
|
app.config[key].add(val)
|
|
for (key, d) in cls.dconfig.iteritems():
|
|
app.config.setdefault(key, {})
|
|
for (subkey, val) in d.iteritems():
|
|
app.config[key][subkey] = val
|
|
|
|
for mcls in cls.actions:
|
|
m = mcls()
|
|
if m.name in app.methods:
|
|
return
|
|
else:
|
|
app.methods[m.name] = m
|
|
|
|
for (datatype, completer) in cls.completers.iteritems():
|
|
app.set_completer(datatype, completer)
|
|
install = classmethod(install)
|
|
|
|
def __init__(self, w):
|
|
assert w is not None
|
|
self.window = w
|
|
Handler.__init__(self)
|
|
|
|
#xyz
|
|
self.init_setting('lmargin', 0)
|
|
self.init_setting('rmargin', 1)
|
|
self.init_setting('header', 0)
|
|
self.init_setting('footer', 0)
|
|
|
|
# first let's add all the "default" actions
|
|
self.add_bindings('start-of-line', ('C-a', 'HOME',))
|
|
self.add_bindings('end-of-line', ('C-e', 'END',))
|
|
self.add_bindings('backward', ('C-b', 'L_ARROW',))
|
|
self.add_bindings('forward', ('C-f', 'R_ARROW',))
|
|
self.add_bindings('center-view', ('C-l',))
|
|
self.add_bindings('next-line', ('C-n', 'D_ARROW',))
|
|
self.add_bindings('previous-line', ('C-p', 'U_ARROW',))
|
|
self.add_bindings('next-section', ('M-n', 'M-D_ARROW',))
|
|
self.add_bindings('previous-section', ('M-p', 'M-U_ARROW',))
|
|
self.add_bindings('page-down', ('C-v', 'PG_DN',))
|
|
self.add_bindings('page-up', ('M-v', 'PG_UP',))
|
|
self.add_bindings('goto-beginning', ('M-<',))
|
|
self.add_bindings('goto-end', ('M->',))
|
|
self.add_bindings('delete-left', ('DELETE', 'BACKSPACE',))
|
|
self.add_bindings('delete-left-word', ('M-DELETE', 'M-BACKSPACE',))
|
|
self.add_bindings('delete-right', ('C-d',))
|
|
self.add_bindings('delete-right-word', ('M-d',))
|
|
self.add_bindings('kill-region', ('C-w',))
|
|
self.add_bindings('copy-region', ('M-w',))
|
|
self.add_bindings('kill', ('C-k',))
|
|
self.add_bindings('copy', ('M-k',))
|
|
self.add_bindings('yank', ('C-y',))
|
|
self.add_bindings('pop-kill', ('M-y',))
|
|
self.add_bindings('right-word', ('M-f',))
|
|
self.add_bindings('left-word', ('M-b',))
|
|
self.add_bindings('set-mark', ('C-@',))
|
|
self.add_bindings('switch-buffer', ('C-x b',))
|
|
self.add_bindings('switch-mark', ('C-x C-x',))
|
|
self.add_bindings('undo', ('C-/', 'C-x u',))
|
|
self.add_bindings('redo', ('M-/', 'M-_', 'C-x r',))
|
|
self.add_bindings('goto-line', ('M-g',))
|
|
self.add_bindings('forward-chars', ('C-x M-c',))
|
|
self.add_bindings('forward-lines', ('C-x M-n',))
|
|
self.add_bindings('search', ('C-s',))
|
|
self.add_bindings('reverse-search', ('C-r',))
|
|
self.add_bindings('regex-search', ('M-C-s',))
|
|
self.add_bindings('regex-reverse-search', ('M-C-r',))
|
|
self.add_bindings('toggle-margins', ('M-m',))
|
|
self.add_bindings('replace', ('M-%',))
|
|
self.add_bindings('regex-replace', ('M-$',))
|
|
self.add_bindings('open-file', ('C-x C-f',))
|
|
self.add_bindings('kill-buffer', ('C-x k',))
|
|
self.add_bindings('list-buffers', ('C-x C-b',))
|
|
self.add_bindings('meta-x', ('M-x',))
|
|
self.add_bindings('wrap-line', ('M-q',))
|
|
self.add_bindings('transpose-words', ('M-t',))
|
|
self.add_bindings('save-buffer', ('C-x C-s',))
|
|
self.add_bindings('save-buffer-as', ('C-x C-w',))
|
|
self.add_bindings('relex-buffer', ('M-r',))
|
|
self.add_bindings('exit', ('C-x C-c',))
|
|
self.add_bindings('split-window', ('C-x s', 'C-x 2',))
|
|
self.add_bindings('unsplit-window', ('C-u s', 'C-x 1',))
|
|
self.add_bindings('toggle-window', ('C-x o',))
|
|
self.add_bindings('delete-left-whitespace', ('C-c DELETE', 'C-c BACKSPACE',))
|
|
self.add_bindings('delete-right-whitespace', ('C-c d',))
|
|
self.add_bindings('insert-space', ('SPACE',))
|
|
self.add_bindings('insert-tab', ('TAB',))
|
|
self.add_bindings('insert-newline', ('RETURN',))
|
|
self.add_bindings('comment-region', ('C-c #',))
|
|
self.add_bindings('uncomment-region', ('C-u C-c #',))
|
|
self.add_bindings('justify-right', ('C-c f',))
|
|
self.add_bindings('justify-left', ('C-c b',))
|
|
self.add_bindings('indent-block', ('C-c >',))
|
|
self.add_bindings('unindent-block', ('C-c <',))
|
|
self.add_bindings('token-complete', ('C-x TAB',))
|
|
self.add_bindings('open-aes-file', ('C-c a',))
|
|
self.add_bindings('open-console', ('M-e',))
|
|
self.add_bindings('show-bindings-buffer', ('C-c M-h',))
|
|
self.add_bindings('show-functions-buffer', ('C-c M-?',))
|
|
self.add_bindings('which-command', ('M-h',))
|
|
self.add_bindings('cmd-help-buffer', ('M-?',))
|
|
self.add_bindings('set-mode', ('C-x m',))
|
|
self.add_bindings('cancel', ('C-]', 'C-g',))
|
|
self.add_bindings('exec', ('C-c e', 'C-c !'))
|
|
self.add_bindings('grep', ('C-c g',))
|
|
self.add_bindings('pipe', ('C-c p', 'C-c |'))
|
|
self.add_bindings('view-buffer-parent', ('C-c .',))
|
|
self.add_bindings('insert-squotes', ('M-\'',))
|
|
self.add_bindings('insert-dquotes', ('M-"',))
|
|
self.add_bindings('get-token', ('C-c t',))
|
|
self.add_bindings('insert-text', ('C-c i',))
|
|
self.add_bindings('insert-text2', ('C-c M-i',))
|
|
self.add_bindings('insert-multiline-text', ('C-c m',))
|
|
self.add_bindings('increment', ('M-+',))
|
|
self.add_bindings('decrement', ('M--',))
|
|
|
|
self.add_bindings('uppercase-word', ('M-u',))
|
|
self.add_bindings('lowercase-word', ('M-l',))
|
|
|
|
# used for all word operations
|
|
if not self.word_letters:
|
|
self.word_letters = w.application.config['word_letters']
|
|
|
|
# create all the insert actions for the basic text input
|
|
for c in string.letters + string.digits + string.punctuation:
|
|
self.add_binding('insert-string-%s' % c, c)
|
|
|
|
# per-mode bindings
|
|
for (name, sequences) in self._bindings.iteritems():
|
|
self.add_bindings(name, sequences)
|
|
|
|
# lexing for highlighting, etc.
|
|
if self.grammar:
|
|
self.lexer = Lexer(self, self.grammar)
|
|
else:
|
|
self.lexer = None
|
|
self.gstack = {}
|
|
self.ghist = {}
|
|
|
|
# tab handling
|
|
if self.tabbercls:
|
|
self.tabber = self.tabbercls(self)
|
|
|
|
# buffer settings
|
|
def get_setting(self, name):
|
|
self.window.buffer.settings.setdefault(self.modename, {})
|
|
return self.window.buffer.settings[self.modename].get(name)
|
|
def init_setting(self, name, value):
|
|
self.window.buffer.settings.setdefault(self.modename, {})
|
|
self.window.buffer.settings[self.modename].setdefault(name, value)
|
|
def set_setting(self, name, value):
|
|
self.window.buffer.settings.setdefault(self.modename, {})
|
|
self.window.buffer.settings[self.modename][name] = value
|
|
|
|
# header
|
|
def showing_header(self):
|
|
return self.header != 0
|
|
def _enable_header(self):
|
|
self.header = self.header_size
|
|
def _disable_header(self):
|
|
self.header = 0
|
|
|
|
def enable_header(self):
|
|
self._enable_header()
|
|
self._slot_reset()
|
|
def disable_header(self):
|
|
self._disable_header()
|
|
self._slot_reset()
|
|
def _slot_reset(self):
|
|
w = self.window
|
|
slot = w.application.bufferlist.find_window_slot(w)
|
|
assert slot is not None
|
|
slot.reset()
|
|
|
|
# line numbers
|
|
def showing_line_numbers(self):
|
|
return self.lmargin != 0
|
|
def enable_line_numbers(self):
|
|
l = len(self.window.buffer.lines)
|
|
self.lmargin = int(math.log(l, 10)) + 3
|
|
def disable_line_numbers(self):
|
|
self.lmargin = 0
|
|
|
|
# headers and margins
|
|
def get_header(self):
|
|
fg, bg = self.header_fg, self.header_bg
|
|
|
|
if self.tabber is None:
|
|
s = "Header support is not available for this mode"
|
|
hs = [[RenderString(s=s, attrs=color.build(fg, bg))]]
|
|
while len(hs) < 3:
|
|
hs.insert(0, [RenderString(s='', attrs=color.build(fg, bg))])
|
|
return hs
|
|
|
|
w = self.window
|
|
y = self.window.first.y
|
|
if self.window.first.x > 0:
|
|
y += 1
|
|
lvl = self.tabber.get_level(y)
|
|
markers = self.tabber.record[y]
|
|
if w.buffer.is_whitespace(y):
|
|
ws = None
|
|
else:
|
|
ws = w.buffer.count_leading_whitespace(y)
|
|
|
|
hs = []
|
|
i = len(markers) - 1
|
|
while i >= 0 and len(hs) < 3:
|
|
marker = markers[i]
|
|
i -= 1
|
|
if marker.y == y:
|
|
continue
|
|
if ws and marker.level > ws:
|
|
continue
|
|
s = w.buffer.lines[marker.y][:w.width - 1]
|
|
hs.insert(0, [RenderString(s=s, attrs=color.build(fg, bg))])
|
|
while len(hs) < 3:
|
|
hs.insert(0, [RenderString(s='', attrs=color.build(fg, bg))])
|
|
return hs
|
|
|
|
def get_footer(self):
|
|
fg, bg = "default", "red"
|
|
return [[RenderString(s='footer', attrs=color.build(fg, bg))]]
|
|
def get_rmargin(self, w, y, x, ended=False, cont=False):
|
|
c, fg, bg = " ", "red", "default"
|
|
if cont:
|
|
c = "\\"
|
|
return [RenderString(s=c, attrs=color.build(fg, bg))]
|
|
def get_lmargin(self, w, y, x, ended=False, cont=False):
|
|
lm = self.lmargin
|
|
if ended:
|
|
i = int(math.log(y, 10)) + 1
|
|
s = ('% *s' % (lm - 1, '-' * i))[-lm:] + ' '
|
|
elif not cont:
|
|
s = ('% *d' % (lm - 1, y + 1))[-lm:] + ' '
|
|
else:
|
|
s = ' ' * lm
|
|
return [RenderString(s=s, attrs=color.build('default', 'default', 'bold'))]
|
|
|
|
# get mode name
|
|
def name(self):
|
|
return self.modename
|
|
|
|
def _get_flag(self):
|
|
b = self.window.buffer
|
|
if b.readonly():
|
|
if b.changed():
|
|
return '%*'
|
|
else:
|
|
return '%%'
|
|
else:
|
|
if b.changed():
|
|
return '**'
|
|
else:
|
|
return '--'
|
|
def _get_perc(self):
|
|
w = self.window
|
|
if w.first_is_visible():
|
|
return "Top"
|
|
elif w.last_is_visible():
|
|
return "Bot"
|
|
else:
|
|
return "%2d%%" % (w.first.y*100 / len(w.buffer.lines))
|
|
def _get_indent(self):
|
|
b = self.window.buffer
|
|
if b.writetabs:
|
|
t = 't'
|
|
else:
|
|
t = 's'
|
|
return '%d%s' % (b.indentlvl, t)
|
|
def _get_mark(self):
|
|
w = self.window
|
|
if w.mark:
|
|
return '(%d,%d)' % (w.mark.y + 1, w.mark.x + 1)
|
|
else:
|
|
return '(None)'
|
|
def get_status_names(self):
|
|
w = self.window
|
|
c = w.logical_cursor()
|
|
d = defaultdict(str)
|
|
d2 = {
|
|
'bname': w.buffer.name(),
|
|
'mname': self.name(),
|
|
'flag': self._get_flag(),
|
|
'perc': self._get_perc(),
|
|
'indent': self._get_indent(),
|
|
'cursor': '(%d,%d)' % (c.y + 1, c.x + 1),
|
|
'first': '(%d,%d)' % (w.first.y + 1, w.first.x + 1),
|
|
'mark': self._get_mark(),
|
|
}
|
|
d.update(d2)
|
|
d.update(dict(w.buffer.metadata))
|
|
return d
|
|
def get_status_bar(self):
|
|
names = self.get_status_names()
|
|
return self.format % names
|
|
|
|
# handle input tokens
|
|
def handle_token(self, t):
|
|
'''self.handle_token(token): handles input "token"'''
|
|
self.window.active_point = None
|
|
self.window.clear_error()
|
|
try:
|
|
act = Handler.handle_token(self, t)
|
|
if act is None:
|
|
self.window.set_error(' '.join(self.curr_tokens))
|
|
return
|
|
else:
|
|
act.execute(self.window)
|
|
self.window.application.last_action = act.name
|
|
except ActionError, e:
|
|
#XYZ: HACK--should fix
|
|
if t in ('C-]', 'C-g'):
|
|
self.window.set_error('Cancelled')
|
|
else:
|
|
self.window.set_error(str(e))
|
|
except Exception, e:
|
|
if DEBUG:
|
|
raise
|
|
else:
|
|
err = "%s in mode '%s'" % (e, self.name())
|
|
self.window.set_error(err)
|
|
|
|
def region_added(self, p, newlines):
|
|
mname = self.name()
|
|
if self.lexer is not None:
|
|
ydelta = len(newlines) - 1
|
|
xdelta = len(newlines[-1])
|
|
ghist = {}
|
|
for name in self.ghist:
|
|
for gp in self.ghist[name]:
|
|
if gp < p:
|
|
newp = gp
|
|
elif ydelta == 0:
|
|
if p.y == gp.y:
|
|
newp = Point(gp.x + xdelta, gp.y)
|
|
else:
|
|
newp = gp
|
|
else:
|
|
if gp.y == p.y:
|
|
newp = Point(gp.x + xdelta, gp.y + ydelta)
|
|
else:
|
|
newp = Point(gp.x, gp.y + ydelta)
|
|
ghist.setdefault(name, {})
|
|
ghist[name][newp] = self.ghist[name][gp]
|
|
self.ghist = ghist
|
|
if self.tabber is not None:
|
|
self.tabber.region_added(p, newlines)
|
|
if self.context is not None:
|
|
self.context.region_added(p, newlines)
|
|
if self.lmargin != 0:
|
|
self.enable_line_numbers()
|
|
|
|
def region_removed(self, p1, p2):
|
|
if self.lexer is not None:
|
|
ydelta = p2.y - p1.y
|
|
xdelta = p2.x - p1.x
|
|
ghist = {}
|
|
for name in self.ghist:
|
|
for gp in self.ghist[name]:
|
|
if gp < p1:
|
|
newp = gp
|
|
elif p1 <= gp and gp < p2:
|
|
continue
|
|
elif ydelta == 0:
|
|
if gp.y == p2.y:
|
|
newp = Point(gp.x - xdelta, gp.y)
|
|
else:
|
|
newp = gp
|
|
else:
|
|
if gp.y == p2.y:
|
|
newp = Point(gp.x - xdelta, gp.y - ydelta)
|
|
else:
|
|
newp = Point(gp.x, gp.y - ydelta)
|
|
ghist.setdefault(name, {})
|
|
ghist[name][newp] = self.ghist[name][gp]
|
|
self.ghist = ghist
|
|
if self.tabber is not None:
|
|
self.tabber.region_removed(p1, p2)
|
|
if self.context is not None:
|
|
self.context.region_removed(p1, p2)
|
|
if self.lmargin != 0:
|
|
self.enable_line_numbers()
|
|
|
|
install = Fundamental.install
|