Will it blend?
[koperkapel.git] / koperkapel / scenes / roach_management.py
1 """ Roach management scene. """
2
3 from pygame.constants import BLEND_RGBA_MULT, BLEND_RGBA_SUB
4 from pgzero.constants import keys, mouse
5 from pgzero.actor import Actor
6 from pgzero.screen import Screen
7 from ..actors.buttons import ImageButton
8 from ..actors.surf import SurfActor
9 from ..constants import WIDTH, HEIGHT
10 from ..roaches import big_roaches, roach_by_name
11 from ..serums import big_serums, roach_is_serumless, SERUMS
12 from ..vehicles.base import Vehicle
13 from .base import Scene, ChangeSceneEvent
14
15
16 TOOLBAR_LEFT_X = WIDTH * 3 // 4
17 TOOLBAR_TOP_Y = 0
18 TOOLBAR_MID_Y = HEIGHT * 1 // 2
19 VEHICLE_MID_X = WIDTH * 3 // 8
20 VEHICLE_MID_Y = HEIGHT * 1 // 2
21 BUTTON_INSET = (20, 20)
22 SERUM_OFFSET = (0, 20)
23 ROACH_PAD_OFFSET = (0, 20)
24
25
26 def inset_button(pos, d):
27     return (
28         pos[0] + d[0] * BUTTON_INSET[0],
29         pos[1] + d[1] * BUTTON_INSET[1])
30
31
32 class RoachesScene(Scene):
33     """ Roach management scene. """
34
35     def __init__(self, level_scene=None):
36         super().__init__()
37         self._level_scene = level_scene
38         self._vehicle = None
39         self._seat_pos = 0
40         self._outside_roach = None
41         self._outside_roach_pos = 0
42         self._inventory_pos = 0
43         self._inventory_item = None
44         self._roach_layer = self.actors.add_layer("roaches", level=10)
45         self._inventory_layer = self.actors.add_layer("inventory", level=10)
46         self._pad_layer = self.actors.add_layer("pads", level=5)
47         self._seat_layer = self.actors.add_layer("seats", level=5)
48         self._button_layer = self.actors.add_layer("buttons", level=6)
49         self._init_pads()
50         self._init_buttons()
51         self._init_serums()
52
53     def enter(self, world):
54         self._vehicle = Vehicle.current(world)
55         self._update_calls = []
56         self._init_bg()
57         self._init_seats()
58         self._init_roaches(world)
59         self._update_inventory(world)
60
61     def _init_bg(self):
62         self.actors.default.clear()
63         overlay = self._vehicle.roach_management_overlay()
64         base = overlay.copy()
65         if self._level_scene is not None:
66             base.fill((0, 0, 0))
67             self._level_scene.draw(Screen(base))
68         else:
69             base.fill((10, 10, 10))
70         base.blit(overlay, (0, 0))
71         frame = self._vehicle.roach_management_frame()
72         if frame is not None:
73             frame = frame.copy()
74             frame.fill((255, 255, 255, 8), None, BLEND_RGBA_MULT)
75             frame_rect = frame.get_rect()
76             base.blit(frame, (
77                 VEHICLE_MID_X - frame_rect.w // 2,
78                 VEHICLE_MID_Y - frame_rect.h // 2),
79                 None, BLEND_RGBA_SUB)
80         self.actors.default.add(SurfActor(base))
81
82     def _init_seats(self):
83         self._seat_layer.clear()
84         for seat in self._vehicle.seats:
85             seat_actor = self._seat_layer.add(seat.actor())
86             seat_actor.pos = (
87                 seat.vehicle_pos[0] + VEHICLE_MID_X,
88                 seat.vehicle_pos[1] + VEHICLE_MID_Y)
89         self._seat_layer[self._seat_pos].selected = True
90
91     def _init_roaches(self, world):
92         self._roach_actors = {}
93         for roach in world.roaches:
94             self._roach_actors[roach.name] = big_roaches.assemble(roach)
95
96     def _init_serums(self):
97         self._serum_actors = {
98             serum: big_serums.assemble(serum) for serum in SERUMS}
99
100     def _update_inventory(self, world):
101         self._inventory_layer.clear()
102         serums = world.serums
103         if not serums:
104             return
105         self._inventory_pos %= len(serums)
106         serum_actor = self._serum_actors[serums[self._inventory_pos]]
107         inv_pad_centre = self._inventory_pad.center
108         serum_actor.pos = (
109             inv_pad_centre[0] + SERUM_OFFSET[0],
110             inv_pad_centre[1] + SERUM_OFFSET[1])
111         self._inventory_layer.add(serum_actor)
112
113     def _update_roaches(self, world):
114         self._roach_layer.clear()
115         seating = self._vehicle.seating(world)
116         self._outside_roaches = []
117         for roach in world.roaches:
118             seat_pos = seating.get(roach.name)
119             if seat_pos is not None:
120                 roach_actor = self._roach_actors[roach.name]
121                 roach_actor.pos = self._seat_layer[seat_pos].pos
122                 self._roach_layer.add(roach_actor)
123             else:
124                 self._outside_roaches.append(roach.name)
125         if self._outside_roaches:
126             self._outside_roach_pos %= len(self._outside_roaches)
127             roach_actor = self._roach_actors[
128                 self._outside_roaches[self._outside_roach_pos]]
129             roach_pad_center = self._roach_pad.center
130             roach_actor.pos = (
131                 roach_pad_center[0] + ROACH_PAD_OFFSET[0],
132                 roach_pad_center[1] + ROACH_PAD_OFFSET[1])
133             self._roach_layer.add(roach_actor)
134         else:
135             self._outside_roach_pos = 0
136
137     def _init_pads(self):
138         self._roach_pad = self._pad_layer.add(
139             Actor("roach_management/roach_pad", anchor=("left", "bottom")))
140         self._roach_pad.pos = (TOOLBAR_LEFT_X, TOOLBAR_MID_Y)
141         self._inventory_pad = self._pad_layer.add(
142             Actor("roach_management/inventory_pad", anchor=("left", "top")))
143         self._inventory_pad.pos = (TOOLBAR_LEFT_X, TOOLBAR_MID_Y)
144
145     def _add_button(self, name, anchor, inset, pos, action):
146         button = self._button_layer.add(
147             ImageButton(name, anchor=anchor, action=action))
148         button.pos = inset_button(pos, inset)
149         return button
150
151     def _init_buttons(self):
152         self._button_layer.clear()
153         self._add_button(
154             "roach_management/left_button", ("left", "bottom"), (1, -1),
155             self._roach_pad.bottomleft, self._roach_left)
156
157         self._add_button(
158             "roach_management/right_button", ("right", "bottom"), (-1, -1),
159             self._roach_pad.bottomright, self._roach_right)
160
161         self._add_button(
162             "roach_management/left_button", ("left", "bottom"), (1, -1),
163             self._inventory_pad.bottomleft, self._inventory_left)
164
165         self._add_button(
166             "roach_management/right_button", ("right", "bottom"), (-1, -1),
167             self._inventory_pad.bottomright, self._inventory_right)
168
169         self._add_button(
170             "roach_management/eject_button", ("right", "top"), (-1, 1),
171             (TOOLBAR_LEFT_X, TOOLBAR_TOP_Y), self._eject_roach)
172
173     def _roach_left(self):
174         self._outside_roach_pos -= 1
175
176     def _roach_right(self):
177         self._outside_roach_pos += 1
178
179     def _inventory_left(self):
180         self._inventory_pos -= 1
181
182     def _inventory_right(self):
183         self._inventory_pos += 1
184
185     def _select_seat(self, seat_pos):
186         self._seat_layer[self._seat_pos].selected = False
187         self._seat_pos = seat_pos
188         self._seat_layer[self._seat_pos].selected = True
189
190     def _eject_roach(self, world=None):
191         if world is None:
192             self._update_calls.append(self._eject_roach)
193             return
194         self._vehicle.seat_roach(world, None, self._seat_pos)
195
196     def _click_roach_pad(self, world=None):
197         if world is None:
198             self._update_calls.append(self._click_roach_pad)
199             return
200         if self._outside_roaches:
201             roach = self._outside_roaches[self._outside_roach_pos]
202             self._vehicle.seat_roach(world, roach, self._seat_pos)
203
204     def _click_inventory_pad(self, world=None):
205         if world is None:
206             self._update_calls.append(self._click_inventory_pad)
207             return
208         roach_name = self._vehicle.roach_at(world, self._seat_pos)
209         if roach_name is None:
210             return
211         roach = roach_by_name(world, roach_name)
212         if roach is None:
213             return
214         serums = list(world.serums)
215         if self._inventory_pos >= len(serums):
216             return
217         serum = serums.pop(self._inventory_pos)
218         if roach_is_serumless(roach):
219             roach[serum] = True
220             world.serums = serums
221             self._update_calls.append((self._update_roach_actor, roach_name))
222
223     def _update_roach_actor(self, world, roach_name):
224         roach = roach_by_name(world, roach_name)
225         self._roach_actors[roach_name] = big_roaches.assemble(roach)
226
227     def update(self, world, engine, dt):
228         update_calls, self._update_calls = self._update_calls, []
229         while update_calls:
230             f, args = update_calls.pop(), ()
231             if type(f) is tuple:
232                 f, args = f[0], f[1:]
233             f(world, *args)
234         events = world.pop_events()
235         self._update_inventory(world)
236         self._update_roaches(world)
237         return events
238
239     def on_key_down(self, key, mod, unicode):
240         if key in (keys.ESCAPE, keys.Z):
241             if self._level_scene is None:
242                 from .menu import MenuScene
243                 return [ChangeSceneEvent(MenuScene())]
244             return [ChangeSceneEvent(self._level_scene)]
245
246     def on_mouse_down(self, pos, button):
247         if button == mouse.LEFT:
248             for actor in self.actors.buttons:
249                 if actor.collidepoint(pos):
250                     return actor.action()
251             for i, actor in enumerate(self.actors.seats):
252                 if actor.collidepoint(pos):
253                     return self._select_seat(i)
254             if self._roach_pad.collidepoint(pos):
255                 return self._click_roach_pad()
256             if self._inventory_pad.collidepoint(pos):
257                 return self._click_inventory_pad()