2023-03-03 06:02:00 +00:00
|
|
|
"""
|
2023-02-14 21:32:54 +00:00
|
|
|
Widget class definitions used by model_select.py, merge_diffusers.py and textual_inversion.py
|
2023-03-03 06:02:00 +00:00
|
|
|
"""
|
|
|
|
import curses
|
2023-02-14 21:32:54 +00:00
|
|
|
import math
|
2023-02-23 05:43:25 +00:00
|
|
|
import os
|
2023-03-03 06:02:00 +00:00
|
|
|
import platform
|
2023-02-23 00:18:07 +00:00
|
|
|
import struct
|
2023-03-03 06:02:00 +00:00
|
|
|
import sys
|
2023-02-23 00:18:07 +00:00
|
|
|
from shutil import get_terminal_size
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
import npyscreen
|
|
|
|
|
|
|
|
|
2023-02-23 00:18:07 +00:00
|
|
|
# -------------------------------------
|
|
|
|
def set_terminal_size(columns: int, lines: int):
|
2023-02-23 05:43:25 +00:00
|
|
|
OS = platform.uname().system
|
2023-03-03 06:02:00 +00:00
|
|
|
if OS == "Windows":
|
|
|
|
os.system(f"mode con: cols={columns} lines={lines}")
|
|
|
|
elif OS in ["Darwin", "Linux"]:
|
2023-02-23 05:43:25 +00:00
|
|
|
import fcntl
|
2023-03-03 06:02:00 +00:00
|
|
|
import termios
|
|
|
|
|
2023-02-23 00:18:07 +00:00
|
|
|
winsize = struct.pack("HHHH", lines, columns, 0, 0)
|
2023-02-23 05:43:25 +00:00
|
|
|
fcntl.ioctl(sys.stdout.fileno(), termios.TIOCSWINSZ, winsize)
|
2023-02-23 00:18:07 +00:00
|
|
|
sys.stdout.write("\x1b[8;{rows};{cols}t".format(rows=lines, cols=columns))
|
|
|
|
sys.stdout.flush()
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-23 00:18:07 +00:00
|
|
|
def set_min_terminal_size(min_cols: int, min_lines: int):
|
|
|
|
# make sure there's enough room for the ui
|
|
|
|
term_cols, term_lines = get_terminal_size()
|
2023-03-03 06:02:00 +00:00
|
|
|
cols = max(term_cols, min_cols)
|
2023-02-23 00:18:07 +00:00
|
|
|
lines = max(term_lines, min_lines)
|
2023-03-03 06:02:00 +00:00
|
|
|
set_terminal_size(cols, lines)
|
|
|
|
|
2023-02-14 21:32:54 +00:00
|
|
|
|
2023-02-17 19:34:48 +00:00
|
|
|
class IntSlider(npyscreen.Slider):
|
|
|
|
def translate_value(self):
|
|
|
|
stri = "%2d / %2d" % (self.value, self.out_of)
|
|
|
|
l = (len(str(self.out_of))) * 2 + 4
|
|
|
|
stri = stri.rjust(l)
|
|
|
|
return stri
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-22 02:33:44 +00:00
|
|
|
# -------------------------------------
|
|
|
|
class CenteredTitleText(npyscreen.TitleText):
|
2023-03-03 06:02:00 +00:00
|
|
|
def __init__(self, *args, **keywords):
|
|
|
|
super().__init__(*args, **keywords)
|
2023-02-22 02:33:44 +00:00
|
|
|
self.resize()
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-22 02:33:44 +00:00
|
|
|
def resize(self):
|
|
|
|
super().resize()
|
|
|
|
maxy, maxx = self.parent.curses_pad.getmaxyx()
|
|
|
|
label = self.name
|
|
|
|
self.relx = (maxx - len(label)) // 2
|
2023-03-03 06:02:00 +00:00
|
|
|
|
|
|
|
|
2023-02-22 02:33:44 +00:00
|
|
|
# -------------------------------------
|
|
|
|
class CenteredButtonPress(npyscreen.ButtonPress):
|
|
|
|
def resize(self):
|
|
|
|
super().resize()
|
|
|
|
maxy, maxx = self.parent.curses_pad.getmaxyx()
|
|
|
|
label = self.name
|
|
|
|
self.relx = (maxx - len(label)) // 2
|
2023-03-03 06:02:00 +00:00
|
|
|
|
|
|
|
|
2023-02-22 02:33:44 +00:00
|
|
|
# -------------------------------------
|
|
|
|
class OffsetButtonPress(npyscreen.ButtonPress):
|
2023-03-03 06:02:00 +00:00
|
|
|
def __init__(self, screen, offset=0, *args, **keywords):
|
2023-02-22 02:33:44 +00:00
|
|
|
super().__init__(screen, *args, **keywords)
|
|
|
|
self.offset = offset
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-22 02:33:44 +00:00
|
|
|
def resize(self):
|
|
|
|
maxy, maxx = self.parent.curses_pad.getmaxyx()
|
|
|
|
width = len(self.name)
|
|
|
|
self.relx = self.offset + (maxx - width) // 2
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-17 19:34:48 +00:00
|
|
|
class IntTitleSlider(npyscreen.TitleText):
|
|
|
|
_entry_type = IntSlider
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-14 21:32:54 +00:00
|
|
|
class FloatSlider(npyscreen.Slider):
|
|
|
|
# this is supposed to adjust display precision, but doesn't
|
|
|
|
def translate_value(self):
|
|
|
|
stri = "%3.2f / %3.2f" % (self.value, self.out_of)
|
|
|
|
l = (len(str(self.out_of))) * 2 + 4
|
|
|
|
stri = stri.rjust(l)
|
|
|
|
return stri
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-14 21:32:54 +00:00
|
|
|
class FloatTitleSlider(npyscreen.TitleText):
|
|
|
|
_entry_type = FloatSlider
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-06-01 04:31:46 +00:00
|
|
|
class SelectColumnBase():
|
2023-02-14 21:32:54 +00:00
|
|
|
def make_contained_widgets(self):
|
|
|
|
self._my_widgets = []
|
|
|
|
column_width = self.width // self.columns
|
|
|
|
for h in range(self.value_cnt):
|
|
|
|
self._my_widgets.append(
|
2023-03-03 06:02:00 +00:00
|
|
|
self._contained_widgets(
|
|
|
|
self.parent,
|
|
|
|
rely=self.rely + (h % self.rows) * self._contained_widget_height,
|
|
|
|
relx=self.relx + (h // self.rows) * column_width,
|
|
|
|
max_width=column_width,
|
|
|
|
max_height=self.__class__._contained_widget_height,
|
|
|
|
)
|
2023-02-14 21:32:54 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
def set_up_handlers(self):
|
|
|
|
super().set_up_handlers()
|
2023-03-03 06:02:00 +00:00
|
|
|
self.handlers.update(
|
|
|
|
{
|
|
|
|
curses.KEY_UP: self.h_cursor_line_left,
|
|
|
|
curses.KEY_DOWN: self.h_cursor_line_right,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2023-02-14 21:32:54 +00:00
|
|
|
def h_cursor_line_down(self, ch):
|
|
|
|
self.cursor_line += self.rows
|
|
|
|
if self.cursor_line >= len(self.values):
|
2023-03-03 06:02:00 +00:00
|
|
|
if self.scroll_exit:
|
|
|
|
self.cursor_line = len(self.values) - self.rows
|
2023-02-14 21:32:54 +00:00
|
|
|
self.h_exit_down(ch)
|
|
|
|
return True
|
2023-03-03 06:02:00 +00:00
|
|
|
else:
|
2023-02-14 21:32:54 +00:00
|
|
|
self.cursor_line -= self.rows
|
|
|
|
return True
|
|
|
|
|
|
|
|
def h_cursor_line_up(self, ch):
|
|
|
|
self.cursor_line -= self.rows
|
2023-03-03 06:02:00 +00:00
|
|
|
if self.cursor_line < 0:
|
2023-02-14 21:32:54 +00:00
|
|
|
if self.scroll_exit:
|
|
|
|
self.cursor_line = 0
|
|
|
|
self.h_exit_up(ch)
|
2023-03-03 06:02:00 +00:00
|
|
|
else:
|
2023-02-14 21:32:54 +00:00
|
|
|
self.cursor_line = 0
|
|
|
|
|
2023-03-03 06:02:00 +00:00
|
|
|
def h_cursor_line_left(self, ch):
|
2023-02-14 21:32:54 +00:00
|
|
|
super().h_cursor_line_up(ch)
|
2023-03-03 06:02:00 +00:00
|
|
|
|
|
|
|
def h_cursor_line_right(self, ch):
|
2023-02-14 21:32:54 +00:00
|
|
|
super().h_cursor_line_down(ch)
|
2023-02-15 06:07:39 +00:00
|
|
|
|
2023-06-01 04:31:46 +00:00
|
|
|
class MultiSelectColumns( SelectColumnBase, npyscreen.MultiSelect):
|
|
|
|
def __init__(self, screen, columns: int = 1, values: list = [], **keywords):
|
|
|
|
self.columns = columns
|
|
|
|
self.value_cnt = len(values)
|
|
|
|
self.rows = math.ceil(self.value_cnt / self.columns)
|
|
|
|
super().__init__(screen, values=values, **keywords)
|
|
|
|
|
|
|
|
|
|
|
|
class SingleSelectColumns(SelectColumnBase, npyscreen.SelectOne):
|
|
|
|
def __init__(self, screen, columns: int = 1, values: list = [], **keywords):
|
|
|
|
self.columns = columns
|
|
|
|
self.value_cnt = len(values)
|
|
|
|
self.rows = math.ceil(self.value_cnt / self.columns)
|
|
|
|
self.on_changed = None
|
|
|
|
super().__init__(screen, values=values, **keywords)
|
|
|
|
|
|
|
|
def h_select(self,ch):
|
|
|
|
super().h_select(ch)
|
|
|
|
if self.on_changed:
|
|
|
|
self.on_changed(self.value)
|
|
|
|
|
|
|
|
def when_value_edited(self):
|
|
|
|
self.h_select(self.cursor_line)
|
|
|
|
|
|
|
|
def when_cursor_moved(self):
|
|
|
|
self.h_select(self.cursor_line)
|
2023-03-03 06:02:00 +00:00
|
|
|
|
2023-02-15 06:07:39 +00:00
|
|
|
class TextBox(npyscreen.MultiLineEdit):
|
|
|
|
def update(self, clear=True):
|
2023-03-03 06:02:00 +00:00
|
|
|
if clear:
|
|
|
|
self.clear()
|
2023-02-15 06:07:39 +00:00
|
|
|
|
|
|
|
HEIGHT = self.height
|
2023-03-03 06:02:00 +00:00
|
|
|
WIDTH = self.width
|
2023-02-15 06:07:39 +00:00
|
|
|
# draw box.
|
|
|
|
self.parent.curses_pad.hline(self.rely, self.relx, curses.ACS_HLINE, WIDTH)
|
2023-03-03 06:02:00 +00:00
|
|
|
self.parent.curses_pad.hline(
|
|
|
|
self.rely + HEIGHT, self.relx, curses.ACS_HLINE, WIDTH
|
|
|
|
)
|
|
|
|
self.parent.curses_pad.vline(
|
|
|
|
self.rely, self.relx, curses.ACS_VLINE, self.height
|
|
|
|
)
|
|
|
|
self.parent.curses_pad.vline(
|
|
|
|
self.rely, self.relx + WIDTH, curses.ACS_VLINE, HEIGHT
|
|
|
|
)
|
|
|
|
|
2023-02-15 06:07:39 +00:00
|
|
|
# draw corners
|
2023-03-03 06:02:00 +00:00
|
|
|
self.parent.curses_pad.addch(
|
|
|
|
self.rely,
|
|
|
|
self.relx,
|
|
|
|
curses.ACS_ULCORNER,
|
|
|
|
)
|
|
|
|
self.parent.curses_pad.addch(
|
|
|
|
self.rely,
|
|
|
|
self.relx + WIDTH,
|
|
|
|
curses.ACS_URCORNER,
|
|
|
|
)
|
|
|
|
self.parent.curses_pad.addch(
|
|
|
|
self.rely + HEIGHT,
|
|
|
|
self.relx,
|
|
|
|
curses.ACS_LLCORNER,
|
|
|
|
)
|
|
|
|
self.parent.curses_pad.addch(
|
|
|
|
self.rely + HEIGHT,
|
|
|
|
self.relx + WIDTH,
|
|
|
|
curses.ACS_LRCORNER,
|
|
|
|
)
|
|
|
|
|
2023-02-15 06:07:39 +00:00
|
|
|
# fool our superclass into thinking drawing area is smaller - this is really hacky but it seems to work
|
2023-03-03 06:02:00 +00:00
|
|
|
(relx, rely, height, width) = (self.relx, self.rely, self.height, self.width)
|
2023-02-15 06:07:39 +00:00
|
|
|
self.relx += 1
|
|
|
|
self.rely += 1
|
|
|
|
self.height -= 1
|
|
|
|
self.width -= 1
|
|
|
|
super().update(clear=False)
|
2023-03-03 06:02:00 +00:00
|
|
|
(self.relx, self.rely, self.height, self.width) = (relx, rely, height, width)
|