diff --git a/Python/bed.py b/Python/bed.py new file mode 100755 index 0000000..1fe27c3 --- /dev/null +++ b/Python/bed.py @@ -0,0 +1,227 @@ +#!/usr/bin/env python3 +import sys +from btui import open_btui +from collections import namedtuple + +Vec2 = namedtuple('Vec2', 'x y') + +def clamp(x, low, high): + if x < low: return low + elif x > high: return high + return x + +class BedFile: + def __init__(self, bt, lines, pos=Vec2(0,0), size=None): + self.bt = bt + self.lines = lines + self.drawn = set() + self.numlen = len(str(len(self.lines))) + self.pos = pos + self.size = size if size else Vec2(self.bt.width-pos.x, self.bt.height-pos.y) + self.scroll = 0 + self._cursor = Vec2(0, 0) + self.unsaved = False + + @property + def cursor(self): + return self._cursor + + @cursor.setter + def cursor(self, p): + x, y = p.x, p.y + y = clamp(y, 0, len(self.lines)-1) + x = clamp(x, 0, len(self.lines[y])) + self._cursor = Vec2(x, y) + + def update_term_cursor(self): + self.bt.move(self.pos.x + self.numlen + 1 + self.cursor.x, + self.pos.y + self.cursor.y - self.scroll) + + def set_scroll(self, newscroll): + newscroll = clamp(newscroll, 0, len(self.lines)-1 - (self.size.y-1)) + if newscroll == self.scroll: return + delta = newscroll - self.scroll + self.bt.scroll(self.pos.y, self.pos.y + self.size.y - 1, delta) + self.scroll = newscroll + self.drawn &= set(range(self.scroll, self.scroll + (self.size.y-1)+1)) + if self.cursor.y < newscroll: + self.cursor = Vec2(self.cursor.x, newscroll) + elif self.cursor.y > newscroll + (self.size.y-1): + self.cursor = Vec2(self.cursor.x, newscroll + (self.size.y-1)) + + def handle_input(self, key, mx=None, my=None): + if key == 'Space': key = ' ' + if key == 'Tab': key = ' ' + + if key == 'Left': + self.cursor = Vec2(self.cursor.x - 1, self.cursor.y) + elif key == 'Right': + self.cursor = Vec2(self.cursor.x + 1, self.cursor.y) + elif key == 'Up': + self.cursor = Vec2(self.cursor.x, self.cursor.y - 1) + if self.cursor.y < self.scroll: + self.set_scroll(self.cursor.y) + elif key == 'Down': + self.cursor = Vec2(self.cursor.x, self.cursor.y + 1) + newscroll = clamp(self.scroll, self.cursor.y - (self.size.y-1), self.cursor.y) + if self.cursor.y > self.scroll + (self.size.y-1): + self.set_scroll(self.cursor.y - (self.size.y-1)) + elif key == 'Page Down' or key == 'Ctrl-d': + self.set_scroll(self.scroll + self.bt.height // 2) + elif key == 'Page Up' or key == 'Ctrl-u': + self.set_scroll(self.scroll - self.bt.height // 2) + elif key == 'Delete': + line = self.lines[self.cursor.y] + i = self.cursor.x + if i == len(line) and self.cursor.y < len(self.lines)-1: + line.extend(self.lines.pop(self.cursor.y+1)) + self.drawn &= {d for d in self.drawn if d < self.cursor.y or d >= len(self.lines)+1} + self.unsaved = True + elif i < len(line)-1: + del line[i:i+1] + self.drawn.discard(self.cursor.y) + self.unsaved = True + elif key == 'Backspace': + line = self.lines[self.cursor.y] + i = self.cursor.x + if i == 0 and self.cursor.y > 0: + self.lines.pop(self.cursor.y) + prevline = self.lines[self.cursor.y - 1] + self.cursor = Vec2(len(prevline), self.cursor.y-1) + prevline.extend(line) + self.drawn &= {d for d in self.drawn if d < self.cursor.y or d >= len(self.lines)+1} + self.unsaved = True + elif i > 0: + del line[i-1:i] + self.drawn.discard(self.cursor.y) + self.cursor = Vec2(self.cursor.x - 1, self.cursor.y) + self.unsaved = True + elif key == 'Enter': + line = self.lines[self.cursor.y] + before, after = line[:self.cursor.x], line[self.cursor.x:] + self.lines[self.cursor.y] = before + self.lines.insert(self.cursor.y+1, after) + self.drawn &= {d for d in self.drawn if d < self.cursor.y or d >= len(self.lines)} + self.cursor = Vec2(0, self.cursor.y + 1) + self.unsaved = True + elif key and (len(key) == 1 or key == ' '): + line = self.lines[self.cursor.y] + i = self.cursor.x + line[i:i] = bytearray(key, 'utf8') + self.cursor = Vec2(self.cursor.x + len(key), self.cursor.y) + self.drawn.discard(self.cursor.y) + self.unsaved = True + + def render(self): + self.bt.hide_cursor() + for i in range(self.size.y): + lineno = self.scroll + i + if lineno in self.drawn: continue + y = self.pos.y + i + x = self.pos.x + self.bt.move(x, y) + if lineno >= len(self.lines): + self.bt.clear('line') + self.drawn.add(lineno) + continue + with self.bt.attributes("faint"): + self.bt.write(("{:>"+str(self.numlen)+"}").format(lineno + 1)) + x += self.numlen + 1 + self.bt.move(x, y) + self.bt.write_bytes(bytes(self.lines[lineno])) + self.bt.clear('right') + self.drawn.add(lineno) + +class BED: + def __init__(self, bt, filename): + self.bt = bt + self.filename = filename + try: + lines = [bytearray(line.rstrip(b'\n')) for line in open(filename, 'rb').readlines()] + except FileNotFoundError: + lines = [bytearray(b'')] + self.file = BedFile(bt, lines, Vec2(0, 1), Vec2(bt.width, bt.height-2)) + self.drawn = set() + self.renders = 0 + + def edit(self): + self.bt.set_cursor('blinking bar') + self.render() + key, mx, my = self.bt.getkey() + while True: + if key == 'Ctrl-c': + sys.exit(1) + elif key == 'Ctrl-q': + if self.file.unsaved: + self.confirm_save() + break + elif key == 'Ctrl-s': + self.save() + else: + self.file.handle_input(key, mx, my) + self.render() + key, mx, my = self.bt.getkey() + + def confirm_save(self): + self.bt.move(0, self.bt.height) + with self.bt.attributes('bold'): + self.bt.write(f"Do you want to save {self.filename}? [y/n]") + key, mx, my = self.bt.getkey() + if key in ('y', 'Y'): + self.save() + return True + elif key in ('n', 'N', 'Ctrl-c', 'Escape', 'Ctrl-q'): + return False + + def save(self): + with open(self.filename, 'wb') as f: + for line in self.file.lines: + f.write(line) + f.write(b'\n') + self.file.unsaved = False + + def render(self): + self.renders += 1 + + k = 'title-modified' if self.file.unsaved else 'title' + if k not in self.drawn: + self.drawn -= {'title', 'title-modified'} + self.bt.move(0,0) + with self.bt.attributes("bold"): + self.bt.write(self.filename) + if self.file.unsaved: + with self.bt.attributes("dim"): + self.bt.write(" (modified)") + self.bt.clear('right') + self.drawn.add(k) + + self.file.render() + + # Just redraw this every time: + with self.bt.attributes("faint"): + self.bt.move(0, self.bt.height-1) + self.bt.write("Ctrl-Q to quit, Ctrl-S to save") + self.bt.clear('right') + s = f"Line {self.file.cursor.y}, Col {self.file.cursor.x}, {self.renders} redraws" + self.bt.move(self.bt.width-len(s), self.bt.height-1) + self.bt.write(s) + + self.file.update_term_cursor() + self.bt.show_cursor() + + +if __name__ == '__main__': + args = sys.argv[1:] + debug = False + if args and args[0] in ('-d', '--debug'): + args.pop(0) + debug = True + + if not args: + print("Usage: bed.py [-d|--debug] file") + sys.exit(1) + + filename = args[0] + with open_btui(debug=debug) as bt: + bed = BED(bt, filename) + bed.edit()