diff options
author | Alan Aguiar <alanjas@hotmail.com> | 2013-08-12 22:43:59 (GMT) |
---|---|---|
committer | Alan Aguiar <alanjas@hotmail.com> | 2013-08-12 22:43:59 (GMT) |
commit | ade073bbfbba5e8ffc26a175b6d1a80d40a6ccf0 (patch) | |
tree | b83df4375da367b57d1464024880fcac51c62557 /EdSexual.activity | |
parent | affb27b6ed933ab51cd1ef6e44d767f3cc40c66c (diff) |
fixes for sugar activity
Diffstat (limited to 'EdSexual.activity')
l--------- | EdSexual.activity/Main.py | 1 | ||||
-rwxr-xr-x | EdSexual.activity/activity.py | 28 | ||||
-rw-r--r-- | EdSexual.activity/activity/activity.info | 8 | ||||
-rw-r--r-- | EdSexual.activity/activity/edsexual.svg | 93 | ||||
l--------- | EdSexual.activity/api | 1 | ||||
l--------- | EdSexual.activity/assets | 1 | ||||
l--------- | EdSexual.activity/game | 1 | ||||
-rwxr-xr-x | EdSexual.activity/setup.py | 6 | ||||
-rwxr-xr-x | EdSexual.activity/sugargame/__init__.py | 1 | ||||
-rwxr-xr-x | EdSexual.activity/sugargame/canvas.py | 62 | ||||
-rwxr-xr-x | EdSexual.activity/sugargame/event.py | 243 |
11 files changed, 445 insertions, 0 deletions
diff --git a/EdSexual.activity/Main.py b/EdSexual.activity/Main.py new file mode 120000 index 0000000..0c2c64d --- /dev/null +++ b/EdSexual.activity/Main.py @@ -0,0 +1 @@ +../src/Main.py
\ No newline at end of file diff --git a/EdSexual.activity/activity.py b/EdSexual.activity/activity.py new file mode 100755 index 0000000..2217c52 --- /dev/null +++ b/EdSexual.activity/activity.py @@ -0,0 +1,28 @@ +#! /usr/bin/env python +# -*- coding: utf-8 -*- + +import sugargame +import sugargame.canvas +from sugar.activity import activity + +import Main + +class Activity(activity.Activity): + + def __init__(self, handle): + activity.Activity.__init__(self, handle) + + self.max_participants = 1 + self.actividad = Main.EdSexualGame() + self._pygamecanvas = sugargame.canvas.PygameCanvas(self) + self.set_canvas(self._pygamecanvas) + self._pygamecanvas.grab_focus() + self._pygamecanvas.run_pygame(self.actividad.main) + print 'saliendo' + + def read_file(self, file_path): + pass + + def write_file(self, file_path): + pass + diff --git a/EdSexual.activity/activity/activity.info b/EdSexual.activity/activity/activity.info new file mode 100644 index 0000000..db31fa6 --- /dev/null +++ b/EdSexual.activity/activity/activity.info @@ -0,0 +1,8 @@ +[Activity] +name = Educacion Sexual +activity_version = 1 +bundle_id = org.ceibaljam.edsexual +icon = edsexual +exec = sugar-activity activity.Activity +license = GPLv3 +summary = Game about Sexual Education diff --git a/EdSexual.activity/activity/edsexual.svg b/EdSexual.activity/activity/edsexual.svg new file mode 100644 index 0000000..1bbdbe7 --- /dev/null +++ b/EdSexual.activity/activity/edsexual.svg @@ -0,0 +1,93 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> + +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + height="55" + id="svg2" + inkscape:version="0.48.4 r9939" + sodipodi:docname="edsexual.svg" + version="1.1" + width="55"> + <defs + id="defs4"> + <inkscape:perspective + id="perspective10" + inkscape:persp3d-origin="372.04724 : 350.78739 : 1" + inkscape:vp_x="0 : 526.18109 : 1" + inkscape:vp_y="0 : 1000 : 0" + inkscape:vp_z="744.09448 : 526.18109 : 1" + sodipodi:type="inkscape:persp3d" /> + <inkscape:perspective + id="perspective2824" + inkscape:persp3d-origin="0.5 : 0.33333333 : 1" + inkscape:vp_x="0 : 0.5 : 1" + inkscape:vp_y="0 : 1000 : 0" + inkscape:vp_z="1 : 0.5 : 1" + sodipodi:type="inkscape:persp3d" /> + </defs> + <sodipodi:namedview + bordercolor="#666666" + borderopacity="1.0" + id="base" + inkscape:current-layer="layer1" + inkscape:cx="-5.8526158" + inkscape:cy="23.273047" + inkscape:document-units="px" + inkscape:pageopacity="0.0" + inkscape:pageshadow="2" + inkscape:window-height="744" + inkscape:window-maximized="1" + inkscape:window-width="1366" + inkscape:window-x="0" + inkscape:window-y="24" + inkscape:zoom="7.9195959" + pagecolor="#ffffff" + showgrid="false" + showguides="true" + inkscape:guide-bbox="true" /> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title /> + </cc:Work> + </rdf:RDF> + </metadata> + <g + id="layer1" + inkscape:groupmode="layer" + inkscape:label="Capa 1" + transform="translate(0,-997.36218)"> + <rect + style="fill:#0000ff;fill-rule:evenodd;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" + id="rect2985" + width="44.82143" + height="43.92857" + x="5.2525382" + y="1002.7103" /> + <path + sodipodi:type="spiral" + style="fill:none;stroke:#000000;stroke-width:1px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1" + id="path2989" + sodipodi:cx="25.253813" + sodipodi:cy="24.442886" + sodipodi:expansion="1" + sodipodi:revolution="3" + sodipodi:radius="23.668631" + sodipodi:argument="-17.973497" + sodipodi:t0="0" + d="m 25.253813,24.442886 c 0.759711,0.911655 -0.854077,1.473854 -1.51523,1.26269 -1.791682,-0.57224 -1.872711,-2.941161 -1.01015,-4.293148 1.542919,-2.418388 4.977496,-2.405898 7.071068,-0.757611 3.072402,2.418926 2.957768,7.042495 0.505072,9.848987 -3.269058,3.740613 -9.117807,3.517771 -12.626907,0.252532 -4.415844,-4.10897 -4.082018,-11.198067 7e-6,-15.404826 4.943879,-5.094949 13.281121,-4.648727 18.182746,0.252547 5.776395,5.775981 5.216983,15.365917 -0.505087,20.960665 C 28.748976,43.024079 17.903455,42.350995 11.616748,35.807096 4.4733915,28.371502 5.2604642,16.268443 12.626913,9.2905922 20.890956,1.4624986 34.252939,2.3637737 41.921336,10.553297 c 8.513375,9.091923 7.497745,23.713845 -1.515244,32.072342" + transform="translate(0.50507628,1001.4028)" /> + </g> +</svg> diff --git a/EdSexual.activity/api b/EdSexual.activity/api new file mode 120000 index 0000000..e839c85 --- /dev/null +++ b/EdSexual.activity/api @@ -0,0 +1 @@ +../src/api
\ No newline at end of file diff --git a/EdSexual.activity/assets b/EdSexual.activity/assets new file mode 120000 index 0000000..2121391 --- /dev/null +++ b/EdSexual.activity/assets @@ -0,0 +1 @@ +../src/assets/
\ No newline at end of file diff --git a/EdSexual.activity/game b/EdSexual.activity/game new file mode 120000 index 0000000..71d12c9 --- /dev/null +++ b/EdSexual.activity/game @@ -0,0 +1 @@ +../src/game/
\ No newline at end of file diff --git a/EdSexual.activity/setup.py b/EdSexual.activity/setup.py new file mode 100755 index 0000000..da6bc28 --- /dev/null +++ b/EdSexual.activity/setup.py @@ -0,0 +1,6 @@ +#! /usr/bin/env python +# -*- coding: utf-8 -*- + +from sugar.activity import bundlebuilder +if __name__ == "__main__": + bundlebuilder.start() diff --git a/EdSexual.activity/sugargame/__init__.py b/EdSexual.activity/sugargame/__init__.py new file mode 100755 index 0000000..439eb0c --- /dev/null +++ b/EdSexual.activity/sugargame/__init__.py @@ -0,0 +1 @@ +__version__ = '1.1' diff --git a/EdSexual.activity/sugargame/canvas.py b/EdSexual.activity/sugargame/canvas.py new file mode 100755 index 0000000..980cb73 --- /dev/null +++ b/EdSexual.activity/sugargame/canvas.py @@ -0,0 +1,62 @@ +import os +import gtk +import gobject +import pygame +import event + +CANVAS = None + +class PygameCanvas(gtk.EventBox): + + """ + mainwindow is the activity intself. + """ + def __init__(self, mainwindow, pointer_hint = True): + gtk.EventBox.__init__(self) + + global CANVAS + assert CANVAS == None, "Only one PygameCanvas can be created, ever." + CANVAS = self + + # Initialize Events translator before widget gets "realized". + self.translator = event.Translator(mainwindow, self) + + self._mainwindow = mainwindow + + self.set_flags(gtk.CAN_FOCUS) + + self._socket = gtk.Socket() + self.add(self._socket) + self.show_all() + + def run_pygame(self, main_fn): + # Run the main loop after a short delay. The reason for the delay is that the + # Sugar activity is not properly created until after its constructor returns. + # If the Pygame main loop is called from the activity constructor, the + # constructor never returns and the activity freezes. + gobject.idle_add(self._run_pygame_cb, main_fn) + + def _run_pygame_cb(self, main_fn): + assert pygame.display.get_surface() is None, "PygameCanvas.run_pygame can only be called once." + + # Preinitialize Pygame with the X window ID. + assert pygame.display.get_init() == False, "Pygame must not be initialized before calling PygameCanvas.run_pygame." + os.environ['SDL_WINDOWID'] = str(self._socket.get_id()) + pygame.init() + + # Restore the default cursor. + self._socket.window.set_cursor(None) + + # Initialize the Pygame window. + r = self.get_allocation() + pygame.display.set_mode((r.width, r.height), pygame.RESIZABLE) + + # Hook certain Pygame functions with GTK equivalents. + self.translator.hook_pygame() + + # Run the Pygame main loop. + main_fn() + return False + + def get_pygame_widget(self): + return self._socket diff --git a/EdSexual.activity/sugargame/event.py b/EdSexual.activity/sugargame/event.py new file mode 100755 index 0000000..4cc3be8 --- /dev/null +++ b/EdSexual.activity/sugargame/event.py @@ -0,0 +1,243 @@ +import gtk +import gobject +import pygame +import pygame.event +import logging + +class _MockEvent(object): + def __init__(self, keyval): + self.keyval = keyval + +class Translator(object): + key_trans = { + 'Alt_L': pygame.K_LALT, + 'Alt_R': pygame.K_RALT, + 'Control_L': pygame.K_LCTRL, + 'Control_R': pygame.K_RCTRL, + 'Shift_L': pygame.K_LSHIFT, + 'Shift_R': pygame.K_RSHIFT, + 'Super_L': pygame.K_LSUPER, + 'Super_R': pygame.K_RSUPER, + 'KP_Page_Up' : pygame.K_KP9, + 'KP_Page_Down' : pygame.K_KP3, + 'KP_End' : pygame.K_KP1, + 'KP_Home' : pygame.K_KP7, + 'KP_Up' : pygame.K_KP8, + 'KP_Down' : pygame.K_KP2, + 'KP_Left' : pygame.K_KP4, + 'KP_Right' : pygame.K_KP6, + + } + + mod_map = { + pygame.K_LALT: pygame.KMOD_LALT, + pygame.K_RALT: pygame.KMOD_RALT, + pygame.K_LCTRL: pygame.KMOD_LCTRL, + pygame.K_RCTRL: pygame.KMOD_RCTRL, + pygame.K_LSHIFT: pygame.KMOD_LSHIFT, + pygame.K_RSHIFT: pygame.KMOD_RSHIFT, + } + + def __init__(self, mainwindow, inner_evb): + """Initialise the Translator with the windows to which to listen""" + self._mainwindow = mainwindow + self._inner_evb = inner_evb + + # Enable events + # (add instead of set here because the main window is already realized) + self._mainwindow.add_events( + gtk.gdk.KEY_PRESS_MASK | \ + gtk.gdk.KEY_RELEASE_MASK \ + ) + + self._inner_evb.set_events( + gtk.gdk.POINTER_MOTION_MASK | \ + gtk.gdk.POINTER_MOTION_HINT_MASK | \ + gtk.gdk.BUTTON_MOTION_MASK | \ + gtk.gdk.BUTTON_PRESS_MASK | \ + gtk.gdk.BUTTON_RELEASE_MASK + ) + + self._mainwindow.set_flags(gtk.CAN_FOCUS) + self._inner_evb.set_flags(gtk.CAN_FOCUS) + + # Callback functions to link the event systems + self._mainwindow.connect('unrealize', self._quit_cb) + self._inner_evb.connect('key_press_event', self._keydown_cb) + self._inner_evb.connect('key_release_event', self._keyup_cb) + self._inner_evb.connect('button_press_event', self._mousedown_cb) + self._inner_evb.connect('button_release_event', self._mouseup_cb) + self._inner_evb.connect('motion-notify-event', self._mousemove_cb) + self._inner_evb.connect('expose-event', self._expose_cb) + self._inner_evb.connect('configure-event', self._resize_cb) + + # Internal data + self.__stopped = False + self.__keystate = [0] * 323 + self.__button_state = [0,0,0] + self.__mouse_pos = (0,0) + self.__repeat = (None, None) + self.__held = set() + self.__held_time_left = {} + self.__held_last_time = {} + self.__tick_id = None + + def hook_pygame(self): + pygame.key.get_pressed = self._get_pressed + pygame.key.set_repeat = self._set_repeat + pygame.mouse.get_pressed = self._get_mouse_pressed + pygame.mouse.get_pos = self._get_mouse_pos + + def _expose_cb(self, event, widget): + if pygame.display.get_init(): + pygame.event.post(pygame.event.Event(pygame.VIDEOEXPOSE)) + return True + + def _resize_cb(self, widget, event): + evt = pygame.event.Event(pygame.VIDEORESIZE, + size=(event.width,event.height), width=event.width, height=event.height) + pygame.event.post(evt) + return False # continue processing + + def _quit_cb(self, data=None): + self.__stopped = True + pygame.event.post(pygame.event.Event(pygame.QUIT)) + + def _keydown_cb(self, widget, event): + key = event.keyval + if key in self.__held: + return True + else: + if self.__repeat[0] is not None: + self.__held_last_time[key] = pygame.time.get_ticks() + self.__held_time_left[key] = self.__repeat[0] + self.__held.add(key) + + return self._keyevent(widget, event, pygame.KEYDOWN) + + def _keyup_cb(self, widget, event): + key = event.keyval + if self.__repeat[0] is not None: + if key in self.__held: + # This is possibly false if set_repeat() is called with a key held + del self.__held_time_left[key] + del self.__held_last_time[key] + self.__held.discard(key) + + return self._keyevent(widget, event, pygame.KEYUP) + + def _keymods(self): + mod = 0 + for key_val, mod_val in self.mod_map.iteritems(): + mod |= self.__keystate[key_val] and mod_val + return mod + + def _keyevent(self, widget, event, type): + key = gtk.gdk.keyval_name(event.keyval) + if key is None: + # No idea what this key is. + return False + + keycode = None + if key in self.key_trans: + keycode = self.key_trans[key] + elif hasattr(pygame, 'K_'+key.upper()): + keycode = getattr(pygame, 'K_'+key.upper()) + elif hasattr(pygame, 'K_'+key.lower()): + keycode = getattr(pygame, 'K_'+key.lower()) + elif key == 'XF86Start': + # view source request, specially handled... + self._mainwindow.view_source() + else: + print 'Key %s unrecognized' % key + + if keycode is not None: + if type == pygame.KEYDOWN: + mod = self._keymods() + self.__keystate[keycode] = type == pygame.KEYDOWN + if type == pygame.KEYUP: + mod = self._keymods() + ukey = unichr(gtk.gdk.keyval_to_unicode(event.keyval)) + if ukey == '\000': + ukey = '' + evt = pygame.event.Event(type, key=keycode, unicode=ukey, mod=mod) + self._post(evt) + + return True + + def _get_pressed(self): + return self.__keystate + + def _get_mouse_pressed(self): + return self.__button_state + + def _mousedown_cb(self, widget, event): + self.__button_state[event.button-1] = 1 + return self._mouseevent(widget, event, pygame.MOUSEBUTTONDOWN) + + def _mouseup_cb(self, widget, event): + self.__button_state[event.button-1] = 0 + return self._mouseevent(widget, event, pygame.MOUSEBUTTONUP) + + def _mouseevent(self, widget, event, type): + evt = pygame.event.Event(type, button=event.button, pos=(event.x, event.y)) + self._post(evt) + return True + + def _mousemove_cb(self, widget, event): + # From http://www.learningpython.com/2006/07/25/writing-a-custom-widget-using-pygtk/ + # if this is a hint, then let's get all the necessary + # information, if not it's all we need. + if event.is_hint: + x, y, state = event.window.get_pointer() + else: + x = event.x + y = event.y + state = event.state + + rel = (x - self.__mouse_pos[0], y - self.__mouse_pos[1]) + self.__mouse_pos = (x, y) + + self.__button_state = [ + state & gtk.gdk.BUTTON1_MASK and 1 or 0, + state & gtk.gdk.BUTTON2_MASK and 1 or 0, + state & gtk.gdk.BUTTON3_MASK and 1 or 0, + ] + + evt = pygame.event.Event(pygame.MOUSEMOTION, + pos=self.__mouse_pos, rel=rel, buttons=self.__button_state) + self._post(evt) + return True + + def _tick_cb(self): + cur_time = pygame.time.get_ticks() + for key in self.__held: + delta = cur_time - self.__held_last_time[key] + self.__held_last_time[key] = cur_time + + self.__held_time_left[key] -= delta + if self.__held_time_left[key] <= 0: + self.__held_time_left[key] = self.__repeat[1] + self._keyevent(None, _MockEvent(key), pygame.KEYDOWN) + + return True + + def _set_repeat(self, delay=None, interval=None): + if delay is not None and self.__repeat[0] is None: + self.__tick_id = gobject.timeout_add(10, self._tick_cb) + elif delay is None and self.__repeat[0] is not None: + gobject.source_remove(self.__tick_id) + self.__repeat = (delay, interval) + + def _get_mouse_pos(self): + return self.__mouse_pos + + def _post(self, evt): + try: + pygame.event.post(evt) + except pygame.error, e: + if str(e) == 'Event queue full': + print "Event queue full!" + pass + else: + raise e |