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.
		
		
		
		
			
				
					182 lines
				
				6.3 KiB
			
		
		
			
		
	
	
					182 lines
				
				6.3 KiB
			| 
											6 days ago
										 | import pyray as rl
 | ||
|  | import qrcode
 | ||
|  | import numpy as np
 | ||
|  | import time
 | ||
|  | 
 | ||
|  | from openpilot.common.api import Api
 | ||
|  | from openpilot.common.swaglog import cloudlog
 | ||
|  | from openpilot.common.params import Params
 | ||
|  | from openpilot.system.ui.widgets import Widget
 | ||
|  | from openpilot.system.ui.lib.application import FontWeight, gui_app
 | ||
|  | from openpilot.system.ui.lib.wrap_text import wrap_text
 | ||
|  | from openpilot.system.ui.lib.text_measure import measure_text_cached
 | ||
|  | from openpilot.selfdrive.ui.ui_state import ui_state
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class IconButton(Widget):
 | ||
|  |   def __init__(self, texture: rl.Texture):
 | ||
|  |     super().__init__()
 | ||
|  |     self._texture = texture
 | ||
|  | 
 | ||
|  |   def _render(self, rect: rl.Rectangle):
 | ||
|  |     color = rl.Color(180, 180, 180, 150) if self.is_pressed else rl.WHITE
 | ||
|  |     draw_x = rect.x + (rect.width - self._texture.width) / 2
 | ||
|  |     draw_y = rect.y + (rect.height - self._texture.height) / 2
 | ||
|  |     rl.draw_texture(self._texture, int(draw_x), int(draw_y), color)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class PairingDialog(Widget):
 | ||
|  |   """Dialog for device pairing with QR code."""
 | ||
|  | 
 | ||
|  |   QR_REFRESH_INTERVAL = 300  # 5 minutes in seconds
 | ||
|  | 
 | ||
|  |   def __init__(self):
 | ||
|  |     super().__init__()
 | ||
|  |     self.params = Params()
 | ||
|  |     self.qr_texture: rl.Texture | None = None
 | ||
|  |     self.last_qr_generation = float('-inf')
 | ||
|  |     self._close_btn = IconButton(gui_app.texture("icons/close.png", 80, 80))
 | ||
|  |     self._close_btn.set_click_callback(lambda: gui_app.set_modal_overlay(None))
 | ||
|  | 
 | ||
|  |   def _get_pairing_url(self) -> str:
 | ||
|  |     try:
 | ||
|  |       dongle_id = self.params.get("DongleId") or ""
 | ||
|  |       token = Api(dongle_id).get_token({'pair': True})
 | ||
|  |     except Exception:
 | ||
|  |       cloudlog.exception("Failed to get pairing token")
 | ||
|  |       token = ""
 | ||
|  |     return f"https://connect.comma.ai/?pair={token}"
 | ||
|  | 
 | ||
|  |   def _generate_qr_code(self) -> None:
 | ||
|  |     try:
 | ||
|  |       qr = qrcode.QRCode(version=1, error_correction=qrcode.constants.ERROR_CORRECT_L, box_size=10, border=4)
 | ||
|  |       qr.add_data(self._get_pairing_url())
 | ||
|  |       qr.make(fit=True)
 | ||
|  | 
 | ||
|  |       pil_img = qr.make_image(fill_color="black", back_color="white").convert('RGBA')
 | ||
|  |       img_array = np.array(pil_img, dtype=np.uint8)
 | ||
|  | 
 | ||
|  |       if self.qr_texture and self.qr_texture.id != 0:
 | ||
|  |         rl.unload_texture(self.qr_texture)
 | ||
|  | 
 | ||
|  |       rl_image = rl.Image()
 | ||
|  |       rl_image.data = rl.ffi.cast("void *", img_array.ctypes.data)
 | ||
|  |       rl_image.width = pil_img.width
 | ||
|  |       rl_image.height = pil_img.height
 | ||
|  |       rl_image.mipmaps = 1
 | ||
|  |       rl_image.format = rl.PixelFormat.PIXELFORMAT_UNCOMPRESSED_R8G8B8A8
 | ||
|  | 
 | ||
|  |       self.qr_texture = rl.load_texture_from_image(rl_image)
 | ||
|  |     except Exception:
 | ||
|  |       cloudlog.exception("QR code generation failed")
 | ||
|  |       self.qr_texture = None
 | ||
|  | 
 | ||
|  |   def _check_qr_refresh(self) -> None:
 | ||
|  |     current_time = time.monotonic()
 | ||
|  |     if current_time - self.last_qr_generation >= self.QR_REFRESH_INTERVAL:
 | ||
|  |       self._generate_qr_code()
 | ||
|  |       self.last_qr_generation = current_time
 | ||
|  | 
 | ||
|  |   def _update_state(self):
 | ||
|  |     if ui_state.prime_state.is_paired():
 | ||
|  |       gui_app.set_modal_overlay(None)
 | ||
|  | 
 | ||
|  |   def _render(self, rect: rl.Rectangle) -> int:
 | ||
|  |     rl.clear_background(rl.Color(224, 224, 224, 255))
 | ||
|  | 
 | ||
|  |     self._check_qr_refresh()
 | ||
|  | 
 | ||
|  |     margin = 70
 | ||
|  |     content_rect = rl.Rectangle(rect.x + margin, rect.y + margin, rect.width - 2 * margin, rect.height - 2 * margin)
 | ||
|  |     y = content_rect.y
 | ||
|  | 
 | ||
