bauiv1lib.purchase

UI related to purchasing items.

  1# Released under the MIT License. See LICENSE for details.
  2#
  3"""UI related to purchasing items."""
  4
  5from __future__ import annotations
  6
  7from typing import TYPE_CHECKING
  8
  9import bauiv1 as bui
 10
 11if TYPE_CHECKING:
 12    from typing import Any
 13
 14
 15class PurchaseWindow(bui.Window):
 16    """Window for purchasing one or more items."""
 17
 18    def __init__(
 19        self,
 20        items: list[str],
 21        transition: str = 'in_right',
 22        header_text: bui.Lstr | None = None,
 23    ):
 24        from bauiv1lib.store.item import instantiate_store_item_display
 25
 26        plus = bui.app.plus
 27        assert plus is not None
 28
 29        assert bui.app.classic is not None
 30        store = bui.app.classic.store
 31
 32        if header_text is None:
 33            header_text = bui.Lstr(
 34                resource='unlockThisText',
 35                fallback_resource='unlockThisInTheStoreText',
 36            )
 37        if len(items) != 1:
 38            raise ValueError('expected exactly 1 item')
 39        self._items = list(items)
 40        self._width = 580
 41        self._height = 520
 42        uiscale = bui.app.ui_v1.uiscale
 43        super().__init__(
 44            root_widget=bui.containerwidget(
 45                size=(self._width, self._height),
 46                transition=transition,
 47                toolbar_visibility='menu_store',
 48                scale=(
 49                    1.2
 50                    if uiscale is bui.UIScale.SMALL
 51                    else 1.1 if uiscale is bui.UIScale.MEDIUM else 1.0
 52                ),
 53                stack_offset=(
 54                    (0, -15) if uiscale is bui.UIScale.SMALL else (0, 0)
 55                ),
 56            )
 57        )
 58        self._is_double = False
 59        self._title_text = bui.textwidget(
 60            parent=self._root_widget,
 61            position=(self._width * 0.5, self._height - 30),
 62            size=(0, 0),
 63            text=header_text,
 64            h_align='center',
 65            v_align='center',
 66            maxwidth=self._width * 0.9 - 120,
 67            scale=1.2,
 68            color=(1, 0.8, 0.3, 1),
 69        )
 70        size = store.get_store_item_display_size(items[0])
 71        display: dict[str, Any] = {}
 72        instantiate_store_item_display(
 73            items[0],
 74            display,
 75            parent_widget=self._root_widget,
 76            b_pos=(
 77                self._width * 0.5
 78                - size[0] * 0.5
 79                + 10
 80                - ((size[0] * 0.5 + 30) if self._is_double else 0),
 81                self._height * 0.5
 82                - size[1] * 0.5
 83                + 30
 84                + (20 if self._is_double else 0),
 85            ),
 86            b_width=size[0],
 87            b_height=size[1],
 88            button=False,
 89        )
 90
 91        # Wire up the parts we need.
 92        if self._is_double:
 93            pass  # not working
 94        else:
 95            if self._items == ['pro']:
 96                price_str = plus.get_price(self._items[0])
 97                pyoffs = -15
 98            else:
 99                pyoffs = 0
