mirror of
https://github.com/dw-0/kiauh.git
synced 2025-12-23 15:53:36 +05:00
Compare commits
1 Commits
0d7074f86f
...
78af5a1ae5
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
78af5a1ae5 |
@@ -17,10 +17,9 @@ from utils.constants import RESET_FORMAT, COLOR_RED, COLOR_CYAN
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
class KlipperRemoveMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
self.options = {
|
||||
"0": self.toggle_all,
|
||||
"1": self.toggle_remove_klipper_service,
|
||||
|
||||
@@ -32,14 +32,13 @@ from utils.logger import Logger
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class KlipperFlashMethodMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
self.options = {
|
||||
"1": self.select_regular,
|
||||
"2": self.select_sdcard,
|
||||
"h": lambda: KlipperFlashMethodHelpMenu(self).run(),
|
||||
"h": KlipperFlashMethodHelpMenu,
|
||||
}
|
||||
self.input_label_txt = "Select flash method"
|
||||
self.footer_type = FooterType.BACK_HELP
|
||||
@@ -76,7 +75,8 @@ class KlipperFlashMethodMenu(BaseMenu):
|
||||
self.goto_next_menu()
|
||||
|
||||
def goto_next_menu(self, **kwargs):
|
||||
KlipperFlashCommandMenu(previous_menu=self).run()
|
||||
next_menu = KlipperFlashCommandMenu(previous_menu=self)
|
||||
next_menu.run()
|
||||
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
@@ -84,15 +84,15 @@ class KlipperFlashMethodMenu(BaseMenu):
|
||||
class KlipperFlashCommandMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
self.options = {
|
||||
"1": self.select_flash,
|
||||
"2": self.select_serialflash,
|
||||
"h": lambda: KlipperFlashCommandHelpMenu(previous_menu=self).run(),
|
||||
"h": KlipperFlashCommandHelpMenu,
|
||||
}
|
||||
self.default_option = self.select_flash
|
||||
self.input_label_txt = "Select flash command"
|
||||
self.previous_menu = previous_menu
|
||||
self.footer_type = FooterType.BACK_HELP
|
||||
|
||||
self.flash_options = FlashOptions()
|
||||
@@ -119,7 +119,8 @@ class KlipperFlashCommandMenu(BaseMenu):
|
||||
self.goto_next_menu()
|
||||
|
||||
def goto_next_menu(self, **kwargs):
|
||||
KlipperSelectMcuConnectionMenu(previous_menu=self).run()
|
||||
next_menu = KlipperSelectMcuConnectionMenu(previous_menu=self)
|
||||
next_menu.run()
|
||||
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
@@ -127,15 +128,15 @@ class KlipperFlashCommandMenu(BaseMenu):
|
||||
class KlipperSelectMcuConnectionMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
self.options = {
|
||||
"1": self.select_usb,
|
||||
"2": self.select_dfu,
|
||||
"3": self.select_usb_dfu,
|
||||
"h": lambda: KlipperMcuConnectionHelpMenu(previous_menu=self).run(),
|
||||
"h": KlipperMcuConnectionHelpMenu,
|
||||
}
|
||||
self.input_label_txt = "Select connection type"
|
||||
self.previous_menu = previous_menu
|
||||
self.footer_type = FooterType.BACK_HELP
|
||||
|
||||
self.flash_options = FlashOptions()
|
||||
@@ -184,8 +185,6 @@ class KlipperSelectMcuConnectionMenu(BaseMenu):
|
||||
Logger.print_status("Identifying MCU connected via USB in DFU mode ...")
|
||||
self.flash_options.mcu_list = find_usb_dfu_device()
|
||||
|
||||
print(self.flash_options.mcu_list)
|
||||
|
||||
if len(self.flash_options.mcu_list) < 1:
|
||||
Logger.print_warn("No MCUs found!")
|
||||
Logger.print_warn("Make sure they are connected and repeat this step.")
|
||||
@@ -193,7 +192,8 @@ class KlipperSelectMcuConnectionMenu(BaseMenu):
|
||||
self.goto_next_menu()
|
||||
|
||||
def goto_next_menu(self, **kwargs):
|
||||
KlipperSelectMcuIdMenu(previous_menu=self).run()
|
||||
next_menu = KlipperSelectMcuIdMenu(previous_menu=self)
|
||||
next_menu.run()
|
||||
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
@@ -201,14 +201,13 @@ class KlipperSelectMcuConnectionMenu(BaseMenu):
|
||||
class KlipperSelectMcuIdMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
self.flash_options = FlashOptions()
|
||||
self.mcu_list = self.flash_options.mcu_list
|
||||
print(self.mcu_list)
|
||||
options = {f"{index}": self.flash_mcu for index in range(len(self.mcu_list))}
|
||||
self.options = options
|
||||
self.input_label_txt = "Select MCU to flash"
|
||||
self.previous_menu = previous_menu
|
||||
self.footer_type = FooterType.BACK_HELP
|
||||
|
||||
def print_menu(self) -> None:
|
||||
@@ -250,17 +249,20 @@ class KlipperSelectMcuIdMenu(BaseMenu):
|
||||
self.goto_next_menu()
|
||||
|
||||
def goto_next_menu(self, **kwargs):
|
||||
from core.menus.main_menu import MainMenu
|
||||
from core.menus.advanced_menu import AdvancedMenu
|
||||
pass
|
||||
# TODO: navigate back to advanced menu after flashing
|
||||
|
||||
AdvancedMenu(previous_menu=MainMenu()).run()
|
||||
# from core.menus.main_menu import MainMenu
|
||||
# from core.menus.advanced_menu import AdvancedMenu
|
||||
#
|
||||
# next_menu = AdvancedMenu()
|
||||
# next_menu.start()
|
||||
|
||||
|
||||
class KlipperFlashMethodHelpMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
|
||||
def print_menu(self) -> None:
|
||||
header = " < ? > Help: Flash MCU < ? > "
|
||||
@@ -302,10 +304,9 @@ class KlipperFlashMethodHelpMenu(BaseMenu):
|
||||
|
||||
|
||||
class KlipperFlashCommandHelpMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
|
||||
def print_menu(self) -> None:
|
||||
header = " < ? > Help: Flash MCU < ? > "
|
||||
@@ -334,10 +335,9 @@ class KlipperFlashCommandHelpMenu(BaseMenu):
|
||||
|
||||
|
||||
class KlipperMcuConnectionHelpMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
|
||||
def print_menu(self) -> None:
|
||||
header = " < ? > Help: Flash MCU < ? > "
|
||||
|
||||
@@ -17,10 +17,9 @@ from utils.constants import RESET_FORMAT, COLOR_YELLOW
|
||||
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class LogUploadMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = True
|
||||
self.logfile_list = get_logfile_list()
|
||||
options = {f"{index}": self.upload for index in range(len(self.logfile_list))}
|
||||
self.options = options
|
||||
|
||||
@@ -16,10 +16,9 @@ from utils.constants import RESET_FORMAT, COLOR_RED, COLOR_CYAN
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
class MoonrakerRemoveMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.header = False
|
||||
self.options = {
|
||||
"0": self.toggle_all,
|
||||
"1": self.toggle_remove_moonraker_service,
|
||||
|
||||
@@ -17,9 +17,9 @@ from utils.constants import RESET_FORMAT, COLOR_RED, COLOR_CYAN
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
class ClientRemoveMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu, client: ClientData):
|
||||
def __init__(self, client: ClientData):
|
||||
super().__init__()
|
||||
self.previous_menu = previous_menu
|
||||
self.header = False
|
||||
self.options = self.get_options(client)
|
||||
|
||||
self.client = client
|
||||
|
||||
@@ -21,11 +21,3 @@ NAVI_OPTIONS = {
|
||||
FooterType.BACK: ["b"],
|
||||
FooterType.BACK_HELP: ["b", "h"],
|
||||
}
|
||||
|
||||
|
||||
class ExitAppException(Exception):
|
||||
pass
|
||||
|
||||
|
||||
class GoBackException(Exception):
|
||||
pass
|
||||
|
||||
@@ -18,17 +18,15 @@ from utils.constants import COLOR_YELLOW, RESET_FORMAT
|
||||
|
||||
|
||||
class AdvancedMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.options = {
|
||||
"1": None,
|
||||
"2": None,
|
||||
"3": None,
|
||||
"4": lambda: KlipperFlashMethodMenu(previous_menu=self).run(),
|
||||
"4": KlipperFlashMethodMenu,
|
||||
"5": None,
|
||||
"6": lambda: KlipperSelectMcuConnectionMenu(previous_menu=self).run(),
|
||||
"6": KlipperSelectMcuConnectionMenu,
|
||||
}
|
||||
|
||||
def print_menu(self):
|
||||
|
||||
@@ -27,10 +27,8 @@ from utils.constants import COLOR_CYAN, RESET_FORMAT, COLOR_YELLOW
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class BackupMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.options = {
|
||||
"1": self.backup_klipper,
|
||||
"2": self.backup_moonraker,
|
||||
|
||||
@@ -15,7 +15,7 @@ import textwrap
|
||||
from abc import abstractmethod, ABC
|
||||
from typing import Dict, Union, Callable, Type
|
||||
|
||||
from core.menus import FooterType, NAVI_OPTIONS, ExitAppException, GoBackException
|
||||
from core.menus import FooterType, NAVI_OPTIONS
|
||||
from utils.constants import (
|
||||
COLOR_GREEN,
|
||||
COLOR_YELLOW,
|
||||
@@ -92,15 +92,16 @@ def print_back_help_footer():
|
||||
print(footer, end="")
|
||||
|
||||
|
||||
Options = Dict[str, Callable]
|
||||
Option = Union[Callable, Type["BaseMenu"], "BaseMenu"]
|
||||
Options = Dict[str, Option]
|
||||
|
||||
|
||||
class BaseMenu(ABC):
|
||||
options: Options = {}
|
||||
options: Options = None
|
||||
options_offset: int = 0
|
||||
default_option: Union[Callable, None] = None
|
||||
default_option: Union[Option, None] = None
|
||||
input_label_txt: str = "Perform action"
|
||||
header: bool = False
|
||||
header: bool = True
|
||||
previous_menu: Union[Type[BaseMenu], BaseMenu] = None
|
||||
footer_type: FooterType = FooterType.BACK
|
||||
|
||||
@@ -129,7 +130,7 @@ class BaseMenu(ABC):
|
||||
self.print_menu()
|
||||
self.print_footer()
|
||||
|
||||
def validate_user_input(self, usr_input: str) -> Callable:
|
||||
def validate_user_input(self, usr_input: str) -> Union[Option, str, None]:
|
||||
"""
|
||||
Validate the user input and either return an Option, a string or None
|
||||
:param usr_input: The user input in form of a string
|
||||
@@ -143,27 +144,26 @@ class BaseMenu(ABC):
|
||||
is_valid_navigation = self.footer_type in NAVI_OPTIONS
|
||||
user_navigated = usr_input in NAVI_OPTIONS[self.footer_type]
|
||||
if is_valid_navigation and user_navigated:
|
||||
if usr_input == "q":
|
||||
raise ExitAppException()
|
||||
elif usr_input == "b":
|
||||
raise GoBackException()
|
||||
elif usr_input == "h":
|
||||
return option
|
||||
return usr_input
|
||||
|
||||
# if usr_input is None or an empty string, we execute the menues default option if specified
|
||||
if usr_input == "" and self.default_option is not None:
|
||||
if option is None or option == "" and self.default_option is not None:
|
||||
return self.default_option
|
||||
|
||||
# user selected a regular option
|
||||
if option is not None:
|
||||
return option
|
||||
|
||||
def handle_user_input(self) -> Callable:
|
||||
return None
|
||||
|
||||
def handle_user_input(self) -> Union[Option, str]:
|
||||
"""Handle the user input, return the validated input or print an error."""
|
||||
while True:
|
||||
print(f"{COLOR_CYAN}###### {self.input_label_txt}: {RESET_FORMAT}", end="")
|
||||
usr_input = input().lower()
|
||||
validated_input = self.validate_user_input(usr_input)
|
||||
|
||||
if (validated_input := self.validate_user_input(usr_input)) is not None:
|
||||
if validated_input is not None:
|
||||
return validated_input
|
||||
else:
|
||||
Logger.print_error("Invalid input!", False)
|
||||
@@ -171,11 +171,36 @@ class BaseMenu(ABC):
|
||||
def run(self) -> None:
|
||||
"""Start the menu lifecycle. When this function returns, the lifecycle of the menu ends."""
|
||||
while True:
|
||||
try:
|
||||
self.display_menu()
|
||||
self.handle_user_input()()
|
||||
except GoBackException:
|
||||
return
|
||||
except ExitAppException:
|
||||
choice = self.handle_user_input()
|
||||
|
||||
if choice == "q":
|
||||
Logger.print_ok("###### Happy printing!", False)
|
||||
sys.exit(0)
|
||||
elif choice == "b":
|
||||
return
|
||||
else:
|
||||
self.execute_option(choice)
|
||||
|
||||
def execute_option(self, option: Option) -> None:
|
||||
if option is None:
|
||||
raise NotImplementedError(f"No implementation for {option}")
|
||||
|
||||
if isinstance(option, type) and issubclass(option, BaseMenu):
|
||||
self.navigate_to_menu(option, True)
|
||||
elif isinstance(option, BaseMenu):
|
||||
self.navigate_to_menu(option, False)
|
||||
elif callable(option):
|
||||
option()
|
||||
|
||||
def navigate_to_menu(self, menu, instantiate: bool) -> None:
|
||||
"""
|
||||
Method for handling the actual menu switch. Can either take in a menu type or an already
|
||||
instantiated menu class. Use instantiated menu classes only if the menu requires specific input parameters
|
||||
:param menu: A menu type or menu instance
|
||||
:param instantiate: Specify if the menu requires instantiation
|
||||
:return: None
|
||||
"""
|
||||
menu = menu() if instantiate else menu
|
||||
menu.previous_menu = self
|
||||
menu.run()
|
||||
|
||||
@@ -22,12 +22,12 @@ from utils.constants import RESET_FORMAT, COLOR_CYAN, COLOR_YELLOW
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class ExtensionsMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
self.header = False
|
||||
self.options: Options = self.get_options()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.extensions = self.discover_extensions()
|
||||
self.options: Options = self.get_options(self.extensions)
|
||||
|
||||
def discover_extensions(self) -> List[BaseExtension]:
|
||||
extensions = []
|
||||
@@ -56,11 +56,11 @@ class ExtensionsMenu(BaseMenu):
|
||||
|
||||
return sorted(extensions, key=lambda ex: ex.metadata.get("index"))
|
||||
|
||||
def get_options(self, extensions: List[BaseExtension]) -> Options:
|
||||
def get_options(self) -> Options:
|
||||
options: Options = {}
|
||||
for extension in extensions:
|
||||
for extension in self.extensions:
|
||||
index = extension.metadata.get("index")
|
||||
options[f"{index}"] = lambda: ExtensionSubmenu(self, extension).run()
|
||||
options[f"{index}"] = ExtensionSubmenu(extension)
|
||||
|
||||
return options
|
||||
|
||||
@@ -90,10 +90,9 @@ class ExtensionsMenu(BaseMenu):
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class ExtensionSubmenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu, extension: BaseExtension):
|
||||
def __init__(self, extension: BaseExtension):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu = previous_menu
|
||||
self.header = False
|
||||
self.options = {
|
||||
"1": extension.install_extension,
|
||||
"2": extension.remove_extension,
|
||||
|
||||
@@ -21,10 +21,8 @@ from utils.constants import COLOR_GREEN, RESET_FORMAT
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class InstallMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.options = {
|
||||
"1": self.install_klipper,
|
||||
"2": self.install_moonraker,
|
||||
|
||||
@@ -40,19 +40,16 @@ from utils.constants import (
|
||||
class MainMenu(BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.options = {
|
||||
"0": lambda: LogUploadMenu(previous_menu=self).run(),
|
||||
"1": lambda: InstallMenu(previous_menu=self).run(),
|
||||
"2": lambda: UpdateMenu(previous_menu=self).run(),
|
||||
"3": lambda: RemoveMenu(previous_menu=self).run(),
|
||||
"4": lambda: AdvancedMenu(previous_menu=self).run(),
|
||||
"5": lambda: BackupMenu(previous_menu=self).run(),
|
||||
"6": None,
|
||||
"e": lambda: ExtensionsMenu(previous_menu=self).run(),
|
||||
"s": lambda: SettingsMenu(previous_menu=self).run(),
|
||||
"0": LogUploadMenu,
|
||||
"1": InstallMenu,
|
||||
"2": UpdateMenu,
|
||||
"3": RemoveMenu,
|
||||
"4": AdvancedMenu,
|
||||
"5": BackupMenu,
|
||||
"e": ExtensionsMenu,
|
||||
"s": SettingsMenu,
|
||||
}
|
||||
self.header = True
|
||||
self.footer_type = FooterType.QUIT
|
||||
|
||||
self.kl_status = ""
|
||||
|
||||
@@ -22,19 +22,13 @@ from utils.constants import COLOR_RED, RESET_FORMAT
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class RemoveMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.options = {
|
||||
"1": lambda: KlipperRemoveMenu(previous_menu=self).run(),
|
||||
"2": lambda: MoonrakerRemoveMenu(previous_menu=self).run(),
|
||||
"3": lambda: ClientRemoveMenu(
|
||||
previous_menu=self, client=load_client_data("mainsail")
|
||||
).run(),
|
||||
"4": lambda: ClientRemoveMenu(
|
||||
previous_menu=self, client=load_client_data("fluidd")
|
||||
).run(),
|
||||
"1": KlipperRemoveMenu,
|
||||
"2": MoonrakerRemoveMenu,
|
||||
"3": ClientRemoveMenu(client=load_client_data("mainsail")),
|
||||
"4": ClientRemoveMenu(client=load_client_data("fluidd")),
|
||||
"5": None,
|
||||
"6": None,
|
||||
"7": None,
|
||||
|
||||
@@ -12,11 +12,9 @@ from core.menus.base_menu import BaseMenu
|
||||
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class SettingsMenu(BaseMenu):
|
||||
def __init__(self, previous_menu: BaseMenu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
|
||||
def print_menu(self):
|
||||
print("self")
|
||||
|
||||
|
||||
@@ -38,10 +38,8 @@ from utils.constants import (
|
||||
# noinspection PyUnusedLocal
|
||||
# noinspection PyMethodMayBeStatic
|
||||
class UpdateMenu(BaseMenu):
|
||||
def __init__(self, previous_menu):
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
|
||||
self.previous_menu: BaseMenu = previous_menu
|
||||
self.options = {
|
||||
"0": self.update_all,
|
||||
"1": self.update_klipper,
|
||||
|
||||
@@ -79,6 +79,7 @@ class MainsailThemeInstallMenu(BaseMenu):
|
||||
|
||||
def __init__(self, instances: List[Klipper]):
|
||||
super().__init__()
|
||||
self.header = False
|
||||
self.themes: List[ThemeData] = self.load_themes()
|
||||
options = {f"{index}": self.install_theme for index in range(len(self.themes))}
|
||||
self.options = options
|
||||
|
||||
Reference in New Issue
Block a user