|  |     # Close button
 | ||
|  |     close_size = 80
 | ||
|  |     pad = 20
 | ||
|  |     close_rect = rl.Rectangle(content_rect.x - pad, y - pad, close_size + pad * 2, close_size + pad * 2)
 | ||
|  |     self._close_btn.render(close_rect)
 | ||
|  | 
 | ||
|  |     y += close_size + 40
 | ||
|  | 
 | ||
|  |     # Title
 | ||
|  |     title = "Pair your device to your comma account"
 | ||
|  |     title_font = gui_app.font(FontWeight.NORMAL)
 | ||
|  |     left_width = int(content_rect.width * 0.5 - 15)
 | ||
|  | 
 | ||
|  |     title_wrapped = wrap_text(title_font, title, 75, left_width)
 | ||
|  |     rl.draw_text_ex(title_font, "\n".join(title_wrapped), rl.Vector2(content_rect.x, y), 75, 0.0, rl.BLACK)
 | ||
|  |     y += len(title_wrapped) * 75 + 60
 | ||
|  | 
 | ||
|  |     # Two columns: instructions and QR code
 | ||
|  |     remaining_height = content_rect.height - (y - content_rect.y)
 | ||
|  |     right_width = content_rect.width // 2 - 20
 | ||
|  | 
 | ||
|  |     # Instructions
 | ||
|  |     self._render_instructions(rl.Rectangle(content_rect.x, y, left_width, remaining_height))
 | ||
|  | 
 | ||
|  |     # QR code
 | ||
|  |     qr_size = min(right_width, content_rect.height) - 40
 | ||
|  |     qr_x = content_rect.x + left_width + 40 + (right_width - qr_size) // 2
 | ||
|  |     qr_y = content_rect.y
 | ||
|  |     self._render_qr_code(rl.Rectangle(qr_x, qr_y, qr_size, qr_size))
 | ||
|  | 
 | ||
|  |     return -1
 | ||
|  | 
 | ||
|  |   def _render_instructions(self, rect: rl.Rectangle) -> None:
 | ||
|  |     instructions = [
 | ||
|  |       "Go to https://connect.comma.ai on your phone",
 | ||
|  |       "Click \"add new device\" and scan the QR code on the right",
 | ||
|  |       "Bookmark connect.comma.ai to your home screen to use it like an app",
 | ||
|  |     ]
 | ||
|  | 
 | ||
|  |     font = gui_app.font(FontWeight.BOLD)
 | ||
|  |     y = rect.y
 | ||
|  | 
 | ||
|  |     for i, text in enumerate(instructions):
 | ||
|  |       circle_radius = 25
 | ||
|  |       circle_x = rect.x + circle_radius + 15
 | ||
|  |       text_x = rect.x + circle_radius * 2 + 40
 | ||
|  |       text_width = rect.width - (circle_radius * 2 + 40)
 | ||
|  | 
 | ||
|  |       wrapped = wrap_text(font, text, 47, int(text_width))
 | ||
|  |       text_height = len(wrapped) * 47
 | ||
|  |       circle_y = y + text_height // 2
 | ||
|  | 
 | ||
|  |       # Circle and number
 | ||
|  |       rl.draw_circle(int(circle_x), int(circle_y), circle_radius, rl.Color(70, 70, 70, 255))
 | ||
|  |       number = str(i + 1)
 | ||
|  |       number_size = measure_text_cached(font, number, 30)
 | ||
|  |       rl.draw_text_ex(font, number, (int(circle_x - number_size.x // 2), int(circle_y - number_size.y // 2)), 30, 0, rl.WHITE)
 | ||
|  | 
 | ||
|  |       # Text
 | ||
|  |       rl.draw_text_ex(font, "\n".join(wrapped), rl.Vector2(text_x, y), 47, 0.0, rl.BLACK)
 | ||
|  |       y += text_height + 50
 | ||
|  | 
 | ||
|  |   def _render_qr_code(self, rect: rl.Rectangle) -> None:
 | ||
|  |     if not self.qr_texture:
 | ||
|  |       rl.draw_rectangle_rounded(rect, 0.1, 20, rl.Color(240, 240, 240, 255))
 | ||
|  |       error_font = gui_app.font(FontWeight.BOLD)
 | ||
|  |       rl.draw_text_ex(
 | ||
|  |         error_font, "QR Code Error", rl.Vector2(rect.x + 20, rect.y + rect.height // 2 - 15), 30, 0.0, rl.RED
 | ||
|  |       )
 | ||
|  |       return
 | ||
|  | 
 | ||
|  |     source = rl.Rectangle(0, 0, self.qr_texture.width, self.qr_texture.height)
 | ||
|  |     rl.draw_texture_pro(self.qr_texture, source, rect, rl.Vector2(0, 0), 0, rl.WHITE)
 | ||
|  | 
 | ||
|  |   def __del__(self):
 | ||
|  |     if self.qr_texture and self.qr_texture.id != 0:
 | ||
|  |       rl.unload_texture(self.qr_texture)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | if __name__ == "__main__":
 | ||
|  |   gui_app.init_window("pairing device")
 | ||
|  |   pairing = PairingDialog()
 | ||
|  |   try:
 | ||
|  |     for _ in gui_app.render():
 | ||
|  |       result = pairing.render(rl.Rectangle(0, 0, gui_app.width, gui_app.height))
 | ||
|  |       if result != -1:
 | ||
|  |         break
 | ||
|  |   finally:
 | ||
|  |     del pairing
 |