You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							268 lines
						
					
					
						
							10 KiB
						
					
					
				
			
		
		
	
	
							268 lines
						
					
					
						
							10 KiB
						
					
					
				| from functools import partial
 | |
| import time
 | |
| from typing import Literal
 | |
| 
 | |
| import pyray as rl
 | |
| 
 | |
| from openpilot.system.ui.lib.application import gui_app, FontWeight
 | |
| from openpilot.system.ui.widgets import Widget
 | |
| from openpilot.system.ui.widgets.button import ButtonStyle, Button
 | |
| from openpilot.system.ui.widgets.inputbox import InputBox
 | |
| from openpilot.system.ui.widgets.label import Label
 | |
| 
 | |
| KEY_FONT_SIZE = 96
 | |
| DOUBLE_CLICK_THRESHOLD = 0.5  # seconds
 | |
| DELETE_REPEAT_DELAY = 0.5
 | |
| DELETE_REPEAT_INTERVAL = 0.07
 | |
| 
 | |
| # Constants for special keys
 | |
| CONTENT_MARGIN = 50
 | |
| BACKSPACE_KEY = "<-"
 | |
| ENTER_KEY = "->"
 | |
| SPACE_KEY = " "
 | |
| SHIFT_INACTIVE_KEY = "SHIFT_OFF"
 | |
| SHIFT_ACTIVE_KEY = "SHIFT_ON"
 | |
| CAPS_LOCK_KEY = "CAPS"
 | |
| NUMERIC_KEY = "123"
 | |
| SYMBOL_KEY = "#+="
 | |
| ABC_KEY = "ABC"
 | |
| 
 | |
| # Define keyboard layouts as a dictionary for easier access
 | |
| KEYBOARD_LAYOUTS = {
 | |
|   "lowercase": [
 | |
|     ["q", "w", "e", "r", "t", "y", "u", "i", "o", "p"],
 | |
|     ["a", "s", "d", "f", "g", "h", "j", "k", "l"],
 | |
|     [SHIFT_INACTIVE_KEY, "z", "x", "c", "v", "b", "n", "m", BACKSPACE_KEY],
 | |
|     [NUMERIC_KEY, "/", "-", SPACE_KEY, ".", ENTER_KEY],
 | |
|   ],
 | |
|   "uppercase": [
 | |
|     ["Q", "W", "E", "R", "T", "Y", "U", "I", "O", "P"],
 | |
|     ["A", "S", "D", "F", "G", "H", "J", "K", "L"],
 | |
|     [SHIFT_ACTIVE_KEY, "Z", "X", "C", "V", "B", "N", "M", BACKSPACE_KEY],
 | |
|     [NUMERIC_KEY, "/", "-", SPACE_KEY, ".", ENTER_KEY],
 | |
|   ],
 | |
|   "numbers": [
 | |
|     ["1", "2", "3", "4", "5", "6", "7", "8", "9", "0"],
 | |
|     ["-", "/", ":", ";", "(", ")", "$", "&", "@", "\""],
 | |
|     [SYMBOL_KEY, ".", ",", "?", "!", "`", BACKSPACE_KEY],
 | |
|     [ABC_KEY, SPACE_KEY, ".", ENTER_KEY],
 | |
|   ],
 | |
|   "specials": [
 | |
|     ["[", "]", "{", "}", "#", "%", "^", "*", "+", "="],
 | |
|     ["_", "\\", "|", "~", "<", ">", "€", "£", "¥", "•"],
 | |
|     [NUMERIC_KEY, ".", ",", "?", "!", "'", BACKSPACE_KEY],
 | |
|     [ABC_KEY, SPACE_KEY, ".", ENTER_KEY],
 | |
|   ],
 | |
| }
 | |
| 
 | |
| 
 | |
| class Keyboard(Widget):
 | |
|   def __init__(self, max_text_size: int = 255, min_text_size: int = 0, password_mode: bool = False, show_password_toggle: bool = False):
 | |
|     super().__init__()
 | |
|     self._layout_name: Literal["lowercase", "uppercase", "numbers", "specials"] = "lowercase"
 | |
