bauiv1lib.gettokens
UI functionality for purchasing/acquiring currency.
1# Released under the MIT License. See LICENSE for details. 2# 3"""UI functionality for purchasing/acquiring currency.""" 4 5from __future__ import annotations 6 7import time 8from enum import Enum 9from functools import partial 10from dataclasses import dataclass 11from typing import TYPE_CHECKING, assert_never, override 12 13import bacommon.cloud 14import bauiv1 as bui 15 16 17if TYPE_CHECKING: 18 from typing import Any, Callable 19 20 21@dataclass 22class _ButtonDef: 23 itemid: str 24 width: float 25 color: tuple[float, float, float] 26 imgdefs: list[_ImgDef] 27 txtdefs: list[_TxtDef] 28 prepad: float = 0.0 29 30 31@dataclass 32class _ImgDef: 33 tex: str 34 pos: tuple[float, float] 35 size: tuple[float, float] 36 color: tuple[float, float, float] = (1, 1, 1) 37 opacity: float = 1.0 38 draw_controller_mult: float | None = None 39 40 41class TextContents(Enum): 42 """Some type of text to show.""" 43 44 PRICE = 'price' 45 46 47@dataclass 48class _TxtDef: 49 text: str | TextContents | bui.Lstr 50 pos: tuple[float, float] 51 maxwidth: float | None 52 scale: float = 1.0 53 color: tuple[float, float, float] = (1, 1, 1) 54 rotate: float | None = None 55 56 57class GetTokensWindow(bui.MainWindow): 58 """Window for purchasing/acquiring classic tickets.""" 59 60 class State(Enum): 61 """What are we doing?""" 62 63 LOADING = 'loading' 64 NOT_SIGNED_IN = 'not_signed_in' 65 HAVE_GOLD_PASS = 'have_gold_pass' 66 SHOWING_STORE = 'showing_store' 67 68 def __init__( 69 self, 70 transition: str | None = 'in_right', 71 origin_widget: bui.Widget | None = None, 72 # restore_previous_call: Callable[[bui.Widget], None] | None = None, 73 ): 74 bwidthstd = 170 75 bwidthwide = 300 76 ycolor = (0, 0, 0.3) 77 pcolor = (0, 0, 0.3) 78 pos1 = 65 79 pos2 = 34 80 titlescale = 0.9 81 pricescale = 0.65 82 bcapcol1 = (0.25, 0.13, 0.02) 83 self._buttondefs: list[_ButtonDef] = [ 84 _ButtonDef( 85 itemid='tokens1', 86 width=bwidthstd, 87 color=ycolor, 88 imgdefs=[ 89 _ImgDef( 90 'tokens1', 91 pos=(-3, 85), 92 size=(172, 172), 93 opacity=1.0, 94 draw_controller_mult=0.5, 95 ), 96 _ImgDef( 97 'windowBottomCap', 98 pos=(1.5, 4), 99 size=(bwidthstd * 0.960, 100), 100 color=bcapcol1, 101 opacity=1.0, 102 ), 103 ], 104 txtdefs=[ 105 _TxtDef( 106 bui.Lstr( 107 resource='tokens.numTokensText', 108 subs=[('${COUNT}', '50')], 109 ), 110 pos=(bwidthstd * 0.5, pos1), 111 color=(1.1, 1.05, 1.0), 112 scale=titlescale, 113 maxwidth=bwidthstd * 0.9, 114 ), 115 _TxtDef( 116 TextContents.PRICE, 117 pos=(bwidthstd * 0.5, pos2), 118 color=(1.1, 1.05, 1.0), 119 scale=pricescale, 120 maxwidth=bwidthstd * 0.9, 121 ), 122 ], 123 ), 124 _ButtonDef( 125 itemid='tokens2', 126 width=bwidthstd, 127 color=ycolor, 128 imgdefs=[ 129 _ImgDef( 130 'tokens2', 131 pos=(-3, 85), 132 size=(172, 172), 133 opacity=1.0, 134 draw_controller_mult=0.5, 135 ), 136 _ImgDef( 137 'windowBottomCap', 138 pos=(1.5, 4), 139 size=(bwidthstd * 0.960, 100), 140 color=bcapcol1, 141 opacity=1.0, 142 ), 143 ], 144 txtdefs=[ 145 _TxtDef( 146 bui.Lstr( 147 resource='tokens.numTokensText', 148 subs=[('${COUNT}', '500')], 149 ), 150 pos=(bwidthstd * 0.5, pos1), 151 color=(1.1, 1.05, 1.0), 152 scale=titlescale, 153 maxwidth=bwidthstd * 0.9, 154 ), 155 _TxtDef( 156 TextContents.PRICE, 157 pos=(bwidthstd * 0.5, pos2), 158 color=(1.1, 1.05, 1.0), 159 scale=pricescale, 160 maxwidth=bwidthstd * 0.9, 161 ), 162 ], 163 ), 164 _ButtonDef( 165 itemid='tokens3', 166 width=bwidthstd, 167 color=ycolor, 168 imgdefs=[ 169 _ImgDef( 170 'tokens3', 171 pos=(-3, 85), 172 size=(172, 172), 173 opacity=1.0, 174 draw_controller_mult=0.5, 175 ), 176 _ImgDef( 177 'windowBottomCap', 178 pos=(1.5, 4), 179 size=(bwidthstd * 0.960, 100), 180 color=bcapcol1, 181 opacity=1.0, 182 ), 183 ], 184 txtdefs=[ 185 _TxtDef( 186 bui.Lstr( 187 resource='tokens.numTokensText', 188 subs=[('${COUNT}', '1200')], 189 ), 190 pos=(bwidthstd * 0.5, pos1), 191 color=(1.1, 1.05, 1.0), 192 scale=titlescale, 193 maxwidth=bwidthstd * 0.9, 194 ), 195 _TxtDef( 196 TextContents.PRICE, 197 pos=(bwidthstd * 0.5, pos2), 198 color=(1.1, 1.05, 1.0), 199 scale=pricescale, 200 maxwidth=bwidthstd * 0.9, 201 ), 202 ], 203 ), 204 _ButtonDef( 205 itemid='tokens4', 206 width=bwidthstd, 207 color=ycolor, 208 imgdefs=[ 209 _ImgDef( 210 'tokens4', 211 pos=(-3, 85), 212 size=(172, 172), 213 opacity=1.0, 214 draw_controller_mult=0.5, 215 ), 216 _ImgDef( 217 'windowBottomCap', 218 pos=(1.5, 4), 219 size=(bwidthstd * 0.960, 100), 220 color=bcapcol1, 221 opacity=1.0, 222 ), 223 ], 224 txtdefs=[ 225 _TxtDef( 226 bui.Lstr( 227 resource='tokens.numTokensText', 228 subs=[('${COUNT}', '2600')], 229 ), 230 pos=(bwidthstd * 0.5, pos1), 231 color=(1.1, 1.05, 1.0), 232 scale=titlescale, 233 maxwidth=bwidthstd * 0.9, 234 ), 235 _TxtDef( 236 TextContents.PRICE, 237 pos=(bwidthstd * 0.5, pos2), 238 color=(1.1, 1.05, 1.0), 239 scale=pricescale, 240 maxwidth=bwidthstd * 0.9, 241 ), 242 ], 243 ), 244 _ButtonDef( 245 itemid='gold_pass', 246 width=bwidthwide, 247 color=pcolor, 248 imgdefs=[ 249 _ImgDef( 250 'goldPass', 251 pos=(-7, 102), 252 size=(312, 156), 253 draw_controller_mult=0.3, 254 ), 255 _ImgDef( 256 'windowBottomCap', 257 pos=(8, 4), 258 size=(bwidthwide * 0.923, 116), 259 color=(0.25, 0.12, 0.15), 260 opacity=1.0, 261 ), 262 ], 263 txtdefs=[ 264 _TxtDef( 265 bui.Lstr(resource='goldPass.goldPassText'), 266 pos=(bwidthwide * 0.5, pos1 + 27), 267 color=(1.1, 1.05, 1.0), 268 scale=titlescale, 269 maxwidth=bwidthwide * 0.8, 270 ), 271 _TxtDef( 272 bui.Lstr(resource='goldPass.desc1InfTokensText'), 273 pos=(bwidthwide * 0.5, pos1 + 6), 274 color=(1.1, 1.05, 1.0), 275 scale=0.4, 276 maxwidth=bwidthwide * 0.8, 277 ), 278 _TxtDef( 279 bui.Lstr(resource='goldPass.desc2NoAdsText'), 280 pos=(bwidthwide * 0.5, pos1 + 6 - 13 * 1), 281 color=(1.1, 1.05, 1.0), 282 scale=0.4, 283 maxwidth=bwidthwide * 0.8, 284 ), 285 _TxtDef( 286 bui.Lstr(resource='goldPass.desc3ForeverText'), 287 pos=(bwidthwide * 0.5, pos1 + 6 - 13 * 2), 288 color=(1.1, 1.05, 1.0), 289 scale=0.4, 290 maxwidth=bwidthwide * 0.8, 291 ), 292 _TxtDef( 293 TextContents.PRICE, 294 pos=(bwidthwide * 0.5, pos2 - 9), 295 color=(1.1, 1.05, 1.0), 296 scale=pricescale, 297 maxwidth=bwidthwide * 0.8, 298 ), 299 ], 300 prepad=-8, 301 ), 302 ] 303 304 self._transitioning_out = False 305 # self._restore_previous_call = restore_previous_call 306 self._textcolor = (0.92, 0.92, 2.0) 307 308 self._query_in_flight = False 309 self._last_query_time = -1.0 310 self._last_query_response: bacommon.cloud.StoreQueryResponse | None = ( 311 None 312 ) 313 314 # If they provided an origin-widget, scale up from that. 315 # scale_origin: tuple[float, float] | None 316 # if origin_widget is not None: 317 # self._transition_out = 'out_scale' 318 # scale_origin = origin_widget.get_screen_space_center() 319 # transition = 'in_scale' 320 # else: 321 # self._transition_out = 'out_right' 322 # scale_origin = None 323 324 uiscale = bui.app.ui_v1.uiscale 325 self._width = 1000.0 if uiscale is bui.UIScale.SMALL else 800.0 326 self._x_inset = 25.0 if uiscale is bui.UIScale.SMALL else 0.0 327 self._height = 550 if uiscale is bui.UIScale.SMALL else 480.0 328 self._y_offset = -60 if uiscale is bui.UIScale.SMALL else 0 329 330 self._r = 'getTokensWindow' 331 332 super().__init__( 333 root_widget=bui.containerwidget( 334 size=(self._width, self._height), 335 # transition=transition, 336 # scale_origin_stack_offset=scale_origin, 337 color=(0.3, 0.23, 0.36), 338 scale=( 339 1.5 340 if uiscale is bui.UIScale.SMALL 341 else 1.2 if uiscale is bui.UIScale.MEDIUM else 1.0 342 ), 343 stack_offset=( 344 (0, -3) if uiscale is bui.UIScale.SMALL else (0, 0) 345 ), 346 # toolbar_visibility='menu_minimal', 347 toolbar_visibility=( 348 'get_tokens' 349 if uiscale is bui.UIScale.SMALL 350 else 'menu_full' 351 ), 352 ), 353 transition=transition, 354 origin_widget=origin_widget, 355 ) 356 357 if uiscale is bui.UIScale.SMALL: 358 bui.containerwidget( 359 edit=self._root_widget, on_cancel_call=self.main_window_back 360 ) 361 self._back_button = bui.get_special_widget('back_button') 362 else: 363 self._back_button = bui.buttonwidget( 364 parent=self._root_widget, 365 position=( 366 55 + self._x_inset, 367 self._height - 80 + self._y_offset, 368 ), 369 size=( 370 # (140, 60) 371 # if self._restore_previous_call is None 372 # else 373 (60, 60) 374 ), 375 scale=1.0, 376 autoselect=True, 377 label=( 378 # bui.Lstr(resource='doneText') 379 # if self._restore_previous_call is None 380 # else 381 bui.charstr(bui.SpecialChar.BACK) 382 ), 383 button_type=( 384 # 'regular' 385 # if self._restore_previous_call is None 386 # else 387 'backSmall' 388 ), 389 on_activate_call=self.main_window_back, 390 ) 391 # if uiscale is bui.UIScale.SMALL: 392 # bui.widget( 393 # edit=self._back_button, 394 # up_widget=bui.get_special_widget('tokens_meter'), 395 # ) 396 bui.containerwidget( 397 edit=self._root_widget, cancel_button=self._back_button 398 ) 399 400 self._title_text = bui.textwidget( 401 parent=self._root_widget, 402 position=(self._width * 0.5, self._height - 42 + self._y_offset), 403 size=(0, 0), 404 color=self._textcolor, 405 flatness=0.0, 406 shadow=1.0, 407 scale=1.2, 408 h_align='center', 409 v_align='center', 410 text=bui.Lstr(resource='tokens.getTokensText'), 411 maxwidth=260, 412 ) 413 414 self._status_text = bui.textwidget( 415 parent=self._root_widget, 416 size=(0, 0), 417 position=(self._width * 0.5, self._height * 0.5), 418 h_align='center', 419 v_align='center', 420 color=(0.6, 0.6, 0.6), 421 scale=0.75, 422 text=bui.Lstr(resource='store.loadingText'), 423 ) 424 425 self._core_widgets = [ 426 self._back_button, 427 self._title_text, 428 self._status_text, 429 ] 430 431 # self._token_count_widget: bui.Widget | None = None 432 # self._smooth_update_timer: bui.AppTimer | None = None 433 # self._smooth_token_count: float | None = None 434 # self._token_count: int = 0 435 # self._smooth_increase_speed = 1.0 436 # self._ticking_sound: bui.Sound | None = None 437 438 # Get all textures used by our buttons preloading so hopefully 439 # they'll be in place by the time we show them. 440 for bdef in self._buttondefs: 441 for bimg in bdef.imgdefs: 442 bui.gettexture(bimg.tex) 443 444 self._state = self.State.LOADING 445 446 self._update_timer = bui.AppTimer( 447 0.789, bui.WeakCall(self._update), repeat=True 448 ) 449 self._update() 450 451 # def __del__(self) -> None: 452 # if self._ticking_sound is not None: 453 # self._ticking_sound.stop() 454 # self._ticking_sound = None 455 456 @override 457 def get_main_window_state(self) -> bui.MainWindowState: 458 # Support recreating our window for back/refresh purposes. 459 cls = type(self) 460 return bui.BasicMainWindowState( 461 create_call=lambda transition, origin_widget: cls( 462 transition=transition, origin_widget=origin_widget 463 ) 464 ) 465 466 def _update(self) -> None: 467 # No-op if our underlying widget is dead or on its way out. 468 if not self._root_widget or self._root_widget.transitioning_out: 469 return 470 471 plus = bui.app.plus 472 473 if plus is None or plus.accounts.primary is None: 474 self._update_state(self.State.NOT_SIGNED_IN) 475 return 476 477 # Poll for relevant changes to the store or our account. 478 now = time.monotonic() 479 if not self._query_in_flight and now - self._last_query_time > 2.0: 480 self._last_query_time = now 481 self._query_in_flight = True 482 with plus.accounts.primary: 483 plus.cloud.send_message_cb( 484 bacommon.cloud.StoreQueryMessage(), 485 on_response=bui.WeakCall(self._on_store_query_response), 486 ) 487 488 # Can't do much until we get a store state. 489 if self._last_query_response is None: 490 return 491 492 # If we've got a gold-pass, just show that. No need to offer any 493 # other purchases. 494 if self._last_query_response.gold_pass: 495 self._update_state(self.State.HAVE_GOLD_PASS) 496 return 497 498 # Ok we seem to be signed in and have store stuff we can show. 499 # Do that. 500 self._update_state(self.State.SHOWING_STORE) 501 502 def _update_state(self, state: State) -> None: 503 504 # We don't do much when state is unchanged. 505 if state is self._state: 506 # Update a few things in store mode though, such as token 507 # count. 508 if state is self.State.SHOWING_STORE: 509 self._update_store_state() 510 return 511 512 # Ok, state is changing. Start by resetting to a blank slate. 513 # self._token_count_widget = None 514 for widget in self._root_widget.get_children(): 515 if widget not in self._core_widgets: 516 widget.delete() 517 518 # Build up new state. 519 if state is self.State.NOT_SIGNED_IN: 520 bui.textwidget( 521 edit=self._status_text, 522 color=(1, 0, 0), 523 text=bui.Lstr(resource='notSignedInErrorText'), 524 ) 525 elif state is self.State.LOADING: 526 raise RuntimeError('Should never return to loading state.') 527 elif state is self.State.HAVE_GOLD_PASS: 528 bui.textwidget( 529 edit=self._status_text, 530 color=(0, 1, 0), 531 text=bui.Lstr(resource='tokens.youHaveGoldPassText'), 532 ) 533 elif state is self.State.SHOWING_STORE: 534 assert self._last_query_response is not None 535 bui.textwidget(edit=self._status_text, text='') 536 self._build_store_for_response(self._last_query_response) 537 else: 538 # Make sure we handle all cases. 539 assert_never(state) 540 541 self._state = state 542 543 def _on_load_error(self) -> None: 544 bui.textwidget( 545 edit=self._status_text, 546 text=bui.Lstr(resource='internal.unavailableNoConnectionText'), 547 color=(1, 0, 0), 548 ) 549 550 def _on_store_query_response( 551 self, response: bacommon.cloud.StoreQueryResponse | Exception 552 ) -> None: 553 self._query_in_flight = False 554 if isinstance(response, bacommon.cloud.StoreQueryResponse): 555 self._last_query_response = response 556 # Hurry along any effects of this response. 557 self._update() 558 559 def _build_store_for_response( 560 self, response: bacommon.cloud.StoreQueryResponse 561 ) -> None: 562 # pylint: disable=too-many-locals 563 plus = bui.app.plus 564 565 uiscale = bui.app.ui_v1.uiscale 566 567 bui.textwidget(edit=self._status_text, text='') 568 569 xinset = 40 570 571 scrollwidth = self._width - 2 * (self._x_inset + xinset) 572 scrollheight = 280 573 buttonpadding = -5 574 575 yoffs = 5 576 577 # We currently don't handle the zero-button case. 578 assert self._buttondefs 579 580 sidepad = 10.0 581 total_button_width = ( 582 sum(b.width + b.prepad for b in self._buttondefs) 583 + buttonpadding * (len(self._buttondefs) - 1) 584 + 2 * sidepad 585 ) 586 587 h_scroll = bui.hscrollwidget( 588 parent=self._root_widget, 589 size=(scrollwidth, scrollheight), 590 position=( 591 self._x_inset + xinset, 592 self._height - 415 + self._y_offset, 593 ), 594 claims_left_right=True, 595 highlight=False, 596 border_opacity=0.3 if uiscale is bui.UIScale.SMALL else 1.0, 597 ) 598 subcontainer = bui.containerwidget( 599 parent=h_scroll, 600 background=False, 601 size=(max(total_button_width, scrollwidth), scrollheight), 602 ) 603 tinfobtn = bui.buttonwidget( 604 parent=self._root_widget, 605 autoselect=True, 606 label=bui.Lstr(resource='learnMoreText'), 607 position=( 608 self._width * 0.5 - 75, 609 self._height - 125 + self._y_offset, 610 ), 611 size=(180, 43), 612 scale=0.8, 613 color=(0.4, 0.25, 0.5), 614 textcolor=self._textcolor, 615 on_activate_call=partial( 616 self._on_learn_more_press, response.token_info_url 617 ), 618 ) 619 if uiscale is bui.UIScale.SMALL: 620 bui.widget( 621 edit=tinfobtn, 622 left_widget=bui.get_special_widget('back_button'), 623 up_widget=bui.get_special_widget('back_button'), 624 ) 625 626 bui.widget( 627 edit=tinfobtn, 628 right_widget=bui.get_special_widget('tokens_meter'), 629 ) 630 631 x = sidepad 632 bwidgets: list[bui.Widget] = [] 633 for i, buttondef in enumerate(self._buttondefs): 634 635 price = None if plus is None else plus.get_price(buttondef.itemid) 636 637 x += buttondef.prepad 638 tdelay = 0.3 - i / len(self._buttondefs) * 0.25 639 btn = bui.buttonwidget( 640 autoselect=True, 641 label='', 642 color=buttondef.color, 643 transition_delay=tdelay, 644 up_widget=tinfobtn, 645 parent=subcontainer, 646 size=(buttondef.width, 275), 647 position=(x, -10 + yoffs), 648 button_type='square', 649 on_activate_call=partial( 650 self._purchase_press, buttondef.itemid 651 ), 652 ) 653 bwidgets.append(btn) 654 655 if i == 0: 656 bui.widget(edit=btn, left_widget=self._back_button) 657 658 for imgdef in buttondef.imgdefs: 659 _img = bui.imagewidget( 660 parent=subcontainer, 661 size=imgdef.size, 662 position=(x + imgdef.pos[0], imgdef.pos[1] + yoffs), 663 draw_controller=btn, 664 draw_controller_mult=imgdef.draw_controller_mult, 665 color=imgdef.color, 666 texture=bui.gettexture(imgdef.tex), 667 transition_delay=tdelay, 668 opacity=imgdef.opacity, 669 ) 670 for txtdef in buttondef.txtdefs: 671 txt: bui.Lstr | str 672 if isinstance(txtdef.text, TextContents): 673 if txtdef.text is TextContents.PRICE: 674 tcolor = ( 675 (1, 1, 1, 0.5) if price is None else txtdef.color 676 ) 677 txt = ( 678 bui.Lstr(resource='unavailableText') 679 if price is None 680 else price 681 ) 682 else: 683 # Make sure we cover all cases. 684 assert_never(txtdef.text) 685 else: 686 tcolor = txtdef.color 687 txt = txtdef.text 688 _txt = bui.textwidget( 689 parent=subcontainer, 690 text=txt, 691 position=(x + txtdef.pos[0], txtdef.pos[1] + yoffs), 692 size=(0, 0), 693 scale=txtdef.scale, 694 h_align='center', 695 v_align='center', 696 draw_controller=btn, 697 color=tcolor, 698 transition_delay=tdelay, 699 flatness=0.0, 700 shadow=1.0, 701 rotate=txtdef.rotate, 702 maxwidth=txtdef.maxwidth, 703 ) 704 x += buttondef.width + buttonpadding 705 bui.containerwidget(edit=subcontainer, visible_child=bwidgets[0]) 706 707 _tinfotxt = bui.textwidget( 708 parent=self._root_widget, 709 position=(self._width * 0.5, self._height - 70 + self._y_offset), 710 color=self._textcolor, 711 shadow=1.0, 712 scale=0.7, 713 size=(0, 0), 714 h_align='center', 715 v_align='center', 716 text=bui.Lstr(resource='tokens.shinyNewCurrencyText'), 717 ) 718 # self._token_count_widget = bui.textwidget( 719 # parent=self._root_widget, 720 # position=( 721 # self._width - self._x_inset - 120.0, 722 # self._height - 48 + self._y_offset, 723 # ), 724 # color=(2.0, 0.7, 0.0), 725 # shadow=1.0, 726 # flatness=0.0, 727 # size=(0, 0), 728 # h_align='left', 729 # v_align='center', 730 # text='', 731 # ) 732 # self._token_count = response.tokens 733 # self._smooth_token_count = float(self._token_count) 734 # self._smooth_update() # will set the text widget. 735 736 # _tlabeltxt = bui.textwidget( 737 # parent=self._root_widget, 738 # position=( 739 # self._width - self._x_inset - 123.0, 740 # self._height - 48 + self._y_offset, 741 # ), 742 # size=(0, 0), 743 # h_align='right', 744 # v_align='center', 745 # text=bui.charstr(bui.SpecialChar.TOKEN), 746 # ) 747 748 def _purchase_press(self, itemid: str) -> None: 749 plus = bui.app.plus 750 751 price = None if plus is None else plus.get_price(itemid) 752 753 if price is None: 754 if plus is not None and plus.supports_purchases(): 755 # Looks like internet is down or something temporary. 756 errmsg = bui.Lstr(resource='purchaseNotAvailableText') 757 else: 758 # Looks like purchases will never work here. 759 errmsg = bui.Lstr(resource='purchaseNeverAvailableText') 760 761 bui.screenmessage(errmsg, color=(1, 0.5, 0)) 762 bui.getsound('error').play() 763 return 764 765 assert plus is not None 766 plus.purchase(itemid) 767 768 def _update_store_state(self) -> None: 769 """Called to make minor updates to an already shown store.""" 770 # assert self._token_count_widget is not None 771 assert self._last_query_response is not None 772 773 # self._token_count = self._last_query_response.tokens 774 775 # Kick off new smooth update if need be. 776 # assert self._smooth_token_count is not None 777 # if ( 778 # self._token_count != int(self._smooth_token_count) 779 # and self._smooth_update_timer is None 780 # ): 781 # self._smooth_update_timer = bui.AppTimer( 782 # 0.05, bui.WeakCall(self._smooth_update), repeat=True 783 # ) 784 # diff = abs(float(self._token_count) - self._smooth_token_count) 785 # self._smooth_increase_speed = ( 786 # diff / 100.0 787 # if diff >= 5000 788 # else ( 789 # diff / 50.0 790 # if diff >= 1500 791 # else diff / 30.0 if diff >= 500 else diff / 15.0 792 # ) 793 # ) 794 795 # def _smooth_update(self) -> None: 796 797 # # Stop if the count widget disappears. 798 # if not self._token_count_widget: 799 # self._smooth_update_timer = None 800 # return 801 802 # finished = False 803 804 # # If we're going down, do it immediately. 805 # assert self._smooth_token_count is not None 806 # if int(self._smooth_token_count) >= self._token_count: 807 # self._smooth_token_count = float(self._token_count) 808 # finished = True 809 # else: 810 # # We're going up; start a sound if need be. 811 # self._smooth_token_count = min( 812 # self._smooth_token_count + 1.0 * self._smooth_increase_speed, 813 # self._token_count, 814 # ) 815 # if int(self._smooth_token_count) >= self._token_count: 816 # finished = True 817 # self._smooth_token_count = float(self._token_count) 818 # elif self._ticking_sound is None: 819 # self._ticking_sound = bui.getsound('scoreIncrease') 820 # self._ticking_sound.play() 821 822 # bui.textwidget( 823 # edit=self._token_count_widget, 824 # text=str(int(self._smooth_token_count)), 825 # ) 826 827 # # If we've reached the target, kill the timer/sound/etc. 828 # if finished: 829 # self._smooth_update_timer = None 830 # if self._ticking_sound is not None: 831 # self._ticking_sound.stop() 832 # self._ticking_sound = None 833 # bui.getsound('cashRegister2').play() 834 835 # def _back(self) -> None: 836 837 # self.main_ 838 # No-op if our underlying widget is dead or on its way out. 839 # if not self._root_widget or self._root_widget.transitioning_out: 840 # return 841 842 # bui.containerwidget( 843 # edit=self._root_widget, transition=self._transition_out 844 # ) 845 # if self._restore_previous_call is not None: 846 # self._restore_previous_call(self._root_widget) 847 848 def _on_learn_more_press(self, url: str) -> None: 849 bui.open_url(url) 850 851 852def show_get_tokens_prompt() -> None: 853 """Show a 'not enough tokens' prompt with an option to purchase more. 854 855 Note that the purchase option may not always be available 856 depending on the build of the game. 857 """ 858 from bauiv1lib.confirm import ConfirmWindow 859 860 assert bui.app.classic is not None 861 862 # Currently always allowing token purchases. 863 if bool(True): 864 ConfirmWindow( 865 bui.Lstr(resource='tokens.notEnoughTokensText'), 866 show_get_tokens_window, 867 ok_text=bui.Lstr(resource='tokens.getTokensText'), 868 width=460, 869 height=130, 870 ) 871 else: 872 ConfirmWindow( 873 bui.Lstr(resource='tokens.notEnoughTokensText'), 874 cancel_button=False, 875 width=460, 876 height=130, 877 ) 878 879 880def show_get_tokens_window(origin_widget: bui.Widget | None = None) -> None: 881 """Show the window allowing token purchases.""" 882 883 # NOTE TO USERS: The code below is not the proper way to do things; 884 # whenever possible one should use a MainWindow's 885 # main_window_replace() or main_window_back() methods. We just need 886 # to do things a bit more manually in this case. 887 888 prev_main_window = bui.app.ui_v1.get_main_window() 889 890 # Special-case: If it seems we're already in the account window, do 891 # nothing. 892 if isinstance(prev_main_window, GetTokensWindow): 893 return 894 895 # Set our new main window. 896 bui.app.ui_v1.set_main_window( 897 GetTokensWindow(origin_widget=origin_widget), 898 from_window=False, 899 is_auxiliary=True, 900 suppress_warning=True, 901 ) 902 903 # Transition out any previous main window. 904 if prev_main_window is not None: 905 prev_main_window.main_window_close()
class
TextContents(enum.Enum):
Some type of text to show.
PRICE =
<TextContents.PRICE: 'price'>
class
GetTokensWindow(bauiv1._uitypes.MainWindow):
58class GetTokensWindow(bui.MainWindow): 59 """Window for purchasing/acquiring classic tickets.""" 60 61 class State(Enum): 62 """What are we doing?""" 63 64 LOADING = 'loading' 65 NOT_SIGNED_IN = 'not_signed_in' 66 HAVE_GOLD_PASS = 'have_gold_pass' 67 SHOWING_STORE = 'showing_store' 68 69 def __init__( 70 self, 71 transition: str | None = 'in_right', 72 origin_widget: bui.Widget | None = None, 73 # restore_previous_call: Callable[[bui.Widget], None] | None = None, 74 ): 75 bwidthstd = 170 76 bwidthwide = 300 77 ycolor = (0, 0, 0.3) 78 pcolor = (0, 0, 0.3) 79 pos1 = 65 80 pos2 = 34 81 titlescale = 0.9 82 pricescale = 0.65 83 bcapcol1 = (0.25, 0.13, 0.02) 84 self._buttondefs: list[_ButtonDef] = [ 85 _ButtonDef( 86 itemid='tokens1', 87 width=bwidthstd, 88 color=ycolor, 89 imgdefs=[ 90 _ImgDef( 91 'tokens1', 92 pos=(-3, 85), 93 size=(172, 172), 94 opacity=1.0, 95 draw_controller_mult=0.5, 96 ), 97 _ImgDef( 98 'windowBottomCap', 99 pos=(1.5, 4), 100 size=(bwidthstd * 0.960, 100), 101 color=bcapcol1, 102 opacity=1.0, 103 ), 104 ], 105 txtdefs=[ 106 _TxtDef( 107 bui.Lstr( 108 resource='tokens.numTokensText', 109 subs=[('${COUNT}', '50')], 110 ), 111 pos=(bwidthstd * 0.5, pos1), 112 color=(1.1, 1.05, 1.0), 113 scale=titlescale, 114 maxwidth=bwidthstd * 0.9, 115 ), 116 _TxtDef( 117 TextContents.PRICE, 118 pos=(bwidthstd * 0.5, pos2), 119 color=(1.1, 1.05, 1.0), 120 scale=pricescale, 121 maxwidth=bwidthstd * 0.9, 122 ), 123 ], 124 ), 125 _ButtonDef( 126 itemid='tokens2', 127 width=bwidthstd, 128 color=ycolor, 129 imgdefs=[ 130 _ImgDef( 131 'tokens2', 132 pos=(-3, 85), 133 size=(172, 172), 134 opacity=1.0, 135 draw_controller_mult=0.5, 136 ), 137 _ImgDef( 138 'windowBottomCap', 139 pos=(1.5, 4), 140 size=(bwidthstd * 0.960, 100), 141 color=bcapcol1, 142 opacity=1.0, 143 ), 144 ], 145 txtdefs=[ 146 _TxtDef( 147 bui.Lstr( 148 resource='tokens.numTokensText', 149 subs=[('${COUNT}', '500')], 150 ), 151 pos=(bwidthstd * 0.5, pos1), 152 color=(1.1, 1.05, 1.0), 153 scale=titlescale, 154 maxwidth=bwidthstd * 0.9, 155 ), 156 _TxtDef( 157 TextContents.PRICE, 158 pos=(bwidthstd * 0.5, pos2), 159 color=(1.1, 1.05, 1.0), 160 scale=pricescale, 161 maxwidth=bwidthstd * 0.9, 162 ), 163 ], 164 ), 165 _ButtonDef( 166 itemid='tokens3', 167 width=bwidthstd, 168 color=ycolor, 169 imgdefs=[ 170 _ImgDef( 171 'tokens3', 172 pos=(-3, 85), 173 size=(172, 172), 174 opacity=1.0, 175 draw_controller_mult=0.5, 176 ), 177 _ImgDef( 178 'windowBottomCap', 179 pos=(1.5, 4), 180 size=(bwidthstd * 0.960, 100), 181 color=bcapcol1, 182 opacity=1.0, 183 ), 184 ], 185 txtdefs=[ 186 _TxtDef( 187 bui.Lstr( 188 resource='tokens.numTokensText', 189 subs=[('${COUNT}', '1200')], 190 ), 191 pos=(bwidthstd * 0.5, pos1), 192 color=(1.1, 1.05, 1.0), 193 scale=titlescale, 194 maxwidth=bwidthstd * 0.9, 195 ), 196 _TxtDef( 197 TextContents.PRICE, 198 pos=(bwidthstd * 0.5, pos2), 199 color=(1.1, 1.05, 1.0), 200 scale=pricescale, 201 maxwidth=bwidthstd * 0.9, 202 ), 203 ], 204 ), 205 _ButtonDef( 206 itemid='tokens4', 207 width=bwidthstd, 208 color=ycolor, 209 imgdefs=[ 210 _ImgDef( 211 'tokens4', 212 pos=(-3, 85), 213 size=(172, 172), 214 opacity=1.0, 215 draw_controller_mult=0.5, 216 ), 217 _ImgDef( 218 'windowBottomCap', 219 pos=(1.5, 4), 220 size=(bwidthstd * 0.960, 100), 221 color=bcapcol1, 222 opacity=1.0, 223 ), 224 ], 225 txtdefs=[ 226 _TxtDef( 227 bui.Lstr( 228 resource='tokens.numTokensText', 229 subs=[('${COUNT}', '2600')], 230 ), 231 pos=(bwidthstd * 0.5, pos1), 232 color=(1.1, 1.05, 1.0), 233 scale=titlescale, 234 maxwidth=bwidthstd * 0.9, 235 ), 236 _TxtDef( 237 TextContents.PRICE, 238 pos=(bwidthstd * 0.5, pos2), 239 color=(1.1, 1.05, 1.0), 240 scale=pricescale, 241 maxwidth=bwidthstd * 0.9, 242 ), 243 ], 244 ), 245 _ButtonDef( 246 itemid='gold_pass', 247 width=bwidthwide, 248 color=pcolor, 249 imgdefs=[ 250 _ImgDef( 251 'goldPass', 252 pos=(-7, 102), 253 size=(312, 156), 254 draw_controller_mult=0.3, 255 ), 256 _ImgDef( 257 'windowBottomCap', 258 pos=(8, 4), 259 size=(bwidthwide * 0.923, 116), 260 color=(0.25, 0.12, 0.15), 261 opacity=1.0, 262 ), 263 ], 264 txtdefs=[ 265 _TxtDef( 266 bui.Lstr(resource='goldPass.goldPassText'), 267 pos=(bwidthwide * 0.5, pos1 + 27), 268 color=(1.1, 1.05, 1.0), 269 scale=titlescale, 270 maxwidth=bwidthwide * 0.8, 271 ), 272 _TxtDef( 273 bui.Lstr(resource='goldPass.desc1InfTokensText'), 274 pos=(bwidthwide * 0.5, pos1 + 6), 275 color=(1.1, 1.05, 1.0), 276 scale=0.4, 277 maxwidth=bwidthwide * 0.8, 278 ), 279 _TxtDef( 280 bui.Lstr(resource='goldPass.desc2NoAdsText'), 281 pos=(bwidthwide * 0.5, pos1 + 6 - 13 * 1), 282 color=(1.1, 1.05, 1.0), 283 scale=0.4, 284 maxwidth=bwidthwide * 0.8, 285 ), 286 _TxtDef( 287 bui.Lstr(resource='goldPass.desc3ForeverText'), 288 pos=(bwidthwide * 0.5, pos1 + 6 - 13 * 2), 289 color=(1.1, 1.05, 1.0), 290 scale=0.4, 291 maxwidth=bwidthwide * 0.8, 292 ), 293 _TxtDef( 294 TextContents.PRICE, 295 pos=(bwidthwide * 0.5, pos2 - 9), 296 color=(1.1, 1.05, 1.0), 297 scale=pricescale, 298 maxwidth=bwidthwide * 0.8, 299 ), 300 ], 301 prepad=-8, 302 ), 303 ] 304 305 self._transitioning_out = False 306 # self._restore_previous_call = restore_previous_call 307 self._textcolor = (0.92, 0.92, 2.0) 308 309 self._query_in_flight = False 310 self._last_query_time = -1.0 311 self._last_query_response: bacommon.cloud.StoreQueryResponse | None = ( 312 None 313 ) 314 315 # If they provided an origin-widget, scale up from that. 316 # scale_origin: tuple[float, float] | None 317 # if origin_widget is not None: 318 # self._transition_out = 'out_scale' 319 # scale_origin = origin_widget.get_screen_space_center() 320 # transition = 'in_scale' 321 # else: 322 # self._transition_out = 'out_right' 323 # scale_origin = None 324 325 uiscale = bui.app.ui_v1.uiscale 326 self._width = 1000.0 if uiscale is bui.UIScale.SMALL else 800.0 327 self._x_inset = 25.0 if uiscale is bui.UIScale.SMALL else 0.0 328 self._height = 550 if uiscale is bui.UIScale.SMALL else 480.0 329 self._y_offset = -60 if uiscale is bui.UIScale.SMALL else 0 330 331 self._r = 'getTokensWindow' 332 333 super().__init__( 334 root_widget=bui.containerwidget( 335 size=(self._width, self._height), 336 # transition=transition, 337 # scale_origin_stack_offset=scale_origin, 338 color=(0.3, 0.23, 0.36), 339 scale=( 340 1.5 341 if uiscale is bui.UIScale.SMALL 342 else 1.2 if uiscale is bui.UIScale.MEDIUM else 1.0 343 ), 344 stack_offset=( 345 (0, -3) if uiscale is bui.UIScale.SMALL else (0, 0) 346 ), 347 # toolbar_visibility='menu_minimal', 348 toolbar_visibility=( 349 'get_tokens' 350 if uiscale is bui.UIScale.SMALL 351 else 'menu_full' 352 ), 353 ), 354 transition=transition, 355 origin_widget=origin_widget, 356 ) 357 358 if uiscale is bui.UIScale.SMALL: 359 bui.containerwidget( 360 edit=self._root_widget, on_cancel_call=self.main_window_back 361 ) 362 self._back_button = bui.get_special_widget('back_button') 363 else: 364 self._back_button = bui.buttonwidget( 365 parent=self._root_widget, 366 position=( 367 55 + self._x_inset, 368 self._height - 80 + self._y_offset, 369 ), 370 size=( 371 # (140, 60) 372 # if self._restore_previous_call is None 373 # else 374 (60, 60) 375 ), 376 scale=1.0, 377 autoselect=True, 378 label=( 379 # bui.Lstr(resource='doneText') 380 # if self._restore_previous_call is None 381 # else 382 bui.charstr(bui.SpecialChar.BACK) 383 ), 384 button_type=( 385 # 'regular' 386 # if self._restore_previous_call is None 387 # else 388 'backSmall' 389 ), 390 on_activate_call=self.main_window_back, 391 ) 392 # if uiscale is bui.UIScale.SMALL: 393 # bui.widget( 394 # edit=self._back_button, 395 # up_widget=bui.get_special_widget('tokens_meter'), 396 # ) 397 bui.containerwidget( 398 edit=self._root_widget, cancel_button=self._back_button 399 ) 400 401 self._title_text = bui.textwidget( 402 parent=self._root_widget, 403 position=(self._width * 0.5, self._height - 42 + self._y_offset), 404 size=(0, 0), 405 color=self._textcolor, 406 flatness=0.0, 407 shadow=1.0, 408 scale=1.2, 409 h_align='center', 410 v_align='center', 411 text=bui.Lstr(resource='tokens.getTokensText'), 412 maxwidth=260, 413 ) 414 415 self._status_text = bui.textwidget( 416 parent=self._root_widget, 417 size=(0, 0), 418 position=(self._width * 0.5, self._height * 0.5), 419 h_align='center', 420 v_align='center', 421 color=(0.6, 0.6, 0.6), 422 scale=0.75, 423 text=bui.Lstr(resource='store.loadingText'), 424 ) 425 426 self._core_widgets = [ 427 self._back_button, 428 self._title_text, 429 self._status_text, 430 ] 431 432 # self._token_count_widget: bui.Widget | None = None 433 # self._smooth_update_timer: bui.AppTimer | None = None 434 # self._smooth_token_count: float | None = None 435 # self._token_count: int = 0 436 # self._smooth_increase_speed = 1.0 437 # self._ticking_sound: bui.Sound | None = None 438 439 # Get all textures used by our buttons preloading so hopefully 440 # they'll be in place by the time we show them. 441 for bdef in self._buttondefs: 442 for bimg in bdef.imgdefs: 443 bui.gettexture(bimg.tex) 444 445 self._state = self.State.LOADING 446 447 self._update_timer = bui.AppTimer( 448 0.789, bui.WeakCall(self._update), repeat=True 449 ) 450 self._update() 451 452 # def __del__(self) -> None: 453 # if self._ticking_sound is not None: 454 # self._ticking_sound.stop() 455 # self._ticking_sound = None 456 457 @override 458 def get_main_window_state(self) -> bui.MainWindowState: 459 # Support recreating our window for back/refresh purposes. 460 cls = type(self) 461 return bui.BasicMainWindowState( 462 create_call=lambda transition, origin_widget: cls( 463 transition=transition, origin_widget=origin_widget 464 ) 465 ) 466 467 def _update(self) -> None: 468 # No-op if our underlying widget is dead or on its way out. 469 if not self._root_widget or self._root_widget.transitioning_out: 470 return 471 472 plus = bui.app.plus 473 474 if plus is None or plus.accounts.primary is None: 475 self._update_state(self.State.NOT_SIGNED_IN) 476 return 477 478 # Poll for relevant changes to the store or our account. 479 now = time.monotonic() 480 if not self._query_in_flight and now - self._last_query_time > 2.0: 481 self._last_query_time = now 482 self._query_in_flight = True 483 with plus.accounts.primary: 484 plus.cloud.send_message_cb( 485 bacommon.cloud.StoreQueryMessage(), 486 on_response=bui.WeakCall(self._on_store_query_response), 487 ) 488 489 # Can't do much until we get a store state. 490 if self._last_query_response is None: 491 return 492 493 # If we've got a gold-pass, just show that. No need to offer any 494 # other purchases. 495 if self._last_query_response.gold_pass: 496 self._update_state(self.State.HAVE_GOLD_PASS) 497 return 498 499 # Ok we seem to be signed in and have store stuff we can show. 500 # Do that. 501 self._update_state(self.State.SHOWING_STORE) 502 503 def _update_state(self, state: State) -> None: 504 505 # We don't do much when state is unchanged. 506 if state is self._state: 507 # Update a few things in store mode though, such as token 508 # count. 509 if state is self.State.SHOWING_STORE: 510 self._update_store_state() 511 return 512 513 # Ok, state is changing. Start by resetting to a blank slate. 514 # self._token_count_widget = None 515 for widget in self._root_widget.get_children(): 516 if widget not in self._core_widgets: 517 widget.delete() 518 519 # Build up new state. 520 if state is self.State.NOT_SIGNED_IN: 521 bui.textwidget( 522 edit=self._status_text, 523 color=(1, 0, 0), 524 text=bui.Lstr(resource='notSignedInErrorText'), 525 ) 526 elif state is self.State.LOADING: 527 raise RuntimeError('Should never return to loading state.') 528 elif state is self.State.HAVE_GOLD_PASS: 529 bui.textwidget( 530 edit=self._status_text, 531 color=(0, 1, 0), 532 text=bui.Lstr(resource='tokens.youHaveGoldPassText'), 533 ) 534 elif state is self.State.SHOWING_STORE: 535 assert self._last_query_response is not None 536 bui.textwidget(edit=self._status_text, text='') 537 self._build_store_for_response(self._last_query_response) 538 else: 539 # Make sure we handle all cases. 540 assert_never(state) 541 542 self._state = state 543 544 def _on_load_error(self) -> None: 545 bui.textwidget( 546 edit=self._status_text, 547 text=bui.Lstr(resource='internal.unavailableNoConnectionText'), 548 color=(1, 0, 0), 549 ) 550 551 def _on_store_query_response( 552 self, response: bacommon.cloud.StoreQueryResponse | Exception 553 ) -> None: 554 self._query_in_flight = False 555 if isinstance(response, bacommon.cloud.StoreQueryResponse): 556 self._last_query_response = response 557 # Hurry along any effects of this response. 558 self._update() 559 560 def _build_store_for_response( 561 self, response: bacommon.cloud.StoreQueryResponse 562 ) -> None: 563 # pylint: disable=too-many-locals 564 plus = bui.app.plus 565 566 uiscale = bui.app.ui_v1.uiscale 567 568 bui.textwidget(edit=self._status_text, text='') 569 570 xinset = 40 571 572 scrollwidth = self._width - 2 * (self._x_inset + xinset) 573 scrollheight = 280 574 buttonpadding = -5 575 576 yoffs = 5 577 578 # We currently don't handle the zero-button case. 579 assert self._buttondefs 580 581 sidepad = 10.0 582 total_button_width = ( 583 sum(b.width + b.prepad for b in self._buttondefs) 584 + buttonpadding * (len(self._buttondefs) - 1) 585 + 2 * sidepad 586 ) 587 588 h_scroll = bui.hscrollwidget( 589 parent=self._root_widget, 590 size=(scrollwidth, scrollheight), 591 position=( 592 self._x_inset + xinset, 593 self._height - 415 + self._y_offset, 594 ), 595 claims_left_right=True, 596 highlight=False, 597 border_opacity=0.3 if uiscale is bui.UIScale.SMALL else 1.0, 598 ) 599 subcontainer = bui.containerwidget( 600 parent=h_scroll, 601 background=False, 602 size=(max(total_button_width, scrollwidth), scrollheight), 603 ) 604 tinfobtn = bui.buttonwidget( 605 parent=self._root_widget, 606 autoselect=True, 607 label=bui.Lstr(resource='learnMoreText'), 608 position=( 609 self._width * 0.5 - 75, 610 self._height - 125 + self._y_offset, 611 ), 612 size=(180, 43), 613 scale=0.8, 614 color=(0.4, 0.25, 0.5), 615 textcolor=self._textcolor, 616 on_activate_call=partial( 617 self._on_learn_more_press, response.token_info_url 618 ), 619 ) 620 if uiscale is bui.UIScale.SMALL: 621 bui.widget( 622 edit=tinfobtn, 623 left_widget=bui.get_special_widget('back_button'), 624 up_widget=bui.get_special_widget('back_button'), 625 ) 626 627 bui.widget( 628 edit=tinfobtn, 629 right_widget=bui.get_special_widget('tokens_meter'), 630 ) 631 632 x = sidepad 633 bwidgets: list[bui.Widget] = [] 634 for i, buttondef in enumerate(self._buttondefs): 635 636 price = None if plus is None else plus.get_price(buttondef.itemid) 637 638 x += buttondef.prepad 639 tdelay = 0.3 - i / len(self._buttondefs) * 0.25 640 btn = bui.buttonwidget( 641 autoselect=True, 642 label='', 643 color=buttondef.color, 644 transition_delay=tdelay, 645 up_widget=tinfobtn, 646 parent=subcontainer, 647 size=(buttondef.width, 275), 648 position=(x, -10 + yoffs), 649 button_type='square', 650 on_activate_call=partial( 651 self._purchase_press, buttondef.itemid 652 ), 653 ) 654 bwidgets.append(btn) 655 656 if i == 0: 657 bui.widget(edit=btn, left_widget=self._back_button) 658 659 for imgdef in buttondef.imgdefs: 660 _img = bui.imagewidget( 661 parent=subcontainer, 662 size=imgdef.size, 663 position=(x + imgdef.pos[0], imgdef.pos[1] + yoffs), 664 draw_controller=btn, 665 draw_controller_mult=imgdef.draw_controller_mult, 666 color=imgdef.color, 667 texture=bui.gettexture(imgdef.tex), 668 transition_delay=tdelay, 669 opacity=imgdef.opacity, 670 ) 671 for txtdef in buttondef.txtdefs: 672 txt: bui.Lstr | str 673 if isinstance(txtdef.text, TextContents): 674 if txtdef.text is TextContents.PRICE: 675 tcolor = ( 676 (1, 1, 1, 0.5) if price is None else txtdef.color 677 ) 678 txt = ( 679 bui.Lstr(resource='unavailableText') 680 if price is None 681 else price 682 ) 683 else: 684 # Make sure we cover all cases. 685 assert_never(txtdef.text) 686 else: 687 tcolor = txtdef.color 688 txt = txtdef.text 689 _txt = bui.textwidget( 690 parent=subcontainer, 691 text=txt, 692 position=(x + txtdef.pos[0], txtdef.pos[1] + yoffs), 693 size=(0, 0), 694 scale=txtdef.scale, 695 h_align='center', 696 v_align='center', 697 draw_controller=btn, 698 color=tcolor, 699 transition_delay=tdelay, 700 flatness=0.0, 701 shadow=1.0, 702 rotate=txtdef.rotate, 703 maxwidth=txtdef.maxwidth, 704 ) 705 x += buttondef.width + buttonpadding 706 bui.containerwidget(edit=subcontainer, visible_child=bwidgets[0]) 707 708 _tinfotxt = bui.textwidget( 709 parent=self._root_widget, 710 position=(self._width * 0.5, self._height - 70 + self._y_offset), 711 color=self._textcolor, 712 shadow=1.0, 713 scale=0.7, 714 size=(0, 0), 715 h_align='center', 716 v_align='center', 717 text=bui.Lstr(resource='tokens.shinyNewCurrencyText'), 718 ) 719 # self._token_count_widget = bui.textwidget( 720 # parent=self._root_widget, 721 # position=( 722 # self._width - self._x_inset - 120.0, 723 # self._height - 48 + self._y_offset, 724 # ), 725 # color=(2.0, 0.7, 0.0), 726 # shadow=1.0, 727 # flatness=0.0, 728 # size=(0, 0), 729 # h_align='left', 730 # v_align='center', 731 # text='', 732 # ) 733 # self._token_count = response.tokens 734 # self._smooth_token_count = float(self._token_count) 735 # self._smooth_update() # will set the text widget. 736 737 # _tlabeltxt = bui.textwidget( 738 # parent=self._root_widget, 739 # position=( 740 # self._width - self._x_inset - 123.0, 741 # self._height - 48 + self._y_offset, 742 # ), 743 # size=(0, 0), 744 # h_align='right', 745 # v_align='center', 746 # text=bui.charstr(bui.SpecialChar.TOKEN), 747 # ) 748 749 def _purchase_press(self, itemid: str) -> None: 750 plus = bui.app.plus 751 752 price = None if plus is None else plus.get_price(itemid) 753 754 if price is None: 755 if plus is not None and plus.supports_purchases(): 756 # Looks like internet is down or something temporary. 757 errmsg = bui.Lstr(resource='purchaseNotAvailableText') 758 else: 759 # Looks like purchases will never work here. 760 errmsg = bui.Lstr(resource='purchaseNeverAvailableText') 761 762 bui.screenmessage(errmsg, color=(1, 0.5, 0)) 763 bui.getsound('error').play() 764 return 765 766 assert plus is not None 767 plus.purchase(itemid) 768 769 def _update_store_state(self) -> None: 770 """Called to make minor updates to an already shown store.""" 771 # assert self._token_count_widget is not None 772 assert self._last_query_response is not None 773 774 # self._token_count = self._last_query_response.tokens 775 776 # Kick off new smooth update if need be. 777 # assert self._smooth_token_count is not None 778 # if ( 779 # self._token_count != int(self._smooth_token_count) 780 # and self._smooth_update_timer is None 781 # ): 782 # self._smooth_update_timer = bui.AppTimer( 783 # 0.05, bui.WeakCall(self._smooth_update), repeat=True 784 # ) 785 # diff = abs(float(self._token_count) - self._smooth_token_count) 786 # self._smooth_increase_speed = ( 787 # diff / 100.0 788 # if diff >= 5000 789 # else ( 790 # diff / 50.0 791 # if diff >= 1500 792 # else diff / 30.0 if diff >= 500 else diff / 15.0 793 # ) 794 # ) 795 796 # def _smooth_update(self) -> None: 797 798 # # Stop if the count widget disappears. 799 # if not self._token_count_widget: 800 # self._smooth_update_timer = None 801 # return 802 803 # finished = False 804 805 # # If we're going down, do it immediately. 806 # assert self._smooth_token_count is not None 807 # if int(self._smooth_token_count) >= self._token_count: 808 # self._smooth_token_count = float(self._token_count) 809 # finished = True 810 # else: 811 # # We're going up; start a sound if need be. 812 # self._smooth_token_count = min( 813 # self._smooth_token_count + 1.0 * self._smooth_increase_speed, 814 # self._token_count, 815 # ) 816 # if int(self._smooth_token_count) >= self._token_count: 817 # finished = True 818 # self._smooth_token_count = float(self._token_count) 819 # elif self._ticking_sound is None: 820 # self._ticking_sound = bui.getsound('scoreIncrease') 821 # self._ticking_sound.play() 822 823 # bui.textwidget( 824 # edit=self._token_count_widget, 825 # text=str(int(self._smooth_token_count)), 826 # ) 827 828 # # If we've reached the target, kill the timer/sound/etc. 829 # if finished: 830 # self._smooth_update_timer = None 831 # if self._ticking_sound is not None: 832 # self._ticking_sound.stop() 833 # self._ticking_sound = None 834 # bui.getsound('cashRegister2').play() 835 836 # def _back(self) -> None: 837 838 # self.main_ 839 # No-op if our underlying widget is dead or on its way out. 840 # if not self._root_widget or self._root_widget.transitioning_out: 841 # return 842 843 # bui.containerwidget( 844 # edit=self._root_widget, transition=self._transition_out 845 # ) 846 # if self._restore_previous_call is not None: 847 # self._restore_previous_call(self._root_widget) 848 849 def _on_learn_more_press(self, url: str) -> None: 850 bui.open_url(url)
Window for purchasing/acquiring classic tickets.
GetTokensWindow( transition: str | None = 'in_right', origin_widget: _bauiv1.Widget | None = None)
69 def __init__( 70 self, 71 transition: str | None = 'in_right', 72 origin_widget: bui.Widget | None = None, 73 # restore_previous_call: Callable[[bui.Widget], None] | None = None, 74 ): 75 bwidthstd = 170 76 bwidthwide = 300 77 ycolor = (0, 0, 0.3) 78 pcolor = (0, 0, 0.3) 79 pos1 = 65 80 pos2 = 34 81 titlescale = 0.9 82 pricescale = 0.65 83 bcapcol1 = (0.25, 0.13, 0.02) 84 self._buttondefs: list[_ButtonDef] = [ 85 _ButtonDef( 86 itemid='tokens1', 87 width=bwidthstd, 88 color=ycolor, 89 imgdefs=[ 90 _ImgDef( 91 'tokens1', 92 pos=(-3, 85), 93 size=(172, 172), 94 opacity=1.0, 95 draw_controller_mult=0.5, 96 ), 97 _ImgDef( 98 'windowBottomCap', 99 pos=(1.5, 4), 100 size=(bwidthstd * 0.960, 100), 101 color=bcapcol1, 102 opacity=1.0, 103 ), 104 ], 105 txtdefs=[ 106 _TxtDef( 107 bui.Lstr( 108 resource='tokens.numTokensText', 109 subs=[('${COUNT}', '50')], 110 ), 111 pos=(bwidthstd * 0.5, pos1), 112 color=(1.1, 1.05, 1.0), 113 scale=titlescale, 114 maxwidth=bwidthstd * 0.9, 115 ), 116 _TxtDef( 117 TextContents.PRICE, 118 pos=(bwidthstd * 0.5, pos2), 119 color=(1.1, 1.05, 1.0), 120 scale=pricescale, 121 maxwidth=bwidthstd * 0.9, 122 ), 123 ], 124 ), 125 _ButtonDef( 126 itemid='tokens2', 127 width=bwidthstd, 128 color=ycolor, 129 imgdefs=[ 130 _ImgDef( 131 'tokens2', 132 pos=(-3, 85), 133 size=(172, 172), 134 opacity=1.0, 135 draw_controller_mult=0.5, 136 ), 137 _ImgDef( 138 'windowBottomCap', 139 pos=(1.5, 4), 140 size=(bwidthstd * 0.960, 100), 141 color=bcapcol1, 142 opacity=1.0, 143 ), 144 ], 145 txtdefs=[ 146 _TxtDef( 147 bui.Lstr( 148 resource='tokens.numTokensText', 149 subs=[('${COUNT}', '500')], 150 ), 151 pos=(bwidthstd * 0.5, pos1), 152 color=(1.1, 1.05, 1.0), 153 scale=titlescale, 154 maxwidth=bwidthstd * 0.9, 155 ), 156 _TxtDef( 157 TextContents.PRICE, 158 pos=(bwidthstd * 0.5, pos2), 159 color=(1.1, 1.05, 1.0), 160 scale=pricescale, 161 maxwidth=bwidthstd * 0.9, 162 ), 163 ], 164 ), 165 _ButtonDef( 166 itemid='tokens3', 167 width=bwidthstd, 168 color=ycolor, 169 imgdefs=[ 170 _ImgDef( 171 'tokens3', 172 pos=(-3, 85), 173 size=(172, 172), 174 opacity=1.0, 175 draw_controller_mult=0.5, 176 ), 177 _ImgDef( 178 'windowBottomCap', 179 pos=(1.5, 4), 180 size=(bwidthstd * 0.960, 100), 181 color=bcapcol1, 182 opacity=1.0, 183 ), 184 ], 185 txtdefs=[ 186 _TxtDef( 187 bui.Lstr( 188 resource='tokens.numTokensText', 189 subs=[('${COUNT}', '1200')], 190 ), 191 pos=(bwidthstd * 0.5, pos1), 192 color=(1.1, 1.05, 1.0), 193 scale=titlescale, 194 maxwidth=bwidthstd * 0.9, 195 ), 196 _TxtDef( 197 TextContents.PRICE, 198 pos=(bwidthstd * 0.5, pos2), 199 color=(1.1, 1.05, 1.0), 200 scale=pricescale, 201 maxwidth=bwidthstd * 0.9, 202 ), 203 ], 204 ), 205 _ButtonDef( 206 itemid='tokens4', 207 width=bwidthstd, 208 color=ycolor, 209 imgdefs=[ 210 _ImgDef( 211 'tokens4', 212 pos=(-3, 85), 213 size=(172, 172), 214 opacity=1.0, 215 draw_controller_mult=0.5, 216 ), 217 _ImgDef( 218 'windowBottomCap', 219 pos=(1.5, 4), 220 size=(bwidthstd * 0.960, 100), 221 color=bcapcol1, 222 opacity=1.0, 223 ), 224 ], 225 txtdefs=[ 226 _TxtDef( 227 bui.Lstr( 228 resource='tokens.numTokensText', 229 subs=[('${COUNT}', '2600')], 230 ), 231 pos=(bwidthstd * 0.5, pos1), 232 color=(1.1, 1.05, 1.0), 233 scale=titlescale, 234 maxwidth=bwidthstd * 0.9, 235 ), 236 _TxtDef( 237 TextContents.PRICE, 238 pos=(bwidthstd * 0.5, pos2), 239 color=(1.1, 1.05, 1.0), 240 scale=pricescale, 241 maxwidth=bwidthstd * 0.9, 242 ), 243 ], 244 ), 245 _ButtonDef( 246 itemid='gold_pass', 247 width=bwidthwide, 248 color=pcolor, 249 imgdefs=[ 250 _ImgDef( 251 'goldPass', 252 pos=(-7, 102), 253 size=(312, 156), 254 draw_controller_mult=0.3, 255 ), 256 _ImgDef( 257 'windowBottomCap', 258 pos=(8, 4), 259 size=(bwidthwide * 0.923, 116), 260 color=(0.25, 0.12, 0.15), 261 opacity=1.0, 262 ), 263 ], 264 txtdefs=[ 265 _TxtDef( 266 bui.Lstr(resource='goldPass.goldPassText'), 267 pos=(bwidthwide * 0.5, pos1 + 27), 268 color=(1.1, 1.05, 1.0), 269 scale=titlescale, 270 maxwidth=bwidthwide * 0.8, 271 ), 272 _TxtDef( 273 bui.Lstr(resource='goldPass.desc1InfTokensText'), 274 pos=(bwidthwide * 0.5, pos1 + 6), 275 color=(1.1, 1.05, 1.0), 276 scale=0.4, 277 maxwidth=bwidthwide * 0.8, 278 ), 279 _TxtDef( 280 bui.Lstr(resource='goldPass.desc2NoAdsText'), 281 pos=(bwidthwide * 0.5, pos1 + 6 - 13 * 1), 282 color=(1.1, 1.05, 1.0), 283 scale=0.4, 284 maxwidth=bwidthwide * 0.8, 285 ), 286 _TxtDef( 287 bui.Lstr(resource='goldPass.desc3ForeverText'), 288 pos=(bwidthwide * 0.5, pos1 + 6 - 13 * 2), 289 color=(1.1, 1.05, 1.0), 290 scale=0.4, 291 maxwidth=bwidthwide * 0.8, 292 ), 293 _TxtDef( 294 TextContents.PRICE, 295 pos=(bwidthwide * 0.5, pos2 - 9), 296 color=(1.1, 1.05, 1.0), 297 scale=pricescale, 298 maxwidth=bwidthwide * 0.8, 299 ), 300 ], 301 prepad=-8, 302 ), 303 ] 304 305 self._transitioning_out = False 306 # self._restore_previous_call = restore_previous_call 307 self._textcolor = (0.92, 0.92, 2.0) 308 309 self._query_in_flight = False 310 self._last_query_time = -1.0 311 self._last_query_response: bacommon.cloud.StoreQueryResponse | None = ( 312 None 313 ) 314 315 # If they provided an origin-widget, scale up from that. 316 # scale_origin: tuple[float, float] | None 317 # if origin_widget is not None: 318 # self._transition_out = 'out_scale' 319 # scale_origin = origin_widget.get_screen_space_center() 320 # transition = 'in_scale' 321 # else: 322 # self._transition_out = 'out_right' 323 # scale_origin = None 324 325 uiscale = bui.app.ui_v1.uiscale 326 self._width = 1000.0 if uiscale is bui.UIScale.SMALL else 800.0 327 self._x_inset = 25.0 if uiscale is bui.UIScale.SMALL else 0.0 328 self._height = 550 if uiscale is bui.UIScale.SMALL else 480.0 329 self._y_offset = -60 if uiscale is bui.UIScale.SMALL else 0 330 331 self._r = 'getTokensWindow' 332 333 super().__init__( 334 root_widget=bui.containerwidget( 335 size=(self._width, self._height), 336 # transition=transition, 337 # scale_origin_stack_offset=scale_origin, 338 color=(0.3, 0.23, 0.36), 339 scale=( 340 1.5 341 if uiscale is bui.UIScale.SMALL 342 else 1.2 if uiscale is bui.UIScale.MEDIUM else 1.0 343 ), 344 stack_offset=( 345 (0, -3) if uiscale is bui.UIScale.SMALL else (0, 0) 346 ), 347 # toolbar_visibility='menu_minimal', 348 toolbar_visibility=( 349 'get_tokens' 350 if uiscale is bui.UIScale.SMALL 351 else 'menu_full' 352 ), 353 ), 354 transition=transition, 355 origin_widget=origin_widget, 356 ) 357 358 if uiscale is bui.UIScale.SMALL: 359 bui.containerwidget( 360 edit=self._root_widget, on_cancel_call=self.main_window_back 361 ) 362 self._back_button = bui.get_special_widget('back_button') 363 else: 364 self._back_button = bui.buttonwidget( 365 parent=self._root_widget, 366 position=( 367 55 + self._x_inset, 368 self._height - 80 + self._y_offset, 369 ), 370 size=( 371 # (140, 60) 372 # if self._restore_previous_call is None 373 # else 374 (60, 60) 375 ), 376 scale=1.0, 377 autoselect=True, 378 label=( 379 # bui.Lstr(resource='doneText') 380 # if self._restore_previous_call is None 381 # else 382 bui.charstr(bui.SpecialChar.BACK) 383 ), 384 button_type=( 385 # 'regular' 386 # if self._restore_previous_call is None 387 # else 388 'backSmall' 389 ), 390 on_activate_call=self.main_window_back, 391 ) 392 # if uiscale is bui.UIScale.SMALL: 393 # bui.widget( 394 # edit=self._back_button, 395 # up_widget=bui.get_special_widget('tokens_meter'), 396 # ) 397 bui.containerwidget( 398 edit=self._root_widget, cancel_button=self._back_button 399 ) 400 401 self._title_text = bui.textwidget( 402 parent=self._root_widget, 403 position=(self._width * 0.5, self._height - 42 + self._y_offset), 404 size=(0, 0), 405 color=self._textcolor, 406 flatness=0.0, 407 shadow=1.0, 408 scale=1.2, 409 h_align='center', 410 v_align='center', 411 text=bui.Lstr(resource='tokens.getTokensText'), 412 maxwidth=260, 413 ) 414 415 self._status_text = bui.textwidget( 416 parent=self._root_widget, 417 size=(0, 0), 418 position=(self._width * 0.5, self._height * 0.5), 419 h_align='center', 420 v_align='center', 421 color=(0.6, 0.6, 0.6), 422 scale=0.75, 423 text=bui.Lstr(resource='store.loadingText'), 424 ) 425 426 self._core_widgets = [ 427 self._back_button, 428 self._title_text, 429 self._status_text, 430 ] 431 432 # self._token_count_widget: bui.Widget | None = None 433 # self._smooth_update_timer: bui.AppTimer | None = None 434 # self._smooth_token_count: float | None = None 435 # self._token_count: int = 0 436 # self._smooth_increase_speed = 1.0 437 # self._ticking_sound: bui.Sound | None = None 438 439 # Get all textures used by our buttons preloading so hopefully 440 # they'll be in place by the time we show them. 441 for bdef in self._buttondefs: 442 for bimg in bdef.imgdefs: 443 bui.gettexture(bimg.tex) 444 445 self._state = self.State.LOADING 446 447 self._update_timer = bui.AppTimer( 448 0.789, bui.WeakCall(self._update), repeat=True 449 ) 450 self._update()
Create a MainWindow given a root widget and transition info.
Automatically handles in and out transitions on the provided widget, so there is no need to set transitions when creating it.
457 @override 458 def get_main_window_state(self) -> bui.MainWindowState: 459 # Support recreating our window for back/refresh purposes. 460 cls = type(self) 461 return bui.BasicMainWindowState( 462 create_call=lambda transition, origin_widget: cls( 463 transition=transition, origin_widget=origin_widget 464 ) 465 )
Return a WindowState to recreate this window, if supported.
class
GetTokensWindow.State(enum.Enum):
61 class State(Enum): 62 """What are we doing?""" 63 64 LOADING = 'loading' 65 NOT_SIGNED_IN = 'not_signed_in' 66 HAVE_GOLD_PASS = 'have_gold_pass' 67 SHOWING_STORE = 'showing_store'
What are we doing?
def
show_get_tokens_prompt() -> None:
853def show_get_tokens_prompt() -> None: 854 """Show a 'not enough tokens' prompt with an option to purchase more. 855 856 Note that the purchase option may not always be available 857 depending on the build of the game. 858 """ 859 from bauiv1lib.confirm import ConfirmWindow 860 861 assert bui.app.classic is not None 862 863 # Currently always allowing token purchases. 864 if bool(True): 865 ConfirmWindow( 866 bui.Lstr(resource='tokens.notEnoughTokensText'), 867 show_get_tokens_window, 868 ok_text=bui.Lstr(resource='tokens.getTokensText'), 869 width=460, 870 height=130, 871 ) 872 else: 873 ConfirmWindow( 874 bui.Lstr(resource='tokens.notEnoughTokensText'), 875 cancel_button=False, 876 width=460, 877 height=130, 878 )
Show a 'not enough tokens' prompt with an option to purchase more.
Note that the purchase option may not always be available depending on the build of the game.
def
show_get_tokens_window(origin_widget: _bauiv1.Widget | None = None) -> None:
881def show_get_tokens_window(origin_widget: bui.Widget | None = None) -> None: 882 """Show the window allowing token purchases.""" 883 884 # NOTE TO USERS: The code below is not the proper way to do things; 885 # whenever possible one should use a MainWindow's 886 # main_window_replace() or main_window_back() methods. We just need 887 # to do things a bit more manually in this case. 888 889 prev_main_window = bui.app.ui_v1.get_main_window() 890 891 # Special-case: If it seems we're already in the account window, do 892 # nothing. 893 if isinstance(prev_main_window, GetTokensWindow): 894 return 895 896 # Set our new main window. 897 bui.app.ui_v1.set_main_window( 898 GetTokensWindow(origin_widget=origin_widget), 899 from_window=False, 900 is_auxiliary=True, 901 suppress_warning=True, 902 ) 903 904 # Transition out any previous main window. 905 if prev_main_window is not None: 906 prev_main_window.main_window_close()
Show the window allowing token purchases.