bauiv1lib.account.v2proxy
V2 account ui bits.
1# Released under the MIT License. See LICENSE for details. 2# 3"""V2 account ui bits.""" 4 5from __future__ import annotations 6 7import time 8import logging 9 10from efro.error import CommunicationError 11import bacommon.cloud 12import bauiv1 as bui 13 14STATUS_CHECK_INTERVAL_SECONDS = 2.0 15 16 17class V2ProxySignInWindow(bui.Window): 18 """A window allowing signing in to a v2 account.""" 19 20 def __init__(self, origin_widget: bui.Widget): 21 self._width = 600 22 self._height = 550 23 self._proxyid: str | None = None 24 self._proxykey: str | None = None 25 self._overlay_web_browser_open = False 26 27 assert bui.app.classic is not None 28 uiscale = bui.app.ui_v1.uiscale 29 super().__init__( 30 root_widget=bui.containerwidget( 31 size=(self._width, self._height), 32 transition='in_scale', 33 scale_origin_stack_offset=( 34 origin_widget.get_screen_space_center() 35 ), 36 scale=( 37 1.16 38 if uiscale is bui.UIScale.SMALL 39 else 1.0 if uiscale is bui.UIScale.MEDIUM else 0.9 40 ), 41 ) 42 ) 43 44 self._state_text = bui.textwidget( 45 parent=self._root_widget, 46 position=(self._width * 0.5, self._height * 0.6), 47 h_align='center', 48 v_align='center', 49 size=(0, 0), 50 scale=1.4, 51 maxwidth=0.9 * self._width, 52 text=bui.Lstr( 53 value='${A}...', 54 subs=[('${A}', bui.Lstr(resource='loadingText'))], 55 ), 56 color=(1, 1, 1), 57 ) 58 self._sub_state_text = bui.textwidget( 59 parent=self._root_widget, 60 position=(self._width * 0.5, self._height * 0.55), 61 h_align='center', 62 v_align='top', 63 scale=0.85, 64 size=(0, 0), 65 maxwidth=0.9 * self._width, 66 text='', 67 ) 68 self._sub_state_text2 = bui.textwidget( 69 parent=self._root_widget, 70 position=(self._width * 0.1, self._height * 0.3), 71 h_align='left', 72 v_align='top', 73 scale=0.7, 74 size=(0, 0), 75 maxwidth=0.9 * self._width, 76 text='', 77 ) 78 79 self._cancel_button = bui.buttonwidget( 80 parent=self._root_widget, 81 position=(30, self._height - 65), 82 size=(130, 50), 83 scale=0.8, 84 label=bui.Lstr(resource='cancelText'), 85 on_activate_call=self._done, 86 autoselect=True, 87 ) 88 89 bui.containerwidget( 90 edit=self._root_widget, cancel_button=self._cancel_button 91 ) 92 93 self._message_in_flight = False 94 self._complete = False 95 self._connection_wait_timeout_time = time.monotonic() + 10.0 96 97 self._update_timer = bui.AppTimer( 98 0.371, bui.WeakCall(self._update), repeat=True 99 ) 100 bui.pushcall(bui.WeakCall(self._update)) 101 102 def _update(self) -> None: 103 104 plus = bui.app.plus 105 assert plus is not None 106 107 # If we've opened an overlay web browser, all we do is kill 108 # ourselves when it closes. 109 if self._overlay_web_browser_open: 110 if not bui.overlay_web_browser_is_open(): 111 self._overlay_web_browser_open = False 112 self._done() 113 return 114 115 if self._message_in_flight or self._complete: 116 return 117 118 now = time.monotonic() 119 120 # Spin for a moment if it looks like we have no server 121 # connection; it might still be getting on its feet. 122 if ( 123 not plus.cloud.connected 124 and now < self._connection_wait_timeout_time 125 ): 126 return 127 128 plus.cloud.send_message_cb( 129 bacommon.cloud.LoginProxyRequestMessage(), 130 on_response=bui.WeakCall(self._on_proxy_request_response), 131 ) 132 self._message_in_flight = True 133 134 def _get_server_address(self) -> str: 135 plus = bui.app.plus 136 assert plus is not None 137 out = plus.get_master_server_address(version=2) 138 assert isinstance(out, str) 139 return out 140 141 def _set_error_state(self, error_location: str) -> None: 142 msaddress = self._get_server_address() 143 addr = msaddress.removeprefix('https://') 144 bui.textwidget( 145 edit=self._state_text, 146 text=f'Unable to connect to {addr}.', 147 color=(1, 0, 0), 148 ) 149 support_email = 'support@froemling.net' 150 bui.textwidget( 151 edit=self._sub_state_text, 152 text=( 153 f'Usually this means your internet is down.\n' 154 f'Please contact {support_email} if this is not the case.' 155 ), 156 color=(1, 0, 0), 157 ) 158 bui.textwidget( 159 edit=self._sub_state_text2, 160 text=( 161 f'debug-info:\n' 162 f' error-location: {error_location}\n' 163 f' connectivity: {bui.app.net.connectivity_state}\n' 164 f' transport: {bui.app.net.transport_state}' 165 ), 166 color=(0.8, 0.2, 0.3), 167 flatness=1.0, 168 shadow=0.0, 169 ) 170 171 def _on_proxy_request_response( 172 self, response: bacommon.cloud.LoginProxyRequestResponse | Exception 173 ) -> None: 174 plus = bui.app.plus 175 assert plus is not None 176 177 if not self._message_in_flight: 178 logging.warning( 179 'v2proxy got _on_proxy_request_response' 180 ' without _message_in_flight set; unexpected.' 181 ) 182 self._message_in_flight = False 183 184 # Something went wrong. Show an error message and schedule retry. 185 if isinstance(response, Exception): 186 self._set_error_state(f'response exc ({type(response).__name__})') 187 self._complete = True 188 return 189 190 self._complete = True 191 192 # Clear out stuff we use to show progress/errors. 193 self._sub_state_text.delete() 194 self._sub_state_text2.delete() 195 196 # If we have overlay-web-browser functionality, bring up 197 # an inline sign-in dialog. 198 if bui.overlay_web_browser_is_supported(): 199 bui.textwidget( 200 edit=self._state_text, 201 text=bui.Lstr(resource='pleaseWaitText'), 202 ) 203 self._show_overlay_sign_in_ui(response) 204 self._overlay_web_browser_open = True 205 else: 206 # Otherwise just show link-button/qr-code for the sign-in. 207 self._state_text.delete() 208 self._show_standard_sign_in_ui(response) 209 210 # In either case, start querying for results now. 211 self._proxyid = response.proxyid 212 self._proxykey = response.proxykey 213 bui.apptimer( 214 STATUS_CHECK_INTERVAL_SECONDS, bui.WeakCall(self._ask_for_status) 215 ) 216 217 def _show_overlay_sign_in_ui( 218 self, response: bacommon.cloud.LoginProxyRequestResponse 219 ) -> None: 220 msaddress = self._get_server_address() 221 address = msaddress + response.url_overlay 222 bui.overlay_web_browser_open_url(address) 223 224 def _show_standard_sign_in_ui( 225 self, response: bacommon.cloud.LoginProxyRequestResponse 226 ) -> None: 227 msaddress = self._get_server_address() 228 229 # Show link(s) the user can use to sign in. 230 address = msaddress + response.url 231 address_pretty = address.removeprefix('https://') 232 233 assert bui.app.classic is not None 234 bui.textwidget( 235 parent=self._root_widget, 236 position=(self._width * 0.5, self._height - 95), 237 size=(0, 0), 238 text=bui.Lstr( 239 resource='accountSettingsWindow.v2LinkInstructionsText' 240 ), 241 color=bui.app.ui_v1.title_color, 242 maxwidth=self._width * 0.9, 243 h_align='center', 244 v_align='center', 245 ) 246 button_width = 450 247 if bui.is_browser_likely_available(): 248 bui.buttonwidget( 249 parent=self._root_widget, 250 position=( 251 (self._width * 0.5 - button_width * 0.5), 252 self._height - 185, 253 ), 254 autoselect=True, 255 size=(button_width, 60), 256 label=bui.Lstr(value=address_pretty), 257 color=(0.55, 0.5, 0.6), 258 textcolor=(0.75, 0.7, 0.8), 259 on_activate_call=lambda: bui.open_url(address), 260 ) 261 qroffs = 0.0 262 else: 263 bui.textwidget( 264 parent=self._root_widget, 265 position=(self._width * 0.5 - 200, self._height - 180), 266 size=(button_width - 50, 50), 267 text=bui.Lstr(value=address_pretty), 268 flatness=1.0, 269 maxwidth=self._width, 270 scale=0.75, 271 h_align='center', 272 v_align='center', 273 autoselect=True, 274 on_activate_call=bui.Call(self._copy_link, address_pretty), 275 selectable=True, 276 ) 277 qroffs = 20.0 278 279 qr_size = 270 280 bui.imagewidget( 281 parent=self._root_widget, 282 position=( 283 self._width * 0.5 - qr_size * 0.5, 284 self._height * 0.36 + qroffs - qr_size * 0.5, 285 ), 286 size=(qr_size, qr_size), 287 texture=bui.get_qrcode_texture(address), 288 ) 289 290 def _ask_for_status(self) -> None: 291 assert self._proxyid is not None 292 assert self._proxykey is not None 293 assert bui.app.plus is not None 294 bui.app.plus.cloud.send_message_cb( 295 bacommon.cloud.LoginProxyStateQueryMessage( 296 proxyid=self._proxyid, proxykey=self._proxykey 297 ), 298 on_response=bui.WeakCall(self._got_status), 299 ) 300 301 def _got_status( 302 self, response: bacommon.cloud.LoginProxyStateQueryResponse | Exception 303 ) -> None: 304 if ( 305 isinstance(response, bacommon.cloud.LoginProxyStateQueryResponse) 306 and response.state is response.State.FAIL 307 ): 308 logging.info('LoginProxy failed.') 309 bui.getsound('error').play() 310 bui.screenmessage(bui.Lstr(resource='errorText'), color=(1, 0, 0)) 311 self._done() 312 return 313 314 # If we got a token, set ourself as signed in. Hooray! 315 if ( 316 isinstance(response, bacommon.cloud.LoginProxyStateQueryResponse) 317 and response.state is response.State.SUCCESS 318 ): 319 plus = bui.app.plus 320 assert plus is not None 321 assert response.credentials is not None 322 plus.accounts.set_primary_credentials(response.credentials) 323 324 # As a courtesy, tell the server we're done with this proxy 325 # so it can clean up (not a huge deal if this fails) 326 assert self._proxyid is not None 327 try: 328 plus.cloud.send_message_cb( 329 bacommon.cloud.LoginProxyCompleteMessage( 330 proxyid=self._proxyid 331 ), 332 on_response=bui.WeakCall(self._proxy_complete_response), 333 ) 334 except CommunicationError: 335 pass 336 except Exception: 337 logging.warning( 338 'Unexpected error sending login-proxy-complete message', 339 exc_info=True, 340 ) 341 342 self._done() 343 return 344 345 # If we're still waiting, ask again soon. 346 if ( 347 isinstance(response, Exception) 348 or response.state is response.State.WAITING 349 ): 350 bui.apptimer( 351 STATUS_CHECK_INTERVAL_SECONDS, 352 bui.WeakCall(self._ask_for_status), 353 ) 354 355 def _proxy_complete_response(self, response: None | Exception) -> None: 356 del response # Not used. 357 # We could do something smart like retry on exceptions here, but 358 # this isn't critical so we'll just let anything slide. 359 360 def _copy_link(self, link: str) -> None: 361 if bui.clipboard_is_supported(): 362 bui.clipboard_set_text(link) 363 bui.screenmessage( 364 bui.Lstr(resource='copyConfirmText'), color=(0, 1, 0) 365 ) 366 367 def _done(self) -> None: 368 # no-op if our underlying widget is dead or on its way out. 369 if not self._root_widget or self._root_widget.transitioning_out: 370 return 371 372 # If we've got an inline browser up, tell it to close. 373 if self._overlay_web_browser_open: 374 bui.overlay_web_browser_close() 375 376 bui.containerwidget(edit=self._root_widget, transition='out_scale')
STATUS_CHECK_INTERVAL_SECONDS =
2.0
class
V2ProxySignInWindow(bauiv1._uitypes.Window):
18class V2ProxySignInWindow(bui.Window): 19 """A window allowing signing in to a v2 account.""" 20 21 def __init__(self, origin_widget: bui.Widget): 22 self._width = 600 23 self._height = 550 24 self._proxyid: str | None = None 25 self._proxykey: str | None = None 26 self._overlay_web_browser_open = False 27 28 assert bui.app.classic is not None 29 uiscale = bui.app.ui_v1.uiscale 30 super().__init__( 31 root_widget=bui.containerwidget( 32 size=(self._width, self._height), 33 transition='in_scale', 34 scale_origin_stack_offset=( 35 origin_widget.get_screen_space_center() 36 ), 37 scale=( 38 1.16 39 if uiscale is bui.UIScale.SMALL 40 else 1.0 if uiscale is bui.UIScale.MEDIUM else 0.9 41 ), 42 ) 43 ) 44 45 self._state_text = bui.textwidget( 46 parent=self._root_widget, 47 position=(self._width * 0.5, self._height * 0.6), 48 h_align='center', 49 v_align='center', 50 size=(0, 0), 51 scale=1.4, 52 maxwidth=0.9 * self._width, 53 text=bui.Lstr( 54 value='${A}...', 55 subs=[('${A}', bui.Lstr(resource='loadingText'))], 56 ), 57 color=(1, 1, 1), 58 ) 59 self._sub_state_text = bui.textwidget( 60 parent=self._root_widget, 61 position=(self._width * 0.5, self._height * 0.55), 62 h_align='center', 63 v_align='top', 64 scale=0.85, 65 size=(0, 0), 66 maxwidth=0.9 * self._width, 67 text='', 68 ) 69 self._sub_state_text2 = bui.textwidget( 70 parent=self._root_widget, 71 position=(self._width * 0.1, self._height * 0.3), 72 h_align='left', 73 v_align='top', 74 scale=0.7, 75 size=(0, 0), 76 maxwidth=0.9 * self._width, 77 text='', 78 ) 79 80 self._cancel_button = bui.buttonwidget( 81 parent=self._root_widget, 82 position=(30, self._height - 65), 83 size=(130, 50), 84 scale=0.8, 85 label=bui.Lstr(resource='cancelText'), 86 on_activate_call=self._done, 87 autoselect=True, 88 ) 89 90 bui.containerwidget( 91 edit=self._root_widget, cancel_button=self._cancel_button 92 ) 93 94 self._message_in_flight = False 95 self._complete = False 96 self._connection_wait_timeout_time = time.monotonic() + 10.0 97 98 self._update_timer = bui.AppTimer( 99 0.371, bui.WeakCall(self._update), repeat=True 100 ) 101 bui.pushcall(bui.WeakCall(self._update)) 102 103 def _update(self) -> None: 104 105 plus = bui.app.plus 106 assert plus is not None 107 108 # If we've opened an overlay web browser, all we do is kill 109 # ourselves when it closes. 110 if self._overlay_web_browser_open: 111 if not bui.overlay_web_browser_is_open(): 112 self._overlay_web_browser_open = False 113 self._done() 114 return 115 116 if self._message_in_flight or self._complete: 117 return 118 119 now = time.monotonic() 120 121 # Spin for a moment if it looks like we have no server 122 # connection; it might still be getting on its feet. 123 if ( 124 not plus.cloud.connected 125 and now < self._connection_wait_timeout_time 126 ): 127 return 128 129 plus.cloud.send_message_cb( 130 bacommon.cloud.LoginProxyRequestMessage(), 131 on_response=bui.WeakCall(self._on_proxy_request_response), 132 ) 133 self._message_in_flight = True 134 135 def _get_server_address(self) -> str: 136 plus = bui.app.plus 137 assert plus is not None 138 out = plus.get_master_server_address(version=2) 139 assert isinstance(out, str) 140 return out 141 142 def _set_error_state(self, error_location: str) -> None: 143 msaddress = self._get_server_address() 144 addr = msaddress.removeprefix('https://') 145 bui.textwidget( 146 edit=self._state_text, 147 text=f'Unable to connect to {addr}.', 148 color=(1, 0, 0), 149 ) 150 support_email = 'support@froemling.net' 151 bui.textwidget( 152 edit=self._sub_state_text, 153 text=( 154 f'Usually this means your internet is down.\n' 155 f'Please contact {support_email} if this is not the case.' 156 ), 157 color=(1, 0, 0), 158 ) 159 bui.textwidget( 160 edit=self._sub_state_text2, 161 text=( 162 f'debug-info:\n' 163 f' error-location: {error_location}\n' 164 f' connectivity: {bui.app.net.connectivity_state}\n' 165 f' transport: {bui.app.net.transport_state}' 166 ), 167 color=(0.8, 0.2, 0.3), 168 flatness=1.0, 169 shadow=0.0, 170 ) 171 172 def _on_proxy_request_response( 173 self, response: bacommon.cloud.LoginProxyRequestResponse | Exception 174 ) -> None: 175 plus = bui.app.plus 176 assert plus is not None 177 178 if not self._message_in_flight: 179 logging.warning( 180 'v2proxy got _on_proxy_request_response' 181 ' without _message_in_flight set; unexpected.' 182 ) 183 self._message_in_flight = False 184 185 # Something went wrong. Show an error message and schedule retry. 186 if isinstance(response, Exception): 187 self._set_error_state(f'response exc ({type(response).__name__})') 188 self._complete = True 189 return 190 191 self._complete = True 192 193 # Clear out stuff we use to show progress/errors. 194 self._sub_state_text.delete() 195 self._sub_state_text2.delete() 196 197 # If we have overlay-web-browser functionality, bring up 198 # an inline sign-in dialog. 199 if bui.overlay_web_browser_is_supported(): 200 bui.textwidget( 201 edit=self._state_text, 202 text=bui.Lstr(resource='pleaseWaitText'), 203 ) 204 self._show_overlay_sign_in_ui(response) 205 self._overlay_web_browser_open = True 206 else: 207 # Otherwise just show link-button/qr-code for the sign-in. 208 self._state_text.delete() 209 self._show_standard_sign_in_ui(response) 210 211 # In either case, start querying for results now. 212 self._proxyid = response.proxyid 213 self._proxykey = response.proxykey 214 bui.apptimer( 215 STATUS_CHECK_INTERVAL_SECONDS, bui.WeakCall(self._ask_for_status) 216 ) 217 218 def _show_overlay_sign_in_ui( 219 self, response: bacommon.cloud.LoginProxyRequestResponse 220 ) -> None: 221 msaddress = self._get_server_address() 222 address = msaddress + response.url_overlay 223 bui.overlay_web_browser_open_url(address) 224 225 def _show_standard_sign_in_ui( 226 self, response: bacommon.cloud.LoginProxyRequestResponse 227 ) -> None: 228 msaddress = self._get_server_address() 229 230 # Show link(s) the user can use to sign in. 231 address = msaddress + response.url 232 address_pretty = address.removeprefix('https://') 233 234 assert bui.app.classic is not None 235 bui.textwidget( 236 parent=self._root_widget, 237 position=(self._width * 0.5, self._height - 95), 238 size=(0, 0), 239 text=bui.Lstr( 240 resource='accountSettingsWindow.v2LinkInstructionsText' 241 ), 242 color=bui.app.ui_v1.title_color, 243 maxwidth=self._width * 0.9, 244 h_align='center', 245 v_align='center', 246 ) 247 button_width = 450 248 if bui.is_browser_likely_available(): 249 bui.buttonwidget( 250 parent=self._root_widget, 251 position=( 252 (self._width * 0.5 - button_width * 0.5), 253 self._height - 185, 254 ), 255 autoselect=True, 256 size=(button_width, 60), 257 label=bui.Lstr(value=address_pretty), 258 color=(0.55, 0.5, 0.6), 259 textcolor=(0.75, 0.7, 0.8), 260 on_activate_call=lambda: bui.open_url(address), 261 ) 262 qroffs = 0.0 263 else: 264 bui.textwidget( 265 parent=self._root_widget, 266 position=(self._width * 0.5 - 200, self._height - 180), 267 size=(button_width - 50, 50), 268 text=bui.Lstr(value=address_pretty), 269 flatness=1.0, 270 maxwidth=self._width, 271 scale=0.75, 272 h_align='center', 273 v_align='center', 274 autoselect=True, 275 on_activate_call=bui.Call(self._copy_link, address_pretty), 276 selectable=True, 277 ) 278 qroffs = 20.0 279 280 qr_size = 270 281 bui.imagewidget( 282 parent=self._root_widget, 283 position=( 284 self._width * 0.5 - qr_size * 0.5, 285 self._height * 0.36 + qroffs - qr_size * 0.5, 286 ), 287 size=(qr_size, qr_size), 288 texture=bui.get_qrcode_texture(address), 289 ) 290 291 def _ask_for_status(self) -> None: 292 assert self._proxyid is not None 293 assert self._proxykey is not None 294 assert bui.app.plus is not None 295 bui.app.plus.cloud.send_message_cb( 296 bacommon.cloud.LoginProxyStateQueryMessage( 297 proxyid=self._proxyid, proxykey=self._proxykey 298 ), 299 on_response=bui.WeakCall(self._got_status), 300 ) 301 302 def _got_status( 303 self, response: bacommon.cloud.LoginProxyStateQueryResponse | Exception 304 ) -> None: 305 if ( 306 isinstance(response, bacommon.cloud.LoginProxyStateQueryResponse) 307 and response.state is response.State.FAIL 308 ): 309 logging.info('LoginProxy failed.') 310 bui.getsound('error').play() 311 bui.screenmessage(bui.Lstr(resource='errorText'), color=(1, 0, 0)) 312 self._done() 313 return 314 315 # If we got a token, set ourself as signed in. Hooray! 316 if ( 317 isinstance(response, bacommon.cloud.LoginProxyStateQueryResponse) 318 and response.state is response.State.SUCCESS 319 ): 320 plus = bui.app.plus 321 assert plus is not None 322 assert response.credentials is not None 323 plus.accounts.set_primary_credentials(response.credentials) 324 325 # As a courtesy, tell the server we're done with this proxy 326 # so it can clean up (not a huge deal if this fails) 327 assert self._proxyid is not None 328 try: 329 plus.cloud.send_message_cb( 330 bacommon.cloud.LoginProxyCompleteMessage( 331 proxyid=self._proxyid 332 ), 333 on_response=bui.WeakCall(self._proxy_complete_response), 334 ) 335 except CommunicationError: 336 pass 337 except Exception: 338 logging.warning( 339 'Unexpected error sending login-proxy-complete message', 340 exc_info=True, 341 ) 342 343 self._done() 344 return 345 346 # If we're still waiting, ask again soon. 347 if ( 348 isinstance(response, Exception) 349 or response.state is response.State.WAITING 350 ): 351 bui.apptimer( 352 STATUS_CHECK_INTERVAL_SECONDS, 353 bui.WeakCall(self._ask_for_status), 354 ) 355 356 def _proxy_complete_response(self, response: None | Exception) -> None: 357 del response # Not used. 358 # We could do something smart like retry on exceptions here, but 359 # this isn't critical so we'll just let anything slide. 360 361 def _copy_link(self, link: str) -> None: 362 if bui.clipboard_is_supported(): 363 bui.clipboard_set_text(link) 364 bui.screenmessage( 365 bui.Lstr(resource='copyConfirmText'), color=(0, 1, 0) 366 ) 367 368 def _done(self) -> None: 369 # no-op if our underlying widget is dead or on its way out. 370 if not self._root_widget or self._root_widget.transitioning_out: 371 return 372 373 # If we've got an inline browser up, tell it to close. 374 if self._overlay_web_browser_open: 375 bui.overlay_web_browser_close() 376 377 bui.containerwidget(edit=self._root_widget, transition='out_scale')
A window allowing signing in to a v2 account.
V2ProxySignInWindow(origin_widget: _bauiv1.Widget)
21 def __init__(self, origin_widget: bui.Widget): 22 self._width = 600 23 self._height = 550 24 self._proxyid: str | None = None 25 self._proxykey: str | None = None 26 self._overlay_web_browser_open = False 27 28 assert bui.app.classic is not None 29 uiscale = bui.app.ui_v1.uiscale 30 super().__init__( 31 root_widget=bui.containerwidget( 32 size=(self._width, self._height), 33 transition='in_scale', 34 scale_origin_stack_offset=( 35 origin_widget.get_screen_space_center() 36 ), 37 scale=( 38 1.16 39 if uiscale is bui.UIScale.SMALL 40 else 1.0 if uiscale is bui.UIScale.MEDIUM else 0.9 41 ), 42 ) 43 ) 44 45 self._state_text = bui.textwidget( 46 parent=self._root_widget, 47 position=(self._width * 0.5, self._height * 0.6), 48 h_align='center', 49 v_align='center', 50 size=(0, 0), 51 scale=1.4, 52 maxwidth=0.9 * self._width, 53 text=bui.Lstr( 54 value='${A}...', 55 subs=[('${A}', bui.Lstr(resource='loadingText'))], 56 ), 57 color=(1, 1, 1), 58 ) 59 self._sub_state_text = bui.textwidget( 60 parent=self._root_widget, 61 position=(self._width * 0.5, self._height * 0.55), 62 h_align='center', 63 v_align='top', 64 scale=0.85, 65 size=(0, 0), 66 maxwidth=0.9 * self._width, 67 text='', 68 ) 69 self._sub_state_text2 = bui.textwidget( 70 parent=self._root_widget, 71 position=(self._width * 0.1, self._height * 0.3), 72 h_align='left', 73 v_align='top', 74 scale=0.7, 75 size=(0, 0), 76 maxwidth=0.9 * self._width, 77 text='', 78 ) 79 80 self._cancel_button = bui.buttonwidget( 81 parent=self._root_widget, 82 position=(30, self._height - 65), 83 size=(130, 50), 84 scale=0.8, 85 label=bui.Lstr(resource='cancelText'), 86 on_activate_call=self._done, 87 autoselect=True, 88 ) 89 90 bui.containerwidget( 91 edit=self._root_widget, cancel_button=self._cancel_button 92 ) 93 94 self._message_in_flight = False 95 self._complete = False 96 self._connection_wait_timeout_time = time.monotonic() + 10.0 97 98 self._update_timer = bui.AppTimer( 99 0.371, bui.WeakCall(self._update), repeat=True 100 ) 101 bui.pushcall(bui.WeakCall(self._update))