Add filtered_rows property

This commit is contained in:
Digital Studium 2024-04-14 15:54:53 +03:00
parent d8443c47d4
commit 592ca0ac37
1 changed files with 17 additions and 26 deletions

43
kls
View File

@ -17,11 +17,14 @@ class Menu:
self.state = state # состояние меню self.state = state # состояние меню
self.name = name # заголовок меню self.name = name # заголовок меню
self.rows = rows # строки меню self.rows = rows # строки меню
self.filter = "" # фильтр строк меню
self.selected_row = 0 # выбранная строка меню self.selected_row = 0 # выбранная строка меню
self.begin_x = begin_x # где начинается меню по х? self.begin_x = begin_x # где начинается меню по х?
self.win = curses.newwin(curses.LINES, width, 0, begin_x) # окно с высотой во весь экран, шириной width, и началом по х в точке begin_x self.win = curses.newwin(curses.LINES, width, 0, begin_x) # окно с высотой во весь экран, шириной width, и началом по х в точке begin_x
self.rows_number = curses.LINES - 10 # максимальное число видимых строк меню, начиная с 0 self.rows_number = curses.LINES - 10 # максимальное число видимых строк меню, начиная с 0
self.filter = "" # фильтр строк меню @property
def filtered_rows(self):
return list(filter(lambda x: (self.filter in x), self.rows)) # фильтрованные строки меню
def execute_cmd(command): def execute_cmd(command):
@ -42,45 +45,38 @@ menus = [
def update_menu3_object(): def update_menu3_object():
menu1_filtered_rows = list(filter(lambda x: (menus[0].filter in x), menus[0].rows)) # фильтруем строки if not menus[0].filtered_rows or not menus[1].filtered_rows:
menu2_filtered_rows = list(filter(lambda x: (menus[1].filter in x), menus[1].rows)) # фильтруем строки
if not menu1_filtered_rows or not menu2_filtered_rows:
menus[2].rows = ["No resources matched criteria.", ] menus[2].rows = ["No resources matched criteria.", ]
else: else:
namespace = menu1_filtered_rows[menus[0].selected_row] namespace = menus[0].filtered_rows[menus[0].selected_row]
api_resource = menu2_filtered_rows[menus[1].selected_row] api_resource = menus[1].filtered_rows[menus[1].selected_row]
menus[2].rows = execute_cmd(f"kubectl get {api_resource} --no-headers -n {namespace} | awk '{{print $1}}'") menus[2].rows = execute_cmd(f"kubectl get {api_resource} --no-headers -n {namespace} | awk '{{print $1}}'")
if not menus[2].rows: menus[2].rows = [f"No resources found in {namespace} namespace.", ] if not menus[2].rows: menus[2].rows = [f"No resources found in {namespace} namespace.", ]
menus[2].selected_row = 0 menus[2].selected_row = 0
def draw_header(menu): def draw_header(menu):
header_attr = curses.A_REVERSE | curses.A_BOLD if menu.state in [1, 2] else curses.A_NORMAL menu.win.addstr(1, 2, menu.name, curses.A_REVERSE | curses.A_BOLD if menu.state in [1, 2] else curses.A_NORMAL)
menu.win.addstr(1, 2, menu.name, header_attr)
menu.win.refresh() menu.win.refresh()
def draw_rows(menu): def draw_rows(menu):
filtered_rows = list(filter(lambda x: (menu.filter in x), menu.rows)) # какие строки сейчас в меню, учитывая фильтр? if not menu.filtered_rows: return # если строк нет, рисовать нечего
if not filtered_rows: return # если строк нет, рисовать нечего
# ограничиваем число отфильтрованных строк высотой окна + выбираем, от какой cтроки меню будет начинаться меню # ограничиваем число отфильтрованных строк высотой окна + выбираем, от какой cтроки меню будет начинаться меню
first_row_index = 0 if menu.selected_row < menu.rows_number else menu.selected_row - menu.rows_number + 1 first_row_index = 0 if menu.selected_row < menu.rows_number else menu.selected_row - menu.rows_number + 1
last_row_index = first_row_index + menu.rows_number last_row_index = first_row_index + menu.rows_number
filtered_rows = filtered_rows[first_row_index:last_row_index] filtered_rows = menu.filtered_rows[first_row_index:last_row_index]
selected_row_index = menu.selected_row - first_row_index # индекс выбранной строки в отфильтрованных строках selected_row_index = menu.selected_row - first_row_index # индекс выбранной строки в отфильтрованных строках
# if menus[1].selected_row != 0: # debug # if menus[1].selected_row != 0: # debug
# raise ValueError(f"{len(filtered_rows)} {selected_row_index} {first_row_index} {last_row_index} {menu.rows_number}") # raise ValueError(f"{len(filtered_rows)} {selected_row_index} {first_row_index} {last_row_index} {menu.rows_number}")
for index, row in enumerate(filtered_rows): # рисуем то, что отфильтровали for index, row in enumerate(filtered_rows): # рисуем то, что отфильтровали
menu.win.addstr(index + 3, 2, row) menu.win.addstr(index + 3, 2, row, curses.A_REVERSE | curses.A_BOLD if index == selected_row_index else curses.A_NORMAL)
menu.win.addstr(selected_row_index + 3, 2, filtered_rows[selected_row_index], curses.A_REVERSE | curses.A_BOLD) # выделяем выбранную строку
menu.win.box() menu.win.box()
menu.win.refresh() menu.win.refresh()
def draw_search_box(menu): def draw_search_box(menu):
search_attr = curses.A_REVERSE | curses.A_BOLD if menu.state in [SELECTED_WITH_SEARCH, NOT_SELECTED_WITH_SEARCH] else curses.A_NORMAL menu.win.addstr(curses.LINES - 2, 2, f"/{menu.filter}" if menu.state in [2, 4] else "Press / for search") # рисуем контент
content = f"/{menu.filter}" if menu.state in [SELECTED_WITH_SEARCH, NOT_SELECTED_WITH_SEARCH] else "Press / for search"
menu.win.addstr(curses.LINES - 2, 2, content, search_attr) # рисуем контент
menu.win.clrtoeol() # очищаем остальную часть строки menu.win.clrtoeol() # очищаем остальную часть строки
menu.win.box() # рисуем рамку menu.win.box() # рисуем рамку
menu.win.refresh() # обновляем окно menu.win.refresh() # обновляем окно
@ -99,13 +95,10 @@ def draw_window():
def run_command(key_pressed): def run_command(key_pressed):
menu3_filtered_rows = list(filter(lambda x: (menus[2].filter in x), menus[2].rows)) # фильтруем строки меню 3 if not menus[2].filtered_rows or menus[2].filtered_rows[0].startswith("No resources"): return # если пусто, выходим
if not menu3_filtered_rows or menu3_filtered_rows[0].startswith("No resources"): return # если пусто, выходим api_resource = menus[1].filtered_rows[menus[1].selected_row]
menu2_filtered_rows = list(filter(lambda x: (menus[1].filter in x), menus[1].rows)) # фильтруем строки меню 2
api_resource = menu2_filtered_rows[menus[1].selected_row]
if key_pressed == "KEY_F(4)" and api_resource != "pods": return # логи можно посмотреть только у подов if key_pressed == "KEY_F(4)" and api_resource != "pods": return # логи можно посмотреть только у подов
menu1_filtered_rows = list(filter(lambda x: (menus[0].filter in x), menus[0].rows)) # фильтруем строки меню 1 namespace = menus[0].filtered_rows[menus[0].selected_row]
namespace = menu1_filtered_rows[menus[0].selected_row]
resource = menus[2].rows[menus[2].selected_row] resource = menus[2].rows[menus[2].selected_row]
commands = { commands = {
"KEY_F(1)": f'kubectl -n {namespace} get {api_resource} {resource} -o yaml | batcat -l yaml --paging always --style numbers', "KEY_F(1)": f'kubectl -n {namespace} get {api_resource} {resource} -o yaml | batcat -l yaml --paging always --style numbers',
@ -114,7 +107,6 @@ def run_command(key_pressed):
"KEY_F(4)": f'kubectl -n {namespace} logs {resource} | batcat -l log --paging always --style numbers' "KEY_F(4)": f'kubectl -n {namespace} logs {resource} | batcat -l log --paging always --style numbers'
} }
curses.def_shell_mode() curses.def_shell_mode()
curses.endwin()
subprocess.call(commands[key_pressed], shell=True) subprocess.call(commands[key_pressed], shell=True)
curses.reset_shell_mode() curses.reset_shell_mode()
draw_window() draw_window()
@ -130,10 +122,9 @@ def navigate_horizontally(direction, menu):
def navigate_vertically(direction, menu): def navigate_vertically(direction, menu):
filtered_rows = list(filter(lambda x: (menu.filter in x), menu.rows)) # какие строки сейчас в меню, учитывая фильтр? if not menu.filtered_rows or len(menu.filtered_rows) == 1: return # если строк нет или строка одна, навигация не нужна
if not filtered_rows or len(filtered_rows) == 1: return # если строк нет или строка одна, навигация не нужна
increment = {"down": 1, "up": -1} increment = {"down": 1, "up": -1}
menu.selected_row = (menu.selected_row + increment[direction]) % len(filtered_rows) # выбираем строку учитывая сколько строк в меню menu.selected_row = (menu.selected_row + increment[direction]) % len(menu.filtered_rows) # выбираем строку учитывая сколько строк в меню
if menu != menus[2]: # перерисовываем третье меню, т. к. оно зависит от вертикальной навигации в меню 1 и 2 if menu != menus[2]: # перерисовываем третье меню, т. к. оно зависит от вертикальной навигации в меню 1 и 2
update_menu3_object() update_menu3_object()
draw_menu(menus[2]) draw_menu(menus[2])