fasttyper/fasttyper/components.py

279 lines
7.9 KiB
Python

import curses
class WindowComponent:
"""
Basic text component printing inside window
"""
def __init__(self, config):
self._window = None
self._height = None
self._width = None
self._begin_x = None
self._begin_y = None
self.cursor_x, self.cursor_y = 0, 0
def update_size(self, height, width, begin_x, begin_y):
self._height = height
self._width = width
self._begin_x = begin_x
self._begin_y = begin_y
def init_window(self):
self._window = curses.newwin(
self._height,
self._width,
self._begin_y,
self._begin_x,
)
def set_box(self, i):
self._window.box(i, i)
def paint_text(self, row, col, text, color):
self._window.addstr(row, col, text, curses.color_pair(color))
def move(self, x, y):
self._window.move(x, y)
def paint(self, screen, application):
pass
def refresh(self):
self._window.refresh()
def update_cursor(self):
self.cursor_x = self.current_line
past_words = self.lines[self.current_line][: self.word_index]
self.cursor_y = (
sum([len(w) for w in past_words])
+ len(past_words)
+ self.buffer.current_char
)
class BorderedBox(WindowComponent):
"""
Adds border to WindowComponent
"""
def __init__(self, config):
super().__init__(config)
self.maxy, self.maxx = None, None
self.pos_y = config.get("top_margin_percentage") / 100
self.pos_x = config.get("left_margin_percentage") / 100
self.height = config.get("lines_on_screen")
self.width = None
def paint_text(self, row, col, text, color):
super().paint_text(row + 1, col + 1, text, color)
def move(self, x, y):
super().move(x + 1, y + 1)
def init(self, screen, application):
self.width = int(self.maxx * (1 - 2 * self.pos_x))
self.update_size(
self.height + 2,
self.width + 2,
int(self.pos_x * self.maxx) - 1,
int(self.pos_y * self.maxy) - 1,
)
self.init_window()
self.set_box(1)
screen.refresh()
class BufferDependentComponent(BorderedBox):
"""
Adds content source from buffer
"""
def __init__(self, config):
super().__init__(config)
self.buffered_lines = 0
self.buffer = None
self.last_hidden_word = 0
self.lines = [[] for _ in range(self.height)]
self.should_repaint = [False for _ in range(self.height)]
from .buffer import CharType
self.chtype_mapper = {
CharType.valid: config.get("user_input_valid_color"),
CharType.invalid: config.get("user_input_invalid_color"),
CharType.reference: config.get("reference_text_color"),
}
self.current_word_idx = 0
self.current_line, self.word_index = 0, 0
def set_buffer(self, buffer):
self.buffer = buffer
def line_len(self, index):
spaces = max(len(self.lines[index]) - 1, 0)
return sum([len(w) for w in self.lines[index]]) + spaces
def paint_line(self, line_nr):
self.move(line_nr, 0)
self._window.clrtoeol()
pos = 0
for i, word in enumerate(self.lines[line_nr]):
for c in word:
self.paint_text(line_nr, pos, c[0], self.chtype_mapper[c[1]])
pos += 1
if i != len(self.lines[line_nr]):
self.paint_text(line_nr, pos, " ", 0)
pos += 1
def fill_lines(self):
line_nr = self.buffered_lines
start_idx = self.last_hidden_word + sum(
[len(line) for line in self.lines[:line_nr]]
)
for i in range(start_idx, self.buffer.total_words):
word = self.buffer.get_word(i)
if len(word) > self.width:
word = word[:width]
if self.line_len(line_nr) + len(word) + 1 > self.width:
self.paint_line(line_nr)
line_nr += 1
if line_nr >= self.height:
break
self.lines[line_nr].append(word)
self.should_repaint[line_nr] = True
self.buffered_lines = self.height
def reorganize_words(self, line_nr=None, move_active=False):
move_active = move_active or line_nr is None
line_nr = line_nr or self.current_line
self.should_repaint[line_nr] = True
if line_nr + 1 < self.height:
self.should_repaint[line_nr + 1] = True
should_organize_next = False
while self.line_len(line_nr) > self.width:
last_word = self.lines[line_nr][-1]
self.lines[line_nr] = self.lines[line_nr][:-1]
if line_nr + 1 < self.height:
self.lines[line_nr + 1] = [last_word] + self.lines[line_nr + 1]
if self.line_len(line_nr + 1) > self.width:
should_organize_next = True
if move_active and self.word_index >= len(self.lines[self.current_line]):
self.word_index = 0
self.current_line += 1
if should_organize_next:
self.reorganize_words(line_nr + 1, move_active)
def lines_to_paint(self):
result = {self.current_line}
if any(self.should_repaint):
result = {
i
for i in range(self.height)
if i == self.current_line or self.should_repaint[i]
}
self.should_repaint = [False for _ in range(self.height)]
return result
def update_current_word(self, word_index):
"""
This is called by buffer. It signals that user changed its state.
First, active word is updated. It changes on added char or deleted word.
"""
old_len = len(self.lines[self.current_line][self.word_index])
self.lines[self.current_line][self.word_index] = self.buffer.get_word(
self.current_word_idx
)
new_len = len(self.lines[self.current_line][self.word_index])
if old_len != new_len and self.line_len(self.current_line) > self.width:
self.reorganize_words()
while word_index > self.current_word_idx:
self.word_index += 1
if self.word_index >= len(self.lines[self.current_line]):
self.word_index = 0
self.current_line += 1
self.current_word_idx += 1
while word_index < self.current_word_idx:
self.word_index -= 1
if self.word_index == -1:
self.current_line -= 1
self.word_index = len(self.lines[self.current_line] - 1)
self.current_word_idx -= 1
self.update_cursor()
class BorderWithImprintedStats(BufferDependentComponent):
"""
Imprints stats on one of borders
"""
def __init__(self, config):
super().__init__(config)
self.stats_template = config.get("stats_template").replace("\n", " ")
self.stats_color = config.get("stats_color")
self.stats_row = -1 if config.get("stats_position") == "top" else self.height
def paint_stats(self):
text = self.stats_template.format(stats=self.buffer.stats)
if len(text) < self.width - 2:
self.paint_text(self.stats_row, 2, text, self.stats_color)
class TextBox(BorderWithImprintedStats):
"""
Calls all inherited paint functions and inits windows
"""
def __init__(self, config):
super().__init__(config)
def paint(self, screen, application):
if self._window is None:
self.maxy, self.maxx = screen.getmaxyx()
self.init(screen, application)
if self.buffered_lines < self.height:
self.fill_lines()
self.paint_stats()
for line in self.lines_to_paint():
self.paint_line(line)
self.move(self.cursor_x, self.cursor_y)
self.refresh()