|     self._caps_lock = False
 | |
|     self._last_shift_press_time = 0
 | |
|     self._title = Label("", 90, FontWeight.BOLD, rl.GuiTextAlignment.TEXT_ALIGN_LEFT, text_padding=20)
 | |
|     self._sub_title = Label("", 55, FontWeight.NORMAL, rl.GuiTextAlignment.TEXT_ALIGN_LEFT, text_padding=20)
 | |
| 
 | |
|     self._max_text_size = max_text_size
 | |
|     self._min_text_size = min_text_size
 | |
|     self._input_box = InputBox(max_text_size)
 | |
|     self._password_mode = password_mode
 | |
|     self._show_password_toggle = show_password_toggle
 | |
| 
 | |
|     # Backspace key repeat tracking
 | |
|     self._backspace_pressed: bool = False
 | |
|     self._backspace_press_time: float = 0.0
 | |
|     self._backspace_last_repeat: float = 0.0
 | |
| 
 | |
|     self._render_return_status = -1
 | |
|     self._cancel_button = Button("Cancel", self._cancel_button_callback)
 | |
| 
 | |
|     self._eye_button = Button("", self._eye_button_callback, button_style=ButtonStyle.TRANSPARENT)
 | |
| 
 | |
|     self._eye_open_texture = gui_app.texture("icons/eye_open.png", 81, 54)
 | |
|     self._eye_closed_texture = gui_app.texture("icons/eye_closed.png", 81, 54)
 | |
|     self._key_icons = {
 | |
|       BACKSPACE_KEY: gui_app.texture("icons/backspace.png", 80, 80),
 | |
|       SHIFT_INACTIVE_KEY: gui_app.texture("icons/shift.png", 80, 80),
 | |
|       SHIFT_ACTIVE_KEY: gui_app.texture("icons/shift-fill.png", 80, 80),
 | |
|       CAPS_LOCK_KEY: gui_app.texture("icons/capslock-fill.png", 80, 80),
 | |
|       ENTER_KEY: gui_app.texture("icons/arrow-right.png", 80, 80),
 | |
|     }
 | |
| 
 | |
|     self._all_keys = {}
 | |
|     for l in KEYBOARD_LAYOUTS:
 | |
|       for _, keys in enumerate(KEYBOARD_LAYOUTS[l]):
 | |
|         for _, key in enumerate(keys):
 | |
|           if key in self._key_icons:
 | |
|             texture = self._key_icons[key]
 | |
|             self._all_keys[key] = Button("", partial(self._key_callback, key), icon=texture,
 | |
|                                         button_style=ButtonStyle.PRIMARY if key == ENTER_KEY else ButtonStyle.KEYBOARD, multi_touch=True)
 | |
|           else:
 | |
|             self._all_keys[key] = Button(key, partial(self._key_callback, key), button_style=ButtonStyle.KEYBOARD, font_size=85, multi_touch=True)
 | |
|     self._all_keys[CAPS_LOCK_KEY] = Button("", partial(self._key_callback, CAPS_LOCK_KEY), icon=self._key_icons[CAPS_LOCK_KEY],
 | |
|                                            button_style=ButtonStyle.KEYBOARD, multi_touch=True)
 | |
| 
 | |
|   def set_text(self, text: str):
 | |
|     self._input_box.text = text
 | |
| 
 | |
|   @property
 | |
|   def text(self):
 | |
|     return self._input_box.text
 | |
| 
 | |
|   def clear(self):
 | |
|     self._layout_name = "lowercase"
 | |
|     self._caps_lock = False
 | |
|     self._input_box.clear()
 | |
|     self._backspace_pressed = False
 | |
| 
 | |
|   def set_title(self, title: str, sub_title: str = ""):
 | |
|     self._title.set_text(title)
 | |
|     self._sub_title.set_text(sub_title)
 | |
| 
 | |
|   def _eye_button_callback(self):
 | |
|     self._password_mode = not self._password_mode
 | |
| 
 | |
|   def _cancel_button_callback(self):
 | |
