#!/usr/bin/env python2.5 # -*- coding: utf-8 -*- __author__ = "" __version__ = "prototype" __copyright__ = "Copyright (c) 2008" __license__ = "GPL3" WIDTH = 480 HEIGHT = 640 FRAMETIME = 1.0 / 20 FULLSCREEN = True APP_TITLE = "epydial" WM_INFO = ("epydial", "epydial") EDJE_FILE_PATH = "data/themes/default/" MAIN_SCREEN_NAME = "pyneo/dialer/main" INCALL_SCREEN_NAME = "pyneo/dialer/incall" from datetime import datetime from dbus import SystemBus import os import sys import time import e_dbus import ecore import ecore.evas import edje.decorators import edje import evas.decorators from pyneo.dbus_support import * from pyneo.sys_support import pr_set_name from ConfigParser import SafeConfigParser class EdjeGroup(edje.Edje): def __init__(self, group_manager, group): # Theme file name is formed as follows: # Last two group name parts, combined by underscore # pyneo/dialer/main -> dialer_main.edj group_parts = group.split("/") file_name = EDJE_FILE_PATH + group_parts[-2] + "_" + group_parts[-1] + ".edj" if not os.path.exists(file_name): raise IOError("Edje theme file for group %s not found: %s" % (group, file_name)) try: edje.Edje.__init__(self, group_manager.get_evas(), file=file_name, group=group) except edje.EdjeLoadError, e: raise SystemExit("Error loading %s: %s" % (file_name, e)) self.size = group_manager.get_evas().size class InCallScreen(EdjeGroup): def __init__(self, screen_manager): EdjeGroup.__init__(self, screen_manager, INCALL_SCREEN_NAME) def register_pyneo_callbacks(self): PyneoController.register_callback("init", self.on_dbus_initialized) def on_dbus_initialized(self): print "Dbus is ready, says InCallScreen" class MainScreen(EdjeGroup): text = None def __init__(self, screen_manager): EdjeGroup.__init__(self, screen_manager, MAIN_SCREEN_NAME) self.text = [] def register_pyneo_callbacks(self): PyneoController.register_callback("init", self.on_dbus_initialized) PyneoController.register_callback("sim_key_required", self.on_sim_key_required) PyneoController.register_callback("sim_ready", self.on_sim_ready) PyneoController.register_callback("gsm_registering", self.on_gsm_registering) PyneoController.register_callback("gsm_registered", self.on_gsm_registered) PyneoController.register_callback("gsm_dialing", self.on_gsm_dialing) def on_dbus_initialized(self): print "Dbus is ready, says MainScreen" def on_sim_key_required(self): print '---', 'opening keyring' self.part_text_set("numberdisplay_text", "Enter " + PyneoController.gsm_keyring_status['code']) def on_sim_ready(self): print '---', 'SIM unlocked' self.part_text_set("numberdisplay_text", "SIM unlocked") self.text = [] def on_gsm_registering(self): self.part_text_set("numberdisplay_text", "Registering ...") def on_gsm_registered(self): self.part_text_set("numberdisplay_text", "Dial when ready") def on_gsm_dialing(self): print '---', 'dial number' self.part_text_set("numberdisplay_text", "Dialing ...") @edje.decorators.signal_callback("dialer_send", "*") def on_edje_signal_numberkey_triggered(self, emission, source): if PyneoController.gsm_sim_locked(): if len(self.text) < 4 and source.isdigit(): self.text.append(source) print ''.join(self.text) self.part_text_set("numberdisplay_text", '*' * len(self.text)) elif source == "backspace": self.text = self.text[:-1] print ''.join(self.text) self.part_text_set("numberdisplay_text", '*' * len(self.text)) elif source == "dial": print '---', 'send pin' self.part_text_set("numberdisplay_text", "Verifying ...") PyneoController.gsm_unlock_sim(''.join(self.text)) else: if source.isdigit() or source in ('*', '#'): self.text.append(source) print ''.join(self.text) self.part_text_set("numberdisplay_text", "".join(self.text)) elif source == "backspace": self.text = self.text[:-1] print ''.join(self.text) self.part_text_set("numberdisplay_text", "".join(self.text)) elif source == "dial": PyneoController.gsm_dial(self.text) class PyneoController(object): _dbus_timer = None _gsm_timer = None _callbacks = None gsm = None gsm_wireless = None gsm_keyring = None gsm_wireless_status = None gsm_keyring_status = None @classmethod def register_callback(class_, event_name, callback): print "In register_callback: ", event_name try: class_._callbacks[event_name].append(callback) except KeyError: # _callbacks[callback_name] undefined class_._callbacks[event_name] = [callback] except TypeError: # _callbacks undefined and thus can't be subscripted class_._callbacks = {event_name: [callback]} @classmethod def notify_callbacks(class_, event_name): try: for cb in class_._callbacks[event_name]: cb() except TypeError: # Is raised when _callbacks is none and thus can't be subscripted pass except KeyError: pass @classmethod def init(class_): try: class_.gsm = object_by_url('dbus:///org/pyneo/GsmDevice') class_.gsm_wireless = object_by_url(class_.gsm.GetDevice('wireless')) class_.gsm_keyring = object_by_url(class_.gsm_wireless.GetKeyring()) except Exception, e: print "XXXXXXX1 " + str(e) if not class_._dbus_timer: class_._dbus_timer = ecore.timer_add(5, class_.init) # We had an error, keep the timer running if we were called by ecore return True # No error (anymore) if class_._dbus_timer: class_._dbus_timer.stop() # Register our own D-Bus callbacks class_.gsm_wireless.connect_to_signal("Status", class_.on_gsm_wireless_status, dbus_interface=DIN_WIRELESS) class_.gsm_keyring.connect_to_signal("Opened", class_.on_gsm_keyring_status, dbus_interface=DIN_KEYRING) # Notify all screens that the interfaces are here so that they can connect their signal callbacks class_.notify_callbacks("init") # D-Bus is ready, let's power up GSM class_.power_up_gsm() @classmethod def power_up_gsm(class_): try: if class_.gsm.GetPower(APP_TITLE, dbus_interface=DIN_POWERED): print '---', 'gsm device is already on' else: class_.gsm.SetPower(APP_TITLE, True, dbus_interface=DIN_POWERED) print '---', 'switching device on' except Exception, e: print "XXXXXXX2 " + str(e) if not class_._gsm_timer: class_._gsm_timer = ecore.timer_add(5, class_.power_up_gsm) # We had an error, keep the timer running if we were called by ecore return True # No error (anymore) if class_._gsm_timer: class_._gsm_timer.stop() # Inquire SIM status and act accordingly to the initial state status = class_.gsm_keyring.GetOpened(dbus_interface=DIN_KEYRING) class_.on_gsm_keyring_status(status) @classmethod def gsm_sim_locked(class_): return class_.gsm_keyring_status['code'] != 'READY' @classmethod def gsm_unlock_sim(class_, key): class_.gsm_keyring.Open(key, dbus_interface=DIN_KEYRING) @classmethod def gsm_dial(class_, number): os.system('alsactl -f /usr/share/openmoko/scenarios/gsmhandset.state restore') name = class_.gsm_wireless.Initiate(number, dbus_interface=DIN_VOICE_CALL_INITIATOR, timeout=200) class_.notify_callbacks("gsm_dialing") time.sleep(20) call = object_by_url(name) call.Hangup(dbus_interface=DIN_CALL) @classmethod def on_gsm_wireless_status(class_, status_map): status = dedbusmap(status_map) class_.gsm_net_status = status print "GSM NET Status: " + str(status) nw_status = status["stat"] if nw_status == 0: class_.notify_callbacks("gsm_unregistered") if nw_status in (1, 5): class_.notify_callbacks("gsm_registered") if nw_status == 2: class_.notify_callbacks("gsm_registering") if nw_status == 3: class_.notify_callbacks("gsm_reg_denied") if nw_status == 4: raise NotImplementedError("GSM registration has unknown state") @classmethod def on_gsm_keyring_status(class_, status_map): status = dedbusmap(status_map) class_.gsm_keyring_status = status print "SIM Status: " + str(status) if status["code"] == "READY": class_.notify_callbacks("sim_ready") # Try registering on the network class_.gsm_wireless.Register(dbus_interface=DIN_WIRELESS) else: class_.notify_callbacks("sim_key_required") class Dialer(object): screens = None evas_canvas = None system_bus = None def __init__(self): # Initialize the GUI edje.frametime_set(FRAMETIME) self.evas_canvas = EvasCanvas(FULLSCREEN, "x11-16") self.screens = {} self.init_screen(MAIN_SCREEN_NAME, MainScreen(self)) self.init_screen(INCALL_SCREEN_NAME, InCallScreen(self)) self.screens[MAIN_SCREEN_NAME].part_text_set("numberdisplay_text", "wait ...") self.show_screen(MAIN_SCREEN_NAME) # Initialize the D-Bus interface to pyneo dbus_ml = e_dbus.DBusEcoreMainLoop() self.system_bus = SystemBus(mainloop=dbus_ml) PyneoController.init() def init_screen(self, screen_name, instance): self.screens[screen_name] = instance self.evas_canvas.evas_obj.data[screen_name] = instance # Attempt to register the screen default callbacks try: instance.register_pyneo_callbacks() except AttributeError: pass def show_screen(self, screen_name): for (name, screen) in self.screens.items(): if name == screen_name: screen.show() else: screen.hide() def get_evas(self): return self.evas_canvas.evas_obj.evas class EvasCanvas(object): def __init__(self, fullscreen, engine_name): if engine_name == "x11": engine = ecore.evas.SoftwareX11 # elif engine_name == "x11-16": # if ecore.evas.engine_type_supported_get("software_x11_16"): # engine = ecore.evas.SoftwareX11_16 else: print "warning: x11-16 is not supported, falling back to x11" engine = ecore.evas.SoftwareX11 self.evas_obj = engine(w=WIDTH, h=HEIGHT) self.evas_obj.callback_delete_request = self.on_delete_request self.evas_obj.callback_resize = self.on_resize self.evas_obj.title = APP_TITLE self.evas_obj.name_class = WM_INFO self.evas_obj.fullscreen = fullscreen # self.evas_obj.size = str(WIDTH) + 'x' + str(HEIGHT) self.evas_obj.show() def on_resize(self, evas_obj): x, y, w, h = evas_obj.evas.viewport size = (w, h) for key in evas_obj.data.keys(): evas_obj.data[key].size = size def on_delete_request(self, evas_obj): ecore.main_loop_quit() if __name__ == "__main__": Dialer() ecore.main_loop_begin() ''' export LDFLAGS="$LDFLAGS -L/opt/e17/lib" export PKG_CONFIG_PATH="/opt/e17/lib/pkgconfig:$PKG_CONFIG_PATH" export PATH="$PATH:/opt/e17/bin" export PYTHONPATH="/home/fgau/usr/lib/python2.5/site-packages" '''