Commit | Line | Data |
---|---|---|
31237524 | 1 | from os.path import exists |
a0462193 | 2 | from os import makedirs |
c991401b | 3 | from logging import info |
3fe18d73 | 4 | from json import loads |
c991401b FC |
5 | from panda3d.core import AmbientLight, Texture, TextPropertiesManager, \ |
6 | TextNode, Spotlight, PerspectiveLens, BitMask32, NodePath | |
36099535 FC |
7 | from panda3d.bullet import BulletPlaneShape, BulletGhostNode |
8 | from direct.gui.OnscreenImage import OnscreenImage | |
2aaa10d3 FC |
9 | from direct.gui.OnscreenText import OnscreenText |
10 | from direct.gui.DirectGui import DirectButton, DirectFrame | |
36099535 | 11 | from direct.gui.DirectGuiGlobals import FLAT, DISABLED, NORMAL |
1be87278 | 12 | from direct.showbase.DirectObject import DirectObject |
fe0a68a0 FC |
13 | from direct.interval.IntervalGlobal import Sequence, Func |
14 | from direct.interval.LerpInterval import LerpFunctionInterval | |
4586cbf6 FC |
15 | from pmachines.items.background import Background |
16 | from pmachines.gui.sidepanel import SidePanel | |
98741d67 | 17 | from pmachines.items.box import Box, HitStrategy |
25c59f4a | 18 | from pmachines.items.basketball import Basketball |
98741d67 | 19 | from pmachines.items.domino import Domino, DownStrategy |
1f76fd96 FC |
20 | from pmachines.items.shelf import Shelf |
21 | from pmachines.items.teetertooter import TeeterTooter | |
b35b1f62 FC |
22 | from ya2.utils.cursor import MouseCursor |
23 | from ya2.p3d.gfx import P3dGfxMgr | |
2d1773b1 | 24 | from ya2.p3d.p3d import LibP3d |
1be87278 FC |
25 | |
26 | ||
27 | class Scene(DirectObject): | |
28 | ||
92c29685 | 29 | scenes_done = [] |
3fe18d73 | 30 | json = {} |
92c29685 FC |
31 | |
32 | def __init__(self, world, exit_cb, auto_close_instr, dbg_items, reload_cb, scenes, pos_mgr, testing, mouse_coords, persistent): | |
1be87278 FC |
33 | super().__init__() |
34 | self._world = world | |
5964572b | 35 | self._exit_cb = exit_cb |
ce302b41 | 36 | self._testing = testing |
7fa58640 | 37 | self._mouse_coords = mouse_coords |
31237524 | 38 | self._dbg_items = dbg_items |
9914cfc9 | 39 | self._reload_cb = reload_cb |
2d1773b1 FC |
40 | self._pos_mgr = pos_mgr |
41 | self._pos_mgr.reset() | |
6168d0c2 | 42 | self._scenes = scenes |
ce302b41 | 43 | self._start_evt_time = None |
fa3662a6 | 44 | self._enforce_res = '' |
92c29685 | 45 | self.__persistent = persistent |
fa3662a6 | 46 | self.accept('enforce_res', self.enforce_res) |
1be87278 | 47 | self._set_camera() |
79f81d48 | 48 | self._cursor = MouseCursor( |
7e487769 | 49 | 'assets/images/buttons/arrowUpLeft.dds', (.04, 1, .04), (.5, .5, .5, 1), |
79f81d48 | 50 | (.01, .01)) |
36099535 | 51 | self._set_gui() |
1be87278 FC |
52 | self._set_lights() |
53 | self._set_input() | |
c8d8653f | 54 | self._set_mouse_plane() |
d5932612 | 55 | self.items = [] |
067a36db | 56 | self._test_items = [] |
d5932612 | 57 | self.reset() |
0a0994e4 | 58 | self._state = 'init' |
9830561d | 59 | self._paused = False |
7c0a81ae | 60 | self._item_active = None |
e669403e | 61 | if auto_close_instr: |
32aa4dae | 62 | self.__store_state() |
e669403e FC |
63 | self.__restore_state() |
64 | else: | |
65 | self._set_instructions() | |
5964572b FC |
66 | self._bg = Background() |
67 | self._side_panel = SidePanel(world, self._mouse_plane_node, (-5, 4), (-3, 1), 1, self.items) | |
68 | self._scene_tsk = taskMgr.add(self.on_frame, 'on_frame') | |
69 | ||
aa577aeb FC |
70 | @classmethod |
71 | def name(cls): | |
72 | if not cls.json: | |
73 | with open(cls.filename) as f: | |
74 | cls.json = loads(f.read()) | |
75 | return _(cls.json['name']) | |
8c9bf90e | 76 | |
3fe18d73 FC |
77 | @classmethod |
78 | def version(cls): | |
79 | if not cls.json: | |
80 | with open(cls.filename) as f: | |
81 | cls.json = loads(f.read()) | |
82 | return cls.json['version'] | |
83 | ||
92c29685 FC |
84 | @classmethod |
85 | def is_done(cls): | |
3fe18d73 | 86 | return bool([(name, version) for name, version in cls.scenes_done if cls.__name__ == name and cls.version() == version]) |
92c29685 | 87 | |
0eff64a3 | 88 | def _instr_txt(self): |
aa577aeb FC |
89 | txt = _('Scene: ') + self.name() + '\n\n' |
90 | for instruction_line in self.__class__.json['instructions']: | |
91 | instruction_line = self.__process_json_escape(instruction_line) | |
92 | txt += _(instruction_line) | |
93 | return txt | |
94 | ||
95 | def __process_json_escape(self, string): | |
96 | return bytes(string, 'utf-8').decode('unicode-escape') | |
0eff64a3 FC |
97 | |
98 | def _set_items(self): | |
99 | self.items = [] | |
067a36db | 100 | self._test_items = [] |
25c59f4a FC |
101 | if not self.__class__.json: |
102 | with open(self.__class__.filename) as f: | |
103 | self.__class__.json = loads(f.read()) | |
104 | for item in self.__class__.json['start_items']: | |
105 | args = { | |
106 | 'world': self._world, | |
107 | 'plane_node': self._mouse_plane_node, | |
108 | 'cb_inst': self.cb_inst, | |
109 | 'curr_bottom': self.current_bottom, | |
110 | 'repos': self.repos, | |
98741d67 FC |
111 | 'count': item['count'], |
112 | 'json': item} | |
25c59f4a FC |
113 | if 'mass' in item: |
114 | args['mass'] = item['mass'] | |
115 | if 'friction' in item: | |
116 | args['friction'] = item['friction'] | |
117 | self.items += [self.__code2class(item['class'])(**args)] | |
1f76fd96 FC |
118 | for item in self.__class__.json['fixed_items']: |
119 | args = { | |
120 | 'world': self._world, | |
121 | 'plane_node': self._mouse_plane_node, | |
122 | 'cb_inst': self.cb_inst, | |
123 | 'curr_bottom': self.current_bottom, | |
124 | 'repos': self.repos, | |
98741d67 | 125 | 'json': item, |
1f76fd96 FC |
126 | 'mass': 0} |
127 | args['pos'] = tuple(item['position']) | |
128 | if 'roll' in item: | |
129 | args['r'] = item['roll'] | |
130 | if 'restitution' in item: | |
131 | args['restitution'] = item['restitution'] | |
132 | if 'friction' in item: | |
133 | args['friction'] = item['friction'] | |
134 | self.items += [self.__code2class(item['class'])(**args)] | |
98741d67 FC |
135 | for item in self.__class__.json['scene_items']: |
136 | args = { | |
137 | 'world': self._world, | |
138 | 'plane_node': self._mouse_plane_node, | |
139 | 'cb_inst': self.cb_inst, | |
140 | 'curr_bottom': self.current_bottom, | |
141 | 'repos': self.repos, | |
142 | 'json': item} | |
143 | args['pos'] = tuple(item['position']) | |
144 | if 'mass' in item: | |
145 | args['mass'] = item['mass'] | |
146 | if 'friction' in item: | |
147 | args['friction'] = item['friction'] | |
148 | if 'roll' in item: | |
149 | args['r'] = item['roll'] | |
150 | if 'model_scale' in item: | |
151 | args['model_scale'] = item['model_scale'] | |
152 | self.items += [self.__code2class(item['class'])(**args)] | |
153 | if 'strategy' in item: | |
154 | match item['strategy']: | |
155 | case 'DownStrategy': | |
156 | self.items[-1].set_strategy(self.__code2class(item['strategy'])(self.items[-1]._np, *item['strategy_args'])) | |
157 | case 'HitStrategy': | |
158 | self.items[-1].set_strategy(self.__code2class(item['strategy'])(self.__item_with_id(item['strategy_args'][0]), self.items[-1].node, self.items[-1]._world)) | |
25c59f4a FC |
159 | |
160 | def __code2class(self, code): | |
161 | return { | |
162 | 'Box': Box, | |
163 | 'Basketball': Basketball, | |
1f76fd96 FC |
164 | 'Domino': Domino, |
165 | 'Shelf': Shelf, | |
98741d67 FC |
166 | 'TeeterTooter': TeeterTooter, |
167 | 'DownStrategy': DownStrategy, | |
168 | 'HitStrategy': HitStrategy | |
25c59f4a | 169 | }[code] |
0eff64a3 | 170 | |
98741d67 FC |
171 | def __item_with_id(self, id): |
172 | for item in self.items: | |
173 | if 'id' in item.json and item.json['id'] == id: | |
174 | return item | |
175 | ||
0d5a5427 | 176 | def screenshot(self, task=None): |
ecdf8933 FC |
177 | tex = Texture('screenshot') |
178 | buffer = base.win.make_texture_buffer('screenshot', 512, 512, tex, True ) | |
179 | cam = base.make_camera(buffer) | |
180 | cam.reparent_to(render) | |
181 | cam.node().get_lens().set_fov(base.camLens.get_fov()) | |
182 | cam.set_pos(0, -20, 0) | |
183 | cam.look_at(0, 0, 0) | |
184 | import simplepbr | |
185 | simplepbr.init( | |
186 | window=buffer, | |
187 | camera_node=cam, | |
188 | use_normal_maps=True, | |
189 | use_emission_maps=False, | |
190 | use_occlusion_maps=True, | |
191 | msaa_samples=4, | |
192 | enable_shadows=True) | |
193 | base.graphicsEngine.renderFrame() | |
194 | base.graphicsEngine.renderFrame() | |
63e7aeb2 | 195 | fname = self.__class__.__name__ |
40cc6e36 FC |
196 | if not exists('assets/images/scenes'): |
197 | makedirs('assets/images/scenes') | |
63e7aeb2 | 198 | buffer.save_screenshot('assets/images/scenes/%s.png' % fname) |
ecdf8933 FC |
199 | # img = DirectButton( |
200 | # frameTexture=buffer.get_texture(), relief=FLAT, | |
201 | # frameSize=(-.2, .2, -.2, .2)) | |
0d5a5427 | 202 | return buffer.get_texture() |
ecdf8933 | 203 | |
d5932612 FC |
204 | def current_bottom(self): |
205 | curr_bottom = 1 | |
206 | for item in self.items: | |
207 | if item.repos_done: | |
208 | curr_bottom = min(curr_bottom, item.get_bottom()) | |
209 | return curr_bottom | |
210 | ||
05fd23ec | 211 | def reset(self): |
d5932612 | 212 | [itm.destroy() for itm in self.items] |
067a36db FC |
213 | [itm.remove_node() for itm in self._test_items] |
214 | self.items = [] | |
215 | self._test_items = [] | |
0eff64a3 | 216 | self._set_items() |
067a36db | 217 | self._set_test_items() |
0a0994e4 | 218 | self._state = 'init' |
32aa4dae FC |
219 | self._commands = [] |
220 | self._command_idx = 0 | |
ce302b41 | 221 | self._start_evt_time = None |
b41381b2 FC |
222 | if hasattr(self, '_success_txt'): |
223 | self._success_txt.destroy() | |
224 | del self._success_txt | |
225 | self.__right_btn['state'] = NORMAL | |
05fd23ec | 226 | |
fa3662a6 FC |
227 | def enforce_res(self, val): |
228 | self._enforce_res = val | |
74ed9732 | 229 | info('enforce res: ' + val) |
fa3662a6 | 230 | |
5964572b | 231 | def destroy(self): |
fe0a68a0 | 232 | self.__intro_sequence.finish() |
fa3662a6 | 233 | self.ignore('enforce_res') |
5964572b FC |
234 | self._unset_gui() |
235 | self._unset_lights() | |
236 | self._unset_input() | |
237 | self._unset_mouse_plane() | |
238 | [itm.destroy() for itm in self.items] | |
067a36db | 239 | [itm.remove_node() for itm in self._test_items] |
5964572b FC |
240 | self._bg.destroy() |
241 | self._side_panel.destroy() | |
407412a5 | 242 | self._cursor.destroy() |
5964572b | 243 | taskMgr.remove(self._scene_tsk) |
9fc7f6fb FC |
244 | if hasattr(self, '_success_txt'): |
245 | self._success_txt.destroy() | |
1be87278 FC |
246 | |
247 | def _set_camera(self): | |
248 | base.camera.set_pos(0, -20, 0) | |
249 | base.camera.look_at(0, 0, 0) | |
fe0a68a0 FC |
250 | def camera_ani(t): |
251 | start_v = (1, -5, 1) | |
252 | end_v = (0, -20, 0) | |
253 | curr_pos = ( | |
254 | start_v[0] + (end_v[0] - start_v[0]) * t, | |
255 | start_v[1] + (end_v[1] - start_v[1]) * t, | |
256 | start_v[2] + (end_v[2] - start_v[2]) * t) | |
257 | base.camera.set_pos(*curr_pos) | |
258 | self.repos() | |
259 | camera_interval = LerpFunctionInterval( | |
260 | camera_ani, | |
261 | 1.2, | |
262 | 0, | |
263 | 1, | |
264 | blendType='easeInOut') | |
265 | self.__intro_sequence = Sequence( | |
266 | camera_interval, | |
267 | Func(self.repos)) | |
268 | self.__intro_sequence.start() | |
1be87278 | 269 | |
a0acba9a | 270 | def __load_img_btn(self, path, col): |
7e487769 | 271 | img = OnscreenImage('assets/images/buttons/%s.dds' % path) |
a0acba9a FC |
272 | img.set_transparency(True) |
273 | img.set_color(col) | |
274 | img.detach_node() | |
275 | return img | |
276 | ||
36099535 | 277 | def _set_gui(self): |
01b221a6 FC |
278 | def load_images_btn(path, col): |
279 | colors = { | |
280 | 'gray': [ | |
281 | (.6, .6, .6, 1), # ready | |
282 | (1, 1, 1, 1), # press | |
283 | (.8, .8, .8, 1), # rollover | |
284 | (.4, .4, .4, .4)], | |
285 | 'green': [ | |
286 | (.1, .68, .1, 1), | |
287 | (.1, 1, .1, 1), | |
288 | (.1, .84, .1, 1), | |
289 | (.4, .1, .1, .4)]}[col] | |
a0acba9a | 290 | return [self.__load_img_btn(path, col) for col in colors] |
36099535 FC |
291 | abl, abr = base.a2dBottomLeft, base.a2dBottomRight |
292 | btn_info = [ | |
5964572b | 293 | ('home', self.on_home, NORMAL, abl, 'gray'), |
9830561d | 294 | ('information', self._set_instructions, NORMAL, abl, 'gray'), |
01b221a6 | 295 | ('right', self.on_play, NORMAL, abr, 'green'), |
f26497a5 FC |
296 | #('next', self.on_next, DISABLED, abr, 'gray'), |
297 | #('previous', self.on_prev, DISABLED, abr, 'gray'), | |
298 | #('rewind', self.reset, NORMAL, abr, 'gray') | |
299 | ] | |
36099535 | 300 | num_l = num_r = 0 |
a0acba9a | 301 | btns = [] |
36099535 | 302 | for binfo in btn_info: |
01b221a6 | 303 | imgs = load_images_btn(binfo[0], binfo[4]) |
36099535 FC |
304 | if binfo[3] == base.a2dBottomLeft: |
305 | sign, num = 1, num_l | |
306 | num_l += 1 | |
307 | else: | |
308 | sign, num = -1, num_r | |
309 | num_r += 1 | |
310 | fcols = (.4, .4, .4, .14), (.3, .3, .3, .05) | |
311 | btn = DirectButton( | |
312 | image=imgs, scale=.05, pos=(sign * (.06 + .11 * num), 1, .06), | |
313 | parent=binfo[3], command=binfo[1], state=binfo[2], relief=FLAT, | |
54a1397e FC |
314 | frameColor=fcols[0] if binfo[2] == NORMAL else fcols[1], |
315 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
316 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
36099535 | 317 | btn.set_transparency(True) |
2d1773b1 | 318 | self._pos_mgr.register(binfo[0], LibP3d.wdg_pos(btn)) |
a0acba9a | 319 | btns += [btn] |
f26497a5 FC |
320 | self.__home_btn, self.__info_btn, self.__right_btn = btns |
321 | # , self.__next_btn, self.__prev_btn, self.__rewind_btn | |
31237524 FC |
322 | if self._dbg_items: |
323 | self._info_txt = OnscreenText( | |
324 | '', parent=base.a2dTopRight, scale=0.04, | |
325 | pos=(-.03, -.06), fg=(.9, .9, .9, 1), align=TextNode.A_right) | |
7fa58640 FC |
326 | if self._mouse_coords: |
327 | self._coords_txt = OnscreenText( | |
328 | '', parent=base.a2dTopRight, scale=0.04, | |
329 | pos=(-.03, -.12), fg=(.9, .9, .9, 1), align=TextNode.A_right) | |
330 | def update_coords(task): | |
331 | pos = None | |
332 | for hit in self._get_hits(): | |
333 | if hit.get_node() == self._mouse_plane_node: | |
334 | pos = hit.get_hit_pos() | |
335 | if pos: | |
336 | txt = '%s %s' % (round(pos.x, 3), | |
337 | round(pos.z, 3)) | |
338 | self._coords_txt['text'] = txt | |
339 | return task.cont | |
340 | self._coords_tsk = taskMgr.add(update_coords, 'update_coords') | |
36099535 | 341 | |
5964572b FC |
342 | def _unset_gui(self): |
343 | btns = [ | |
344 | self.__home_btn, self.__info_btn, self.__right_btn, | |
f26497a5 FC |
345 | #self.__next_btn, self.__prev_btn, self.__rewind_btn |
346 | ] | |
5964572b | 347 | [btn.destroy() for btn in btns] |
31237524 FC |
348 | if self._dbg_items: |
349 | self._info_txt.destroy() | |
7fa58640 FC |
350 | if self._mouse_coords: |
351 | taskMgr.remove(self._coords_tsk) | |
352 | self._coords_txt.destroy() | |
5964572b | 353 | |
64eae9c7 FC |
354 | def _set_spotlight(self, name, pos, look_at, color, shadows=False): |
355 | light = Spotlight(name) | |
356 | if shadows: | |
357 | light.setLens(PerspectiveLens()) | |
1be87278 | 358 | light_np = render.attach_new_node(light) |
64eae9c7 FC |
359 | light_np.set_pos(pos) |
360 | light_np.look_at(look_at) | |
1be87278 FC |
361 | light.set_color(color) |
362 | render.set_light(light_np) | |
5964572b | 363 | return light_np |
1be87278 FC |
364 | |
365 | def _set_lights(self): | |
366 | alight = AmbientLight('alight') # for ao | |
64eae9c7 FC |
367 | alight.set_color((.15, .15, .15, 1)) |
368 | self._alnp = render.attach_new_node(alight) | |
369 | render.set_light(self._alnp) | |
370 | self._key_light = self._set_spotlight( | |
371 | 'key light', (-5, -80, 5), (0, 0, 0), (2.8, 2.8, 2.8, 1)) | |
372 | self._shadow_light = self._set_spotlight( | |
373 | 'key light', (-5, -80, 5), (0, 0, 0), (.58, .58, .58, 1), True) | |
374 | self._shadow_light.node().set_shadow_caster(True, 2048, 2048) | |
375 | self._shadow_light.node().get_lens().set_film_size(2048, 2048) | |
376 | self._shadow_light.node().get_lens().set_near_far(1, 256) | |
377 | self._shadow_light.node().set_camera_mask(BitMask32(0x01)) | |
5964572b FC |
378 | |
379 | def _unset_lights(self): | |
64eae9c7 | 380 | for light in [self._alnp, self._key_light, self._shadow_light]: |
5964572b FC |
381 | render.clear_light(light) |
382 | light.remove_node() | |
1be87278 FC |
383 | |
384 | def _set_input(self): | |
49c79300 | 385 | self.accept('mouse1', self.on_click_l) |
c8d8653f | 386 | self.accept('mouse1-up', self.on_release) |
49c79300 FC |
387 | self.accept('mouse3', self.on_click_r) |
388 | self.accept('mouse3-up', self.on_release) | |
c8d8653f | 389 | |
5964572b FC |
390 | def _unset_input(self): |
391 | for evt in ['mouse1', 'mouse1-up', 'mouse3', 'mouse3-up']: | |
392 | self.ignore(evt) | |
393 | ||
c8d8653f | 394 | def _set_mouse_plane(self): |
651713a9 | 395 | shape = BulletPlaneShape((0, -1, 0), 0) |
c8d8653f FC |
396 | #self._mouse_plane_node = BulletRigidBodyNode('mouse plane') |
397 | self._mouse_plane_node = BulletGhostNode('mouse plane') | |
398 | self._mouse_plane_node.addShape(shape) | |
399 | #np = render.attachNewNode(self._mouse_plane_node) | |
400 | #self._world.attachRigidBody(self._mouse_plane_node) | |
06af3aa9 | 401 | self._world.attach_ghost(self._mouse_plane_node) |
1be87278 | 402 | |
5964572b FC |
403 | def _unset_mouse_plane(self): |
404 | self._world.remove_ghost(self._mouse_plane_node) | |
405 | ||
37ba71d8 FC |
406 | def _get_hits(self): |
407 | if not base.mouseWatcherNode.has_mouse(): return [] | |
13263131 | 408 | p_from, p_to = P3dGfxMgr.world_from_to(base.mouseWatcherNode.get_mouse()) |
37ba71d8 FC |
409 | return self._world.ray_test_all(p_from, p_to).get_hits() |
410 | ||
31237524 FC |
411 | def _update_info(self, item): |
412 | txt = '' | |
413 | if item: | |
414 | txt = '%.3f %.3f\n%.3f°' % ( | |
415 | item._np.get_x(), item._np.get_z(), item._np.get_r()) | |
416 | self._info_txt['text'] = txt | |
417 | ||
49c79300 | 418 | def _on_click(self, method): |
a0acba9a FC |
419 | if self._paused: |
420 | return | |
c8d8653f FC |
421 | for hit in self._get_hits(): |
422 | if hit.get_node() == self._mouse_plane_node: | |
423 | pos = hit.get_hit_pos() | |
37ba71d8 | 424 | for hit in self._get_hits(): |
ef3c36bf | 425 | for item in [i for i in self.items if hit.get_node() == i.node and i.interactable]: |
7c0a81ae FC |
426 | if not self._item_active: |
427 | self._item_active = item | |
49c79300 | 428 | getattr(item, method)(pos) |
79f81d48 | 429 | img = 'move' if method == 'on_click_l' else 'rotate' |
a6843832 | 430 | if not (img == 'rotate' and not item._instantiated): |
7e487769 | 431 | self._cursor.set_image('assets/images/buttons/%s.dds' % img) |
49c79300 FC |
432 | |
433 | def on_click_l(self): | |
434 | self._on_click('on_click_l') | |
435 | ||
436 | def on_click_r(self): | |
437 | self._on_click('on_click_r') | |
c8d8653f FC |
438 | |
439 | def on_release(self): | |
32aa4dae FC |
440 | if self._item_active and not self._item_active._first_command: |
441 | self._commands = self._commands[:self._command_idx] | |
442 | self._commands += [self._item_active] | |
443 | self._command_idx += 1 | |
f26497a5 FC |
444 | #self.__prev_btn['state'] = NORMAL |
445 | #fcols = (.4, .4, .4, .14), (.3, .3, .3, .05) | |
446 | #self.__prev_btn['frameColor'] = fcols[0] | |
447 | #if self._item_active._command_idx == len(self._item_active._commands) - 1: | |
448 | # self.__next_btn['state'] = DISABLED | |
449 | # self.__next_btn['frameColor'] = fcols[1] | |
7c0a81ae | 450 | self._item_active = None |
49c79300 | 451 | [item.on_release() for item in self.items] |
7e487769 | 452 | self._cursor.set_image('assets/images/buttons/arrowUpLeft.dds') |
37ba71d8 | 453 | |
e1438449 | 454 | def repos(self): |
d5932612 FC |
455 | for item in self.items: |
456 | item.repos_done = False | |
e1438449 | 457 | self.items = sorted(self.items, key=lambda itm: itm.__class__.__name__) |
651713a9 | 458 | [item.on_aspect_ratio_changed() for item in self.items] |
a5dc83f4 | 459 | self._side_panel.update(self.items) |
7790323d | 460 | max_x = -float('inf') |
e1438449 FC |
461 | for item in self.items: |
462 | if not item._instantiated: | |
463 | max_x = max(item._np.get_x(), max_x) | |
464 | for item in self.items: | |
465 | if not item._instantiated: | |
466 | item.repos_x(max_x) | |
467 | ||
468 | def on_aspect_ratio_changed(self): | |
469 | self.repos() | |
651713a9 | 470 | |
0a0994e4 | 471 | def _win_condition(self): |
b8161d21 | 472 | return all(itm.strategy.win_condition() for itm in self.items) and not self._paused |
0e86689f | 473 | |
0a0994e4 FC |
474 | def _fail_condition(self): |
475 | return all(itm.fail_condition() for itm in self.items) and not self._paused and self._state == 'playing' | |
476 | ||
37ba71d8 | 477 | def on_frame(self, task): |
c8d8653f FC |
478 | hits = self._get_hits() |
479 | pos = None | |
480 | for hit in self._get_hits(): | |
481 | if hit.get_node() == self._mouse_plane_node: | |
482 | pos = hit.get_hit_pos() | |
483 | hit_nodes = [hit.get_node() for hit in hits] | |
7c0a81ae FC |
484 | if self._item_active: |
485 | items_hit = [self._item_active] | |
486 | else: | |
487 | items_hit = [itm for itm in self.items if itm.node in hit_nodes] | |
37ba71d8 FC |
488 | items_no_hit = [itm for itm in self.items if itm not in items_hit] |
489 | [itm.on_mouse_on() for itm in items_hit] | |
490 | [itm.on_mouse_off() for itm in items_no_hit] | |
7c0a81ae FC |
491 | if pos and self._item_active: |
492 | self._item_active.on_mouse_move(pos) | |
31237524 FC |
493 | if self._dbg_items: |
494 | self._update_info(items_hit[0] if items_hit else None) | |
0a0994e4 | 495 | if self._win_condition(): |
ce302b41 | 496 | self._start_evt_time = None |
fa3662a6 | 497 | self._set_fail() if self._enforce_res == 'fail' else self._set_win() |
0a0994e4 | 498 | elif self._state == 'playing' and self._fail_condition(): |
ce302b41 FC |
499 | self._start_evt_time = None |
500 | self._set_win() if self._enforce_res == 'win' else self._set_fail() | |
501 | elif self._testing and self._start_evt_time and globalClock.getFrameTime() - self._start_evt_time > 5.0: | |
502 | self._start_evt_time = None | |
fa3662a6 | 503 | self._set_win() if self._enforce_res == 'win' else self._set_fail() |
93cf86b2 FC |
504 | if any(itm._overlapping for itm in self.items): |
505 | self._cursor.cursor_img.img.set_color(.9, .1, .1, 1) | |
506 | else: | |
507 | self._cursor.cursor_img.img.set_color(.9, .9, .9, 1) | |
37ba71d8 | 508 | return task.cont |
c8d8653f | 509 | |
7850ccc4 FC |
510 | def cb_inst(self, item): |
511 | self.items += [item] | |
512 | ||
c8d8653f | 513 | def on_play(self): |
0a0994e4 | 514 | self._state = 'playing' |
f26497a5 FC |
515 | #self.__prev_btn['state'] = DISABLED |
516 | #self.__next_btn['state'] = DISABLED | |
b41381b2 | 517 | self.__right_btn['state'] = DISABLED |
c8d8653f | 518 | [itm.play() for itm in self.items] |
ce302b41 | 519 | self._start_evt_time = globalClock.getFrameTime() |
36099535 FC |
520 | |
521 | def on_next(self): | |
32aa4dae FC |
522 | self._commands[self._command_idx].redo() |
523 | self._command_idx += 1 | |
c991401b | 524 | #fcols = (.4, .4, .4, .14), (.3, .3, .3, .05) |
f26497a5 FC |
525 | #self.__prev_btn['state'] = NORMAL |
526 | #self.__prev_btn['frameColor'] = fcols[0] | |
c991401b | 527 | #more_commands = self._command_idx < len(self._commands) |
f26497a5 FC |
528 | #self.__next_btn['state'] = NORMAL if more_commands else DISABLED |
529 | #self.__next_btn['frameColor'] = fcols[0] if more_commands else fcols[1] | |
36099535 FC |
530 | |
531 | def on_prev(self): | |
32aa4dae FC |
532 | self._command_idx -= 1 |
533 | self._commands[self._command_idx].undo() | |
c991401b | 534 | #fcols = (.4, .4, .4, .14), (.3, .3, .3, .05) |
f26497a5 FC |
535 | #self.__next_btn['state'] = NORMAL |
536 | #self.__next_btn['frameColor'] = fcols[0] | |
537 | #self.__prev_btn['state'] = NORMAL if self._command_idx else DISABLED | |
538 | #self.__prev_btn['frameColor'] = fcols[0] if self._command_idx else fcols[1] | |
36099535 | 539 | |
36099535 | 540 | def on_home(self): |
5964572b | 541 | self._exit_cb() |
36099535 | 542 | |
2aaa10d3 | 543 | def _set_instructions(self): |
9830561d FC |
544 | self._paused = True |
545 | self.__store_state() | |
2aaa10d3 FC |
546 | mgr = TextPropertiesManager.get_global_ptr() |
547 | for name in ['mouse_l', 'mouse_r']: | |
7e487769 | 548 | graphic = OnscreenImage('assets/images/buttons/%s.dds' % name) |
2aaa10d3 FC |
549 | graphic.set_scale(.5) |
550 | graphic.get_texture().set_minfilter(Texture.FTLinearMipmapLinear) | |
551 | graphic.get_texture().set_anisotropic_degree(2) | |
552 | mgr.set_graphic(name, graphic) | |
553 | graphic.set_z(-.2) | |
554 | graphic.set_transparency(True) | |
555 | graphic.detach_node() | |
556 | frm = DirectFrame(frameColor=(.4, .4, .4, .06), | |
557 | frameSize=(-.6, .6, -.3, .3)) | |
558 | font = base.loader.load_font('assets/fonts/Hanken-Book.ttf') | |
559 | font.clear() | |
560 | font.set_pixels_per_unit(60) | |
561 | font.set_minfilter(Texture.FTLinearMipmapLinear) | |
562 | font.set_outline((0, 0, 0, 1), .8, .2) | |
2aaa10d3 | 563 | self._txt = OnscreenText( |
0eff64a3 FC |
564 | self._instr_txt(), parent=frm, font=font, scale=0.06, |
565 | fg=(.9, .9, .9, 1), align=TextNode.A_left) | |
2aaa10d3 FC |
566 | u_l = self._txt.textNode.get_upper_left_3d() |
567 | l_r = self._txt.textNode.get_lower_right_3d() | |
568 | w, h = l_r[0] - u_l[0], u_l[2] - l_r[2] | |
a0acba9a FC |
569 | btn_scale = .05 |
570 | mar = .06 # margin | |
2aaa10d3 | 571 | z = h / 2 - font.get_line_height() * self._txt['scale'][1] |
a0acba9a | 572 | z += (btn_scale + 2 * mar) / 2 |
2aaa10d3 FC |
573 | self._txt['pos'] = -w / 2, z |
574 | u_l = self._txt.textNode.get_upper_left_3d() | |
575 | l_r = self._txt.textNode.get_lower_right_3d() | |
a0acba9a FC |
576 | c_l_r = l_r[0], l_r[1], l_r[2] - 2 * mar - btn_scale |
577 | fsz = u_l[0] - mar, l_r[0] + mar, c_l_r[2] - mar, u_l[2] + mar | |
2aaa10d3 | 578 | frm['frameSize'] = fsz |
a0acba9a FC |
579 | colors = [ |
580 | (.6, .6, .6, 1), # ready | |
581 | (1, 1, 1, 1), # press | |
582 | (.8, .8, .8, 1), # rollover | |
583 | (.4, .4, .4, .4)] | |
584 | imgs = [self.__load_img_btn('exitRight', col) for col in colors] | |
585 | btn = DirectButton( | |
586 | image=imgs, scale=btn_scale, | |
587 | pos=(l_r[0] - btn_scale, 1, l_r[2] - mar - btn_scale), | |
588 | parent=frm, command=self.__on_close_instructions, extraArgs=[frm], | |
589 | relief=FLAT, frameColor=(.6, .6, .6, .08), | |
590 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
591 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
592 | btn.set_transparency(True) | |
2d1773b1 | 593 | self._pos_mgr.register('close_instructions', LibP3d.wdg_pos(btn)) |
a0acba9a | 594 | |
0a0994e4 | 595 | def _set_win(self): |
3fe18d73 | 596 | self.__persistent.save_scene(self.__class__.__name__, self.version()) |
9914cfc9 FC |
597 | loader.load_sfx('assets/audio/sfx/success.ogg').play() |
598 | self._paused = True | |
599 | self.__store_state() | |
600 | frm = DirectFrame(frameColor=(.4, .4, .4, .06), | |
601 | frameSize=(-.6, .6, -.3, .3)) | |
602 | font = base.loader.load_font('assets/fonts/Hanken-Book.ttf') | |
603 | font.clear() | |
604 | font.set_pixels_per_unit(60) | |
605 | font.set_minfilter(Texture.FTLinearMipmapLinear) | |
606 | font.set_outline((0, 0, 0, 1), .8, .2) | |
607 | self._txt = OnscreenText( | |
608 | _('You win!'), | |
609 | parent=frm, | |
610 | font=font, scale=0.2, | |
611 | fg=(.9, .9, .9, 1)) | |
612 | u_l = self._txt.textNode.get_upper_left_3d() | |
613 | l_r = self._txt.textNode.get_lower_right_3d() | |
c991401b FC |
614 | #w, h = l_r[0] - u_l[0], u_l[2] - l_r[2] |
615 | h = u_l[2] - l_r[2] | |
9914cfc9 FC |
616 | btn_scale = .05 |
617 | mar = .06 # margin | |
618 | z = h / 2 - font.get_line_height() * self._txt['scale'][1] | |
619 | z += (btn_scale + 2 * mar) / 2 | |
620 | self._txt['pos'] = 0, z | |
621 | u_l = self._txt.textNode.get_upper_left_3d() | |
622 | l_r = self._txt.textNode.get_lower_right_3d() | |
623 | c_l_r = l_r[0], l_r[1], l_r[2] - 2 * mar - btn_scale | |
624 | fsz = u_l[0] - mar, l_r[0] + mar, c_l_r[2] - mar, u_l[2] + mar | |
625 | frm['frameSize'] = fsz | |
626 | colors = [ | |
627 | (.6, .6, .6, 1), # ready | |
628 | (1, 1, 1, 1), # press | |
629 | (.8, .8, .8, 1), # rollover | |
630 | (.4, .4, .4, .4)] | |
631 | imgs = [self.__load_img_btn('home', col) for col in colors] | |
632 | btn = DirectButton( | |
633 | image=imgs, scale=btn_scale, | |
634 | pos=(-2.8 * btn_scale, 1, l_r[2] - mar - btn_scale), | |
635 | parent=frm, command=self._on_end_home, extraArgs=[frm], | |
636 | relief=FLAT, frameColor=(.6, .6, .6, .08), | |
637 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
638 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
639 | btn.set_transparency(True) | |
2d1773b1 | 640 | self._pos_mgr.register('home_win', LibP3d.wdg_pos(btn)) |
9914cfc9 FC |
641 | imgs = [self.__load_img_btn('rewind', col) for col in colors] |
642 | btn = DirectButton( | |
643 | image=imgs, scale=btn_scale, | |
644 | pos=(0, 1, l_r[2] - mar - btn_scale), | |
645 | parent=frm, command=self._on_restart, extraArgs=[frm], | |
646 | relief=FLAT, frameColor=(.6, .6, .6, .08), | |
647 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
648 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
2d1773b1 | 649 | self._pos_mgr.register('replay', LibP3d.wdg_pos(btn)) |
9914cfc9 | 650 | btn.set_transparency(True) |
6168d0c2 | 651 | enabled = self._scenes.index(self.__class__) < len(self._scenes) - 1 |
9914cfc9 | 652 | if enabled: |
6168d0c2 | 653 | next_scene = self._scenes[self._scenes.index(self.__class__) + 1] |
9914cfc9 FC |
654 | else: |
655 | next_scene = None | |
656 | imgs = [self.__load_img_btn('right', col) for col in colors] | |
657 | btn = DirectButton( | |
658 | image=imgs, scale=btn_scale, | |
659 | pos=(2.8 * btn_scale, 1, l_r[2] - mar - btn_scale), | |
660 | parent=frm, command=self._on_next_scene, | |
661 | extraArgs=[frm, next_scene], relief=FLAT, | |
662 | frameColor=(.6, .6, .6, .08), | |
663 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
664 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
665 | btn['state'] = NORMAL if enabled else DISABLED | |
2d1773b1 | 666 | self._pos_mgr.register('next', LibP3d.wdg_pos(btn)) |
9914cfc9 FC |
667 | btn.set_transparency(True) |
668 | ||
0a0994e4 FC |
669 | def _set_fail(self): |
670 | loader.load_sfx('assets/audio/sfx/success.ogg').play() | |
671 | self._paused = True | |
672 | self.__store_state() | |
673 | frm = DirectFrame(frameColor=(.4, .4, .4, .06), | |
674 | frameSize=(-.6, .6, -.3, .3)) | |
675 | font = base.loader.load_font('assets/fonts/Hanken-Book.ttf') | |
676 | font.clear() | |
677 | font.set_pixels_per_unit(60) | |
678 | font.set_minfilter(Texture.FTLinearMipmapLinear) | |
679 | font.set_outline((0, 0, 0, 1), .8, .2) | |
680 | self._txt = OnscreenText( | |
681 | _('You have failed!'), | |
682 | parent=frm, | |
683 | font=font, scale=0.2, | |
684 | fg=(.9, .9, .9, 1)) | |
685 | u_l = self._txt.textNode.get_upper_left_3d() | |
686 | l_r = self._txt.textNode.get_lower_right_3d() | |
c991401b FC |
687 | #w, h = l_r[0] - u_l[0], u_l[2] - l_r[2] |
688 | h = u_l[2] - l_r[2] | |
0a0994e4 FC |
689 | btn_scale = .05 |
690 | mar = .06 # margin | |
691 | z = h / 2 - font.get_line_height() * self._txt['scale'][1] | |
692 | z += (btn_scale + 2 * mar) / 2 | |
693 | self._txt['pos'] = 0, z | |
694 | u_l = self._txt.textNode.get_upper_left_3d() | |
695 | l_r = self._txt.textNode.get_lower_right_3d() | |
696 | c_l_r = l_r[0], l_r[1], l_r[2] - 2 * mar - btn_scale | |
697 | fsz = u_l[0] - mar, l_r[0] + mar, c_l_r[2] - mar, u_l[2] + mar | |
698 | frm['frameSize'] = fsz | |
699 | colors = [ | |
700 | (.6, .6, .6, 1), # ready | |
701 | (1, 1, 1, 1), # press | |
702 | (.8, .8, .8, 1), # rollover | |
703 | (.4, .4, .4, .4)] | |
704 | imgs = [self.__load_img_btn('home', col) for col in colors] | |
705 | btn = DirectButton( | |
706 | image=imgs, scale=btn_scale, | |
707 | pos=(-2.8 * btn_scale, 1, l_r[2] - mar - btn_scale), | |
708 | parent=frm, command=self._on_end_home, extraArgs=[frm], | |
709 | relief=FLAT, frameColor=(.6, .6, .6, .08), | |
710 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
711 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
2d1773b1 | 712 | self._pos_mgr.register('home_win', LibP3d.wdg_pos(btn)) |
0a0994e4 FC |
713 | btn.set_transparency(True) |
714 | imgs = [self.__load_img_btn('rewind', col) for col in colors] | |
715 | btn = DirectButton( | |
716 | image=imgs, scale=btn_scale, | |
717 | pos=(0, 1, l_r[2] - mar - btn_scale), | |
718 | parent=frm, command=self._on_restart, extraArgs=[frm], | |
719 | relief=FLAT, frameColor=(.6, .6, .6, .08), | |
720 | rolloverSound=loader.load_sfx('assets/audio/sfx/rollover.ogg'), | |
721 | clickSound=loader.load_sfx('assets/audio/sfx/click.ogg')) | |
2d1773b1 | 722 | self._pos_mgr.register('replay', LibP3d.wdg_pos(btn)) |
0a0994e4 FC |
723 | btn.set_transparency(True) |
724 | ||
9914cfc9 FC |
725 | def _on_restart(self, frm): |
726 | self.__on_close_instructions(frm) | |
727 | self.reset() | |
728 | ||
729 | def _on_end_home(self, frm): | |
730 | self.__on_close_instructions(frm) | |
731 | self.on_home() | |
732 | ||
733 | def _on_next_scene(self, frm, scene): | |
734 | self.__on_close_instructions(frm) | |
735 | self._reload_cb(scene) | |
736 | ||
a0acba9a FC |
737 | def __store_state(self): |
738 | btns = [ | |
739 | self.__home_btn, self.__info_btn, self.__right_btn, | |
f26497a5 FC |
740 | #self.__next_btn, self.__prev_btn, self.__rewind_btn |
741 | ] | |
a0acba9a FC |
742 | self.__btn_state = [btn['state'] for btn in btns] |
743 | for btn in btns: | |
744 | btn['state'] = DISABLED | |
745 | [itm.store_state() for itm in self.items] | |
746 | ||
747 | def __restore_state(self): | |
748 | btns = [ | |
749 | self.__home_btn, self.__info_btn, self.__right_btn, | |
f26497a5 FC |
750 | #self.__next_btn, self.__prev_btn, self.__rewind_btn |
751 | ] | |
a0acba9a FC |
752 | for btn, state in zip(btns, self.__btn_state): |
753 | btn['state'] = state | |
754 | [itm.restore_state() for itm in self.items] | |
755 | self._paused = False | |
756 | ||
757 | def __on_close_instructions(self, frm): | |
758 | frm.remove_node() | |
759 | self.__restore_state() | |
067a36db FC |
760 | |
761 | def _set_test_items(self): | |
762 | def frame_after(task): | |
763 | self._define_test_items() | |
764 | for itm in self._test_items: | |
765 | self._pos_mgr.register(itm.name, P3dGfxMgr.pos2d_p2d(itm)) | |
766 | taskMgr.doMethodLater(.01, frame_after, 'frame after') | |
767 | ||
da03f030 FC |
768 | def _define_test_items(self): |
769 | if not self.__class__.json: | |
770 | with open(self.__class__.filename) as f: | |
771 | self.__class__.json = loads(f.read()) | |
772 | for item in self.__class__.json['test_items']['pixel_space']: | |
773 | self._pos_mgr.register(item['id'], tuple(item['position'])) | |
774 | for item in self.__class__.json['test_items']['world_space']: | |
775 | self._set_test_item(item['id'], tuple(item['position'])) | |
776 | ||
067a36db FC |
777 | def _set_test_item(self, name, pos): |
778 | self._test_items += [NodePath(name)] | |
779 | self._test_items[-1].set_pos(pos[0], 0, pos[1]) |