100                price = self._price = plus.get_v1_account_misc_read_val(
101                    'price.' + str(items[0]), -1
102                )
103                price_str = bui.charstr(bui.SpecialChar.TICKET) + str(price)
104            self._price_text = bui.textwidget(
105                parent=self._root_widget,
106                position=(self._width * 0.5, 150 + pyoffs),
107                size=(0, 0),
108                text=price_str,
109                h_align='center',
110                v_align='center',
111                maxwidth=self._width * 0.9,
112                scale=1.4,
113                color=(0.2, 1, 0.2),
114            )
115
116        self._update_timer = bui.AppTimer(
117            1.0, bui.WeakCall(self._update), repeat=True
118        )
119
120        self._cancel_button = bui.buttonwidget(
121            parent=self._root_widget,
122            position=(50, 40),
123            size=(150, 60),
124            scale=1.0,
125            on_activate_call=self._cancel,
126            autoselect=True,
127            label=bui.Lstr(resource='cancelText'),
128        )
129        self._purchase_button = bui.buttonwidget(
130            parent=self._root_widget,
131            position=(self._width - 200, 40),
132            size=(150, 60),
133            scale=1.0,
134            on_activate_call=self._purchase,
135            autoselect=True,
136            label=bui.Lstr(resource='store.purchaseText'),
137        )
138
139        bui.containerwidget(
140            edit=self._root_widget,
141            cancel_button=self._cancel_button,
142            start_button=self._purchase_button,
143            selected_child=self._purchase_button,
144        )
145
146    def _update(self) -> None:
147        can_die = False
148
149        plus = bui.app.plus
150        assert plus is not None
151
152        # We go away if we see that our target item is owned.
153        if self._items == ['pro']:
154            assert bui.app.classic is not None
155            if bui.app.classic.accounts.have_pro():
156                can_die = True
157        else:
158            if plus.get_v1_account_product_purchased(self._items[0]):
159                can_die = True
160
161        if can_die:
162            bui.containerwidget(edit=self._root_widget, transition='out_left')
163
164    def _purchase(self) -> None:
165
166        plus = bui.app.plus
167        assert plus is not None
168
169        if self._items == ['pro']:
170            plus.purchase('pro')
171        else:
172            ticket_count: int | None
173            try:
174                ticket_count = plus.get_v1_account_ticket_count()
175            except Exception:
176                ticket_count = None
177            if ticket_count is not None and ticket_count < self._price:
178                bui.getsound('error').play()
179                bui.screenmessage(
180                    bui.Lstr(resource='notEnoughTicketsText'),
181                    color=(1, 0, 0),
182                )
183                # gettickets.show_get_tickets_prompt()
184                return
185
186            def do_it() -> None:
187                assert plus is not None
188
189                plus.in_game_purchase(self._items[0], self._price)
190
191            bui.getsound('swish').play()
192            do_it()
193
194    def _cancel(self) -> None:
195        bui.containerwidget(edit=self._root_widget, transition='out_right')
class PurchaseWindow(bauiv1._uitypes.Window):
 16class PurchaseWindow(bui.Window):
 17    """Window for purchasing one or more items."""
 18
 19    def __init__(
 20        self,
 21        items: list[str],
 22        transition: str = 'in_right',
 23        header_text: bui.Lstr | None = None,
 24    ):
 25        from bauiv1lib.store.item import instantiate_store_item_display
 26
 27        plus = bui.app.plus
 28        assert plus is not None
 29
 30        assert bui.app.classic is not None
 31        store = bui.app.classic.store
 32
 33        if header_text is None:
 34            header_text = bui.Lstr(
 35                resource='unlockThisText',
 36                fallback_resource='unlockThisInTheStoreText',
 37            )
 38        if len(items) != 1:
 39            raise ValueError('expected exactly 1 item')
 40        self._items = list(items)
 41        self._width = 580
 42        self._height = 520
 43        uiscale = bui.app.ui_v1.uiscale
 44        super().__init__(
 45            root_widget=bui.containerwidget(
 46                size=(self._width, self._height),
 47                transition=transition,
 48                toolbar_visibility='menu_store',
 49                scale=(
 50                    1.2
 51                    if uiscale is bui.UIScale.SMALL
 52                    else 1.1 if uiscale is bui.UIScale.MEDIUM else 1.0
 53                ),
 54                stack_offset=(
 55                    (0, -15) if uiscale is bui.UIScale.SMALL else (0, 0)
 56                ),
 57            )
 58        )
 59        self._is_double = False
 60        self._title_text = bui.textwidget(
 61            parent=self._root_widget,
 62            position=(self._width * 0.5, self._height - 30),
 63            size=(0, 0),
 64            text=header_text,
 65            h_align='center',
 66            v_align='center',
 67            maxwidth=self._width * 0.9 - 120,
 68            scale=1.2,
 69            color=(1, 0.8, 0.3, 1),
 70        )
 71        size = store.get_store_item_display_size(items[0])
 72        display: dict[str, Any] = {}
 73        instantiate_store_item_display(
 74            items[0],
 75            display,
 76            parent_widget=self._root_widget,
 77            b_pos=(
 78                self._width * 0.5
 79                - size[0] * 0.5
 80                + 10
 81                - ((size[0] * 0.5 + 30) if self._is_double else 0),
 82                self._height * 0.5
 83                - size[1] * 0.5
 84                + 30
 85                + (20 if self._is_double else 0),
 86            ),
 87            b_width=size[0],
 88            b_height=size[1],
 89            button=False,
 90        )
 91
 92        # Wire up the parts we need.
 93        if self._is_double:
 94            pass  # not working
 95        else:
 96            if self._items == ['pro']:
 97                price_str = plus.get_price(self._items[0])
 98                pyoffs = -15
 99            else:
100                pyoffs = 0
101                price = self._price = plus.get_v1_account_misc_read_val(
102                    'price.' + str(items[0]), -1
103                )
104                price_str = bui.charstr(bui.SpecialChar.TICKET) + str(price)
105            self._price_text = bui.textwidget(
106                parent=self._root_widget,
107                position=(self._width * 0.5, 150 + pyoffs),
108                size=(0, 0),
109                text=price_str,
110                h_align='center',
111                v_align='center',
112                maxwidth=self._width * 0.9,
113                scale=1.4,
114                color=(0.2, 1, 0.2),
115            )
116
117        self._update_timer = bui.AppTimer(
118            1.0, bui.WeakCall(self._update), repeat=True
119        )
120
121        self._cancel_button = bui.buttonwidget(
122            parent=self._root_widget,
123            position=(50, 40),
124            size=(150, 60),
125            scale=1.0,
126            on_activate_call=self._cancel,
127            autoselect=True,
128            label=bui.Lstr(resource='cancelText'),
129        )
130        self._purchase_button = bui.buttonwidget(
131            parent=self._root_widget,
132            position=(self._width - 200, 40),
133            size=(150, 60),
134            scale=1.0,
135            on_activate_call=self._purchase,
136            autoselect=True,
137            label=bui.Lstr(resource='store.purchaseText'),
138        )
139
140        bui.containerwidget(
141            edit=self._root_widget,
142            cancel_button=self._cancel_button,
143            start_button=self._purchase_button,
144            selected_child=self._purchase_button,
145        )
146
147    def _update(self) -> None:
148        can_die = False
149
150        plus = bui.app.plus
151        assert plus is not None
152
153        # We go away if we see that our target item is owned.
154        if self._items == ['pro']:
155            assert bui.app.classic is not None
156            if bui.app.classic.accounts.have_pro():
157                can_die = True
158        else:
159            if plus.get_v1_account_product_purchased(self._items[0]):
160                can_die = True
161
162        if can_die:
163            bui.containerwidget(edit=self._root_widget, transition='out_left')
164
165    def _purchase(self) -> None:
166
167        plus = bui.app.plus
168        assert plus is not None
169
170        if self._items == ['pro']:
171            plus.purchase('pro')
172        else:
173            ticket_count: int | None
174            try:
175                ticket_count = plus.get_v1_account_ticket_count()
176            except Exception:
177                ticket_count = None
178            if ticket_count is not None and ticket_count < self._price:
179                bui.getsound('error').play()
180                bui.screenmessage(
181                    bui.Lstr(resource='notEnoughTicketsText'),
182                    color=(1, 0, 0),
183                )
184                # gettickets.show_get_tickets_prompt()
185                return
186
187            def do_it() -> None:
188                assert plus is not None
189
190                plus.in_game_purchase(self._items[0], self._price)
191
192            bui.getsound('swish').play()
193            do_it()
194
195    def _cancel(self) -> None:
196        bui.containerwidget(edit=self._root_widget, transition='out_right')

Window for purchasing one or more items.

PurchaseWindow( items: list[str], transition: str = 'in_right', header_text: babase.Lstr | None = None)
 19    def __init__(
 20        self,
 21        items: list[str],
 22        transition: str = 'in_right',
 23        header_text: bui.Lstr | None = None,
 24    ):
 25        from bauiv1lib.store.item import instantiate_store_item_display
 26
 27        plus = bui.app.plus
 28        assert plus is not None
 29
 30        assert bui.app.classic is not None
 31        store = bui.app.classic.store
 32
 33        if header_text is None:
 34            header_text = bui.Lstr(
 35                resource='unlockThisText',
 36                fallback_resource='unlockThisInTheStoreText',
 37            )
 38        if len(items) != 1:
 39            raise ValueError('expected exactly 1 item')
 40        self._items = list(items)
 41        self._width = 580
 42        self._height = 520
 43        uiscale = bui.app.ui_v1.uiscale
 44        super().__init__(
 45            root_widget=bui.containerwidget(
 46                size=(self._width, self._height),
 47                transition=transition,
 48                toolbar_visibility='menu_store',
 49                scale=(
 50                    1.2
 51                    if uiscale is bui.UIScale.SMALL
 52                    else 1.1 if uiscale is bui.UIScale.MEDIUM else 1.0
 53                ),
 54                stack_offset=(
 55                    (0, -15) if uiscale is bui.UIScale.SMALL else (0, 0)
 56                ),
 57            )
 58        )
 59        self._is_double = False
 60        self._title_text = bui.textwidget(
 61            parent=self._root_widget,
 62            position=(self._width * 0.5, self._height - 30),
 63            size=(0, 0),
 64            text=header_text,
 65            h_align='center',
 66            v_align='center',
 67            maxwidth=self._width * 0.9 - 120,
 68            scale=1.2,
 69            color=(1, 0.8, 0.3, 1),
 70        )
 71        size = store.get_store_item_display_size(items[0])
 72        display: dict[str, Any] = {}
 73        instantiate_store_item_display(
 74            items[0],
 75            display,
 76            parent_widget=self._root_widget,
 77            b_pos=(
 78                self._width * 0.5
 79                - size[0] * 0.5
 80                + 10
 81                - ((size[0] * 0.5 + 30) if self._is_double else 0),
 82                self._height * 0.5
 83                - size[1] * 0.5
 84                + 30
 85                + (20 if self._is_double else 0),
 86            ),
 87            b_width=size[0],
 88            b_height=size[1],
 89            button=False,
 90        )
 91
 92        # Wire up the parts we need.
 93        if self._is_double:
 94            pass  # not working
 95        else:
 96            if self._items == ['pro']:
 97                price_str = plus.get_price(self._items[0])
 98                pyoffs = -15
 99            else:
100                pyoffs = 0
101                price = self._price = plus.get_v1_account_misc_read_val(
102                    'price.' + str(items[0]), -1
103                )
104                price_str = bui.charstr(bui.SpecialChar.TICKET) + str(price)
105            self._price_text = bui.textwidget(
106                parent=self._root_widget,
107                position=(self._width * 0.5, 150 + pyoffs),
108                size=(0, 0),
109                text=price_str,
110                h_align='center',
111                v_align='center',
112                maxwidth=self._width * 0.9,
113                scale=1.4,
114                color=(0.2, 1, 0.2),
115            )
116
117        self._update_timer = bui.AppTimer(
118            1.0, bui.WeakCall(self._update), repeat=True
119        )
120
121        self._cancel_button = bui.buttonwidget(
122            parent=self._root_widget,
123            position=(50, 40),
124            size=(150, 60),
125            scale=1.0,
126            on_activate_call=self._cancel,
127            autoselect=True,
128            label=bui.Lstr(resource='cancelText'),
129        )
130        self._purchase_button = bui.buttonwidget(
131            parent=self._root_widget,
132            position=(self._width - 200, 40),
133            size=(150, 60),
134            scale=1.0,
135            on_activate_call=self._purchase,
136            autoselect=True,
137            label=bui.Lstr(resource='store.purchaseText'),
138        )
139
140        bui.containerwidget(
141            edit=self._root_widget,
142            cancel_button=self._cancel_button,
143            start_button=self._purchase_button,
144            selected_child=self._purchase_button,
145        )