neo1973-germany/epydial/epydial.py

457 lines
13 KiB
Python
Raw Normal View History

#!/usr/bin/env python2.5
# -*- coding: utf-8 -*-
__author__ = "Soeren Apel (abraxa@dar-clan.de), fgau, Thomas Gstaedner (thomas (a) gstaedtner (.) net)"
__version__ = "prototype"
__copyright__ = "Copyright (c) 2008"
__license__ = "GPL3"
WIDTH = 480
HEIGHT = 640
FRAMETIME = 1.0 / 20
IMAGE_CACHE_SIZE = 6
FONT_CACHE_SIZE = 2
FULLSCREEN = True
APP_TITLE = "epydial"
WM_INFO = ("epydial", "epydial")
EDJE_FILE_PATH = "data/themes/blackwhite/"
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))
#!/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 TITLE = "epydial" WM_INFO = ("epydial", "epydial") EDJE_GROUP_NAME = "pyneo/dialer/main" EDJE_FILE_NAME = "data/themes/dialer.edj" 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, main, group): self.main = main if not os.path.exists(EDJE_FILE_NAME): raise IOError("Edje theme file not found: " + EDJE_FILE_NAME) try: edje.Edje.__init__(self, self.main.evas_canvas.evas_obj.evas, file=EDJE_FILE_NAME, group=group) except edje.EdjeLoadError, e: raise SystemExit("Error loading %s: %s" % (f, e)) self.size = self.main.evas_canvas.evas_obj.evas.size class MainScreen(EdjeGroup): def __init__(self, main): EdjeGroup.__init__(self, main, EDJE_GROUP_NAME) self.text = [] dbus_ml = e_dbus.DBusEcoreMainLoop() self.system_bus = SystemBus(mainloop=dbus_ml) ecore.timer_add(5, self.init_dbus) def init_dbus(self): try: self.gsm = object_by_url('dbus:///org/pyneo/GsmDevice') self.wireless = object_by_url('dbus:///org/pyneo/gsmdevice/Network') self.keyring = object_by_url('dbus:///org/pyneo/GsmDevice') except Exception, e: print e #if not self.dbus_timer: # self.dbus_timer = ecore.timer_add(2, self.init_dbus) # We had an error, keep the timer running #return True # No error, all went well #if self.dbus_timer: self.dbus_timer.stop() # D-Bus is ready, let's init GSM self.gsm_on() def gsm_on(self): try: if self.gsm.GetPower('sample', dbus_interface=DIN_POWERED, ): print '---', 'gsm device is already on' else: self.gsm.SetPower('sample', True, dbus_interface=DIN_POWERED, ) print '---', 'switching device on' except Exception, e: print e #if not self.gsm_timer: # self.gsm_timer = ecore.timer_add(5, self.gsm_on) # We had an error, keep the timer running #return True # No error #if self.gsm_timer: self.gsm_timer.stop() # GSM ready, let's ask SIM PIN self.sim_pin() def sim_pin(self): self.res = dedbusmap(self.keyring.GetOpened(dbus_interface=DIN_KEYRING)) if self.res['code'] != 'READY': print '---', 'opening keyring' self.part_text_set("numberdisplay_text", "Enter " + self.res['code']) self.res = dedbusmap(self.keyring.GetOpened(dbus_interface=DIN_KEYRING)) else: print '---', 'already authorized' self.nw_register() def nw_register(self): self.nw_res = dedbusmap(self.wireless.GetStatus(dbus_interface=DIN_WIRELESS)) if not self.nw_res['stat'] in (1, 5, ): print '---', 'registering to gsm network' self.wireless.Register(dbus_interface=DIN_WIRELESS) self.nw_res = dedbusmap(self.wireless.GetStatus(dbus_interface=DIN_WIRELESS)) else: self.part_text_set("numberdisplay_text", "please dial") print '---', 'already registered' @edje.decorators.signal_callback("dialer_send", "*") def on_edje_signal_numberkey_triggered(self, emission, source): if self.res['code'] != 'READY': if source in ("0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "#", "*", ): 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": print '---', 'send pin' self.keyring.Open(''.join(self.text), dbus_interface=DIN_KEYRING, ) self.part_text_set("numberdisplay_text", "register ...") self.nw_register() else: if source in ("0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "#", "*", ): 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": print '---', 'dial number' self.part_text_set("numberdisplay_text", "calling ...") os.system('alsactl -f /usr/share/openmoko/scenarios/gsmhandset.state restore') name = self.wireless.Initiate(''.join(self.text), dbus_interface=DIN_VOICE_CALL_INITIATOR, timeout=200, ) time.sleep(20) call = object_by_url(name) call.Hangup(dbus_interface=DIN_CALL) class TestView(object): def __init__(self): edje.frametime_set(FRAMETIME) self.evas_canvas = EvasCanvas(FULLSCREEN, "x11-16") self.groups = {} self.groups[EDJE_GROUP_NAME] = MainScreen(self) self.evas_canvas.evas_obj.data[EDJE_GROUP_NAME] = self.groups[EDJE_GROUP_NAME] self.groups[EDJE_GROUP_NAME].show() self.groups[EDJE_GROUP_NAME].part_text_set("numberdisplay_text", "wait ...") 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 = 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__": TestView() 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" edje_cc -v -id ../images -fd ../fonts *edc ''' git-svn-id: http://www.neo1973-germany.de/svn@102 46df4e5c-bc4e-4628-a0fc-830ba316316d
2008-08-29 15:36:52 +00:00
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("gsm_number_display", self.on_gsm_number_display)
def on_gsm_number_display(self, number):
self.part_text_set("incall_number_text", number)
@edje.decorators.signal_callback("dialer_incall_send", "*")
def on_edje_signal_dialer_incall_triggered(self, emission, source):
if source == "Hangup Call":
print source
PyneoController.gsm_hangup()
if source == "Accept Call":
print source
class MainScreen(EdjeGroup):
text = None
def __init__(self, screen_manager):
EdjeGroup.__init__(self, screen_manager, MAIN_SCREEN_NAME)
self.text = []
ecore.timer_add(60.0, self.display_time)
self.display_time()
def register_pyneo_callbacks(self):
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)
PyneoController.register_callback("gsm_operator_change", self.on_gsm_operator_change)
PyneoController.register_callback("gsm_signal_strength_change", self.on_gsm_signal_strength_change)
def on_sim_key_required(self, key_type):
print '---', 'opening keyring'
self.part_text_set("numberdisplay_text", "Enter " + key_type)
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 ...")
def on_gsm_operator_change(self, operator):
self.part_text_set("operater_text", operator)
def on_gsm_signal_strength_change(self, rssi):
self.part_text_set("signalq_text", "%s dBm"%str(rssi))
def display_time(self):
self.part_text_set("time_text", time.strftime("%H:%M", time.localtime()));
return True;
@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_down":
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 == "clear":
self.text = []
print ''.join(self.text)
self.part_text_set("numberdisplay_text", "".join(self.text))
elif source == "dial":
PyneoController.gsm_dial("".join(self.text))
class PyneoController(object):
_dbus_timer = None
_gsm_timer = None
_keyring_timer = None
_callbacks = {}
_calls = {}
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]
@classmethod
def notify_callbacks(class_, event_name, *args):
try:
for cb in class_._callbacks[event_name]:
cb(*args)
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'))
except Exception, e:
print "Pyneo error: " + 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)
# 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 "GSM error: " + 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()
# Now we can request the keyring object for the SIM, it didn't exist before powering up
class_.get_gsm_keyring()
@classmethod
def get_gsm_keyring(class_):
try:
class_.gsm_keyring = object_by_url(class_.gsm_wireless.GetKeyring())
except Exception, e:
print "SIM error: " + str(e)
if not class_._keyring_timer:
class_._keyring_timer = ecore.timer_add(5, class_.get_gsm_keyring)
# We had an error, keep the timer running if we were called by ecore
return True
# No error (anymore)
if class_._keyring_timer: class_._keyring_timer.stop()
class_.gsm_keyring.connect_to_signal("Opened", class_.on_gsm_keyring_status, dbus_interface=DIN_KEYRING)
# 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)
call = object_by_url(name)
# Initialize "active call" counter
class_._calls[call] = 1
class_.notify_callbacks("gsm_dialing")
@classmethod
def gsm_hangup(class_):
# Find call with highest "active call" counter - it'll be the one currently active
call = None
highest = 0
for (call_obj, counter) in class_._calls.items():
if counter > highest:
highest = counter
call = call_obj
# if call:
call.Hangup(dbus_interface=DIN_CALL)
# Remove the call from our list
# class_._calls.__delitem__(call_obj)
@classmethod
def on_gsm_wireless_status(class_, status_map):
status = dedbusmap(status_map)
class_.gsm_net_status = status
print "GSM NET Status: " + str(status)
if status.has_key('stat'):
nw_status = status['stat']
if nw_status == 0:
class_.notify_callbacks("gsm_unregistered")
elif nw_status in (1, 5):
class_.notify_callbacks("gsm_registered")
elif nw_status == 2:
class_.notify_callbacks("gsm_registering")
elif nw_status == 3:
class_.notify_callbacks("gsm_reg_denied")
elif nw_status == 4:
raise NotImplementedError("GSM registration has unknown state")
if status.has_key('phone_activity_status'):
ph_status = status['phone_activity_status']
if ph_status == 0:
class_.notify_callbacks("gsm_phone_call_end")
if ph_status == 3:
class_.notify_callbacks("gsm_phone_ringing")
if ph_status == 4:
class_.notify_callbacks("gsm_phone_call_start")
if status.has_key('rssi'):
class_.notify_callbacks("gsm_signal_strength_change", status['rssi'])
if status.has_key('oper'):
class_.notify_callbacks("gsm_operator_change", status['oper'])
if status.has_key('number'):
class_.notify_callbacks("gsm_number_display", status['number'])
@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", status["code"])
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()
# Register our own callbacks
PyneoController.register_callback("gsm_phone_ringing", self.on_ringing)
PyneoController.register_callback("gsm_phone_call_start", self.on_call_start)
PyneoController.register_callback("gsm_phone_call_end", self.on_call_end)
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
def on_ringing(self):
self.show_screen(INCALL_SCREEN_NAME)
def on_call_start(self):
self.show_screen(INCALL_SCREEN_NAME)
def on_call_end(self):
self.show_screen(MAIN_SCREEN_NAME)
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.evas.image_cache_set(IMAGE_CACHE_SIZE*1024*1024)
self.evas_obj.evas.font_cache_set(FONT_CACHE_SIZE*1024*1024)
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"
'''