|     self.clear()
 | |
|     self._render_return_status = 0
 | |
| 
 | |
|   def _key_callback(self, k):
 | |
|     if k == ENTER_KEY:
 | |
|       self._render_return_status = 1
 | |
|     else:
 | |
|       self.handle_key_press(k)
 | |
| 
 | |
|   def _render(self, rect: rl.Rectangle):
 | |
|     rect = rl.Rectangle(rect.x + CONTENT_MARGIN, rect.y + CONTENT_MARGIN, rect.width - 2 * CONTENT_MARGIN, rect.height - 2 * CONTENT_MARGIN)
 | |
|     self._title.render(rl.Rectangle(rect.x, rect.y, rect.width, 95))
 | |
|     self._sub_title.render(rl.Rectangle(rect.x, rect.y + 95, rect.width, 60))
 | |
|     self._cancel_button.render(rl.Rectangle(rect.x + rect.width - 386, rect.y, 386, 125))
 | |
| 
 | |
|     # Draw input box and password toggle
 | |
|     input_margin = 25
 | |
|     input_box_rect = rl.Rectangle(rect.x + input_margin, rect.y + 160, rect.width - input_margin, 100)
 | |
|     self._render_input_area(input_box_rect)
 | |
| 
 | |
|     # Process backspace key repeat if it's held down
 | |
|     if not self._all_keys[BACKSPACE_KEY].is_pressed:
 | |
|       self._backspace_pressed = False
 | |
| 
 | |
|     if self._backspace_pressed:
 | |
|       current_time = time.monotonic()
 | |
|       time_since_press = current_time - self._backspace_press_time
 | |
| 
 | |
|       # After initial delay, start repeating with shorter intervals
 | |
|       if time_since_press > DELETE_REPEAT_DELAY:
 | |
|         time_since_last_repeat = current_time - self._backspace_last_repeat
 | |
|         if time_since_last_repeat > DELETE_REPEAT_INTERVAL:
 | |
|           self._input_box.delete_char_before_cursor()
 | |
|           self._backspace_last_repeat = current_time
 | |
| 
 | |
|     layout = KEYBOARD_LAYOUTS[self._layout_name]
 | |
| 
 | |
|     h_space, v_space = 15, 15
 | |
|     row_y_start = rect.y + 300  # Starting Y position for the first row
 | |
|     key_height = (rect.height - 300 - 3 * v_space) / 4
 | |
|     key_max_width = (rect.width - (len(layout[2]) - 1) * h_space) / len(layout[2])
 | |
| 
 | |
|     # Iterate over the rows of keys in the current layout
 | |
|     for row, keys in enumerate(layout):
 | |
|       key_width = min((rect.width - (180 if row == 1 else 0) - h_space * (len(keys) - 1)) / len(keys), key_max_width)
 | |
|       start_x = rect.x + (90 if row == 1 else 0)
 | |
| 
 | |
|       for i, key in enumerate(keys):
 | |
|         if i > 0:
 | |
|           start_x += h_space
 | |
| 
 | |
|         new_width = (key_width * 3 + h_space * 2) if key == SPACE_KEY else (key_width * 2 + h_space if key == ENTER_KEY else key_width)
 | |
|         key_rect = rl.Rectangle(start_x, row_y_start + row * (key_height + v_space), new_width, key_height)
 | |
|         start_x += new_width
 | |
| 
 | |
|         is_enabled = key != ENTER_KEY or len(self._input_box.text) >= self._min_text_size
 | |
| 
 | |
|         if key == BACKSPACE_KEY and self._all_keys[BACKSPACE_KEY].is_pressed and not self._backspace_pressed:
 | |
|           self._backspace_pressed = True
 | |
|           self._backspace_press_time = time.monotonic()
 | |
|           self._backspace_last_repeat = time.monotonic()
 | |
| 
 | |
|         if key in self._key_icons:
 | |
|           if key == SHIFT_ACTIVE_KEY and self._caps_lock:
 | |
|             key = CAPS_LOCK_KEY
 | |
|           self._all_keys[key].set_enabled(is_enabled)
 | |
|           self._all_keys[key].render(key_rect)
 | |
|         else:
 | |
|           self._all_keys[key].set_enabled(is_enabled)
 | |
|           self._all_keys[key].render(key_rect)
 | |
| 
 | |
|     return self._render_return_status
 | |
| 
 | |
|   def _render_input_area(self, input_rect: rl.Rectangle):
 | |
|     if self._show_password_toggle:
 | |
|       self._input_box.set_password_mode(self._password_mode)
 | |
|       self._input_box.render(rl.Rectangle(input_rect.x, input_rect.y, input_rect.width - 100, input_rect.height))
 | |
| 
 | |
|       # render eye icon
 | |
|       eye_texture = self._eye_closed_texture if self._password_mode else self._eye_open_texture
 | |
| 
 | |
|       eye_rect = rl.Rectangle(input_rect.x + input_rect.width - 90, input_rect.y, 80, input_rect.height)
 | |
|       self._eye_button.render(eye_rect)
 | |
| 
 | |
|       eye_x = eye_rect.x + (eye_rect.width - eye_texture.width) / 2
 | |
|       eye_y = eye_rect.y + (eye_rect.height - eye_texture.height) / 2
 | |
| 
 | |
|       rl.draw_texture_v(eye_texture, rl.Vector2(eye_x, eye_y), rl.WHITE)
 | |
|     else:
 | |
|       self._input_box.render(input_rect)
 | |
| 
 | |
|     rl.draw_line_ex(
 | |
|       rl.Vector2(input_rect.x, input_rect.y + input_rect.height - 2),
 | |
|       rl.Vector2(input_rect.x + input_rect.width, input_rect.y + input_rect.height - 2),
 | |
|       3.0,  # 3 pixel thickness
 | |
|       rl.Color(189, 189, 189, 255),
 | |
|     )
 | |
| 
 | |
|   def handle_key_press(self, key):
 | |
|     if key in (CAPS_LOCK_KEY, ABC_KEY):
 | |
|       self._caps_lock = False
 | |
|       self._layout_name = "lowercase"
 | |
|     elif key == SHIFT_INACTIVE_KEY:
 | |
|       self._last_shift_press_time = time.monotonic()
 | |
|       self._layout_name = "uppercase"
 | |
|     elif key == SHIFT_ACTIVE_KEY:
 | |
|       if time.monotonic() - self._last_shift_press_time < DOUBLE_CLICK_THRESHOLD:
 | |
|         self._caps_lock = True
 | |
|       else:
 | |
|         self._layout_name = "lowercase"
 | |
|     elif key == NUMERIC_KEY:
 | |
|       self._layout_name = "numbers"
 | |
|     elif key == SYMBOL_KEY:
 | |
|       self._layout_name = "specials"
 | |
|     elif key == BACKSPACE_KEY:
 | |
|       self._input_box.delete_char_before_cursor()
 | |
|     else:
 | |
|       self._input_box.add_char_at_cursor(key)
 | |
|       if not self._caps_lock and self._layout_name == "uppercase":
 | |
|         self._layout_name = "lowercase"
 | |
| 
 | |
|   def reset(self, min_text_size: int | None = None):
 | |
|     if min_text_size is not None:
 | |
|       self._min_text_size = min_text_size
 | |
|     self._render_return_status = -1
 | |
|     self.clear()
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|   gui_app.init_window("Keyboard")
 | |
|   keyboard = Keyboard(min_text_size=8, show_password_toggle=True)
 | |
|   for _ in gui_app.render():
 | |
|     keyboard.set_title("Keyboard Input", "Type your text below")
 | |
|     result = keyboard.render(rl.Rectangle(0, 0, gui_app.width, gui_app.height))
 | |
|     if result == 1:
 | |
|       print(f"You typed: {keyboard.text}")
 | |
|       gui_app.request_close()
 | |
|     elif result == 0:
 | |
|       print("Canceled")
 | |
|       gui_app.request_close()
 | |
|   gui_app.close()
 | |
| 
 |