Web   ·   Wiki   ·   Activities   ·   Blog   ·   Lists   ·   Chat   ·   Meeting   ·   Bugs   ·   Git   ·   Translate   ·   Archive   ·   People   ·   Donate
summaryrefslogtreecommitdiffstats
path: root/src/jarabe/view/keyhandler.py
diff options
context:
space:
mode:
Diffstat (limited to 'src/jarabe/view/keyhandler.py')
-rw-r--r--src/jarabe/view/keyhandler.py289
1 files changed, 289 insertions, 0 deletions
diff --git a/src/jarabe/view/keyhandler.py b/src/jarabe/view/keyhandler.py
new file mode 100644
index 0000000..d6b7565
--- /dev/null
+++ b/src/jarabe/view/keyhandler.py
@@ -0,0 +1,289 @@
+# Copyright (C) 2006-2007, Red Hat, Inc.
+#
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+
+import os
+import signal
+import logging
+import subprocess
+import errno
+
+import dbus
+import gtk
+
+from sugar._sugarext import KeyGrabber
+
+from jarabe.hardware import hardwaremanager
+import view.Shell
+from jarabe.view.tabbinghandler import TabbingHandler
+from jarabe.model.shellmodel import ShellModel
+
+_BRIGHTNESS_STEP = 2
+_VOLUME_STEP = hardwaremanager.VOL_CHANGE_INCREMENT_RECOMMENDATION
+_BRIGHTNESS_MAX = 15
+_VOLUME_MAX = 100
+_TABBING_MODIFIER = gtk.gdk.MOD1_MASK
+
+_actions_table = {
+ 'F1' : 'zoom_mesh',
+ 'F2' : 'zoom_friends',
+ 'F3' : 'zoom_home',
+ 'F4' : 'zoom_activity',
+ 'F9' : 'brightness_down',
+ 'F10' : 'brightness_up',
+ '<alt>F9' : 'brightness_min',
+ '<alt>F10' : 'brightness_max',
+ 'F11' : 'volume_down',
+ 'F12' : 'volume_up',
+ '<alt>F11' : 'volume_min',
+ '<alt>F12' : 'volume_max',
+ '<alt>1' : 'screenshot',
+ '0x93' : 'frame',
+ '0xEB' : 'rotate',
+ '<alt>Tab' : 'next_window',
+ '<alt><shift>Tab': 'previous_window',
+ '<alt>Escape' : 'close_window',
+ '0xDC' : 'open_search',
+# the following are intended for emulator users
+ '<alt><shift>f' : 'frame',
+ '<alt><shift>q' : 'quit_emulator',
+ '<alt><shift>o' : 'open_search',
+ '<alt><shift>r' : 'rotate',
+ '<alt><shift>s' : 'say_text',
+}
+
+J_DBUS_SERVICE = 'org.laptop.Journal'
+J_DBUS_PATH = '/org/laptop/Journal'
+J_DBUS_INTERFACE = 'org.laptop.Journal'
+
+SPEECH_DBUS_SERVICE = 'org.laptop.Speech'
+SPEECH_DBUS_PATH = '/org/laptop/Speech'
+SPEECH_DBUS_INTERFACE = 'org.laptop.Speech'
+
+class KeyHandler(object):
+ def __init__(self):
+ self._screen_rotation = 0
+ self._key_pressed = None
+ self._keycode_pressed = 0
+ self._keystate_pressed = 0
+ self._speech_proxy = None
+
+ self._key_grabber = KeyGrabber()
+ self._key_grabber.connect('key-pressed',
+ self._key_pressed_cb)
+ self._key_grabber.connect('key-released',
+ self._key_released_cb)
+
+ self._tabbing_handler = TabbingHandler(_TABBING_MODIFIER)
+
+ self._key_grabber.grab_keys(_actions_table.keys())
+
+ def _change_volume(self, step=None, value=None):
+ hw_manager = hardwaremanager.get_manager()
+
+ if step is not None:
+ volume = hw_manager.get_volume() + step
+ elif value is not None:
+ volume = value
+
+ volume = min(max(0, volume), _VOLUME_MAX)
+
+ hw_manager.set_volume(volume)
+ hw_manager.set_muted(volume == 0)
+
+ def _change_brightness(self, step=None, value=None):
+ hw_manager = hardwaremanager.get_manager()
+
+ if step is not None:
+ level = hw_manager.get_display_brightness() + step
+ elif value is not None:
+ level = value
+
+ level = min(max(0, level), _BRIGHTNESS_MAX)
+
+ hw_manager.set_display_brightness(level)
+ if level == 0:
+ hw_manager.set_display_mode(hardwaremanager.B_AND_W_MODE)
+ else:
+ hw_manager.set_display_mode(hardwaremanager.COLOR_MODE)
+
+ def _get_speech_proxy(self):
+ if self._speech_proxy is None:
+ bus = dbus.SessionBus()
+ speech_obj = bus.get_object(SPEECH_DBUS_SERVICE, SPEECH_DBUS_PATH,
+ follow_name_owner_changes=True)
+ self._speech_proxy = dbus.Interface(speech_obj,
+ SPEECH_DBUS_INTERFACE)
+ return self._speech_proxy
+
+ def _on_speech_err(self, ex):
+ logging.error("An error occurred with the ESpeak service: %r" % (ex, ))
+
+ def _primary_selection_cb(self, clipboard, text, user_data):
+ logging.debug('KeyHandler._primary_selection_cb: %r' % text)
+ if text:
+ self._get_speech_proxy().SayText(text, reply_handler=lambda: None, \
+ error_handler=self._on_speech_err)
+
+ def handle_say_text(self):
+ clipboard = gtk.clipboard_get(selection="PRIMARY")
+ clipboard.request_text(self._primary_selection_cb)
+
+ def handle_previous_window(self):
+ self._tabbing_handler.previous_activity()
+
+ def handle_next_window(self):
+ self._tabbing_handler.next_activity()
+
+ def handle_close_window(self):
+ view.Shell.get_instance().close_current_activity()
+
+ def handle_zoom_mesh(self):
+ view.Shell.get_instance().set_zoom_level(ShellModel.ZOOM_MESH)
+
+ def handle_zoom_friends(self):
+ view.Shell.get_instance().set_zoom_level(ShellModel.ZOOM_FRIENDS)
+
+ def handle_zoom_home(self):
+ view.Shell.get_instance().set_zoom_level(ShellModel.ZOOM_HOME)
+
+ def handle_zoom_activity(self):
+ view.Shell.get_instance().set_zoom_level(ShellModel.ZOOM_ACTIVITY)
+
+ def handle_brightness_max(self):
+ self._change_brightness(value=_BRIGHTNESS_MAX)
+
+ def handle_brightness_min(self):
+ self._change_brightness(value=0)
+
+ def handle_volume_max(self):
+ self._change_volume(value=_VOLUME_MAX)
+
+ def handle_volume_min(self):
+ self._change_volume(value=0)
+
+ def handle_brightness_up(self):
+ self._change_brightness(step=_BRIGHTNESS_STEP)
+
+ def handle_brightness_down(self):
+ self._change_brightness(step=-_BRIGHTNESS_STEP)
+
+ def handle_volume_up(self):
+ self._change_volume(step=_VOLUME_STEP)
+
+ def handle_volume_down(self):
+ self._change_volume(step=-_VOLUME_STEP)
+
+ def handle_screenshot(self):
+ view.Shell.get_instance().take_screenshot()
+
+ def handle_frame(self):
+ view.Shell.get_instance().get_frame().notify_key_press()
+
+
+ def handle_rotate(self):
+ """
+ Handles rotation of the display (using xrandr) and of the d-pad.
+
+ Notes: default mappings for keypad on MP
+ KP_Up 80
+ KP_Right 85
+ KP_Down 88
+ KP_Left 83
+ """
+
+ states = [ 'normal', 'left', 'inverted', 'right']
+ keycodes = (80, 85, 88, 83, 80, 85, 88, 83)
+ keysyms = ("KP_Up", "KP_Right", "KP_Down", "KP_Left")
+
+ self._screen_rotation -= 1
+ self._screen_rotation %= 4
+
+ actual_keycodes = keycodes[self._screen_rotation:self._screen_rotation
+ + 4]
+ # code_pairs now contains a mapping of keycode -> keysym in the current
+ # orientation
+ code_pairs = zip(actual_keycodes, keysyms)
+
+ # Using the mappings in code_pairs, we dynamically build up an xmodmap
+ # command to rotate the dpad keys.
+ argv = ['xmodmap']
+ for arg in [('-e', 'keycode %i = %s' % p) for p in code_pairs]:
+ argv.extend(arg)
+
+ # If either the xmodmap or xrandr command fails, check_call will fail
+ # with CalledProcessError, which we raise.
+ try:
+ subprocess.check_call(argv)
+ subprocess.check_call(['xrandr', '-o',
+ states[self._screen_rotation]])
+ except OSError, e:
+ if e.errno != errno.EINTR:
+ raise
+
+
+ def handle_quit_emulator(self):
+ if os.environ.has_key('SUGAR_EMULATOR_PID'):
+ pid = int(os.environ['SUGAR_EMULATOR_PID'])
+ os.kill(pid, signal.SIGTERM)
+
+ def focus_journal_search(self):
+ bus = dbus.SessionBus()
+ obj = bus.get_object(J_DBUS_SERVICE, J_DBUS_PATH)
+ journal = dbus.Interface(obj, J_DBUS_INTERFACE)
+ journal.FocusSearch()
+
+ def handle_open_search(self):
+ self.focus_journal_search()
+
+ def _key_pressed_cb(self, grabber, keycode, state):
+ key = grabber.get_key(keycode, state)
+ logging.debug('_key_pressed_cb: %i %i %s' % (keycode, state, key))
+ if key:
+ self._key_pressed = key
+ self._keycode_pressed = keycode
+ self._keystate_pressed = state
+
+ action = _actions_table[key]
+ if self._tabbing_handler.is_tabbing():
+ # Only accept window tabbing events, everything else
+ # cancels the tabbing operation.
+ if not action in ["next_window", "previous_window"]:
+ self._tabbing_handler.stop()
+ return True
+
+ method = getattr(self, 'handle_' + action)
+ method()
+
+ return True
+ else:
+ # If this is not a registered key, then cancel tabbing.
+ if self._tabbing_handler.is_tabbing():
+ if not grabber.is_modifier(keycode):
+ self._tabbing_handler.stop()
+ return True
+
+ return False
+
+ def _key_released_cb(self, grabber, keycode, state):
+ if self._tabbing_handler.is_tabbing():
+ # We stop tabbing and switch to the new window as soon as the
+ # modifier key is raised again.
+ if grabber.is_modifier(keycode, mask=_TABBING_MODIFIER):
+ self._tabbing_handler.stop()
+
+ return True
+ return False
+