From b63283a213f7cbf279d03e3d88a962dd1ab9793e Mon Sep 17 00:00:00 2001
From: James Meyer <james.meyer@operamail.com>
Date: Wed, 7 Dec 2011 13:41:34 -0600
Subject: powermate_myth: add ability to control the FE using a geffen
 powermate.

---
 abs/core/powermate_myth/PKGBUILD          |  16 +++
 abs/core/powermate_myth/powermate-myth.py | 214 ++++++++++++++++++++++++++++++
 abs/core/powermate_myth/powermate.py      | 106 +++++++++++++++
 3 files changed, 336 insertions(+)
 create mode 100644 abs/core/powermate_myth/PKGBUILD
 create mode 100644 abs/core/powermate_myth/powermate-myth.py
 create mode 100644 abs/core/powermate_myth/powermate.py

diff --git a/abs/core/powermate_myth/PKGBUILD b/abs/core/powermate_myth/PKGBUILD
new file mode 100644
index 0000000..1581477
--- /dev/null
+++ b/abs/core/powermate_myth/PKGBUILD
@@ -0,0 +1,16 @@
+pkgname=powermate_myth
+pkgver=1.0
+pkgrel=1
+arch=('i686')
+pkgdesc="Provides support and sample program for using the powermate with mtythv"
+url=""
+depends=(python2 xdotool)
+source=(powermate-myth.py  powermate.py)
+
+build() {
+  cd $startdir/src
+  install -D -m0755 powermate.py  $startdir/pkg/usr/LH/bin/powermate.py
+  install -D -m0755 powermate-myth.py  $startdir/pkg/usr/LH/bin/powermate-myth.py
+}
+md5sums=('44a03725355ed53d9f81099923d94329'
+         '90bed7b80570ad6e63ebff827830e14b')
diff --git a/abs/core/powermate_myth/powermate-myth.py b/abs/core/powermate_myth/powermate-myth.py
new file mode 100644
index 0000000..85e5165
--- /dev/null
+++ b/abs/core/powermate_myth/powermate-myth.py
@@ -0,0 +1,214 @@
+#!/usr/bin/python2
+
+import powermate
+import time
+import os
+
+EVENT_BUTTON_PRESS = 1
+EVENT_RELATIVE_MOTION = 2
+
+
+button_pressed = False
+single_tap = False
+mode = "nav"
+
+current_event=(0,0,time.time())
+last_event=current_event
+skipped_event = 0
+pressed_time = 0
+button_held = False
+led_brightness = 255
+#pm = powermate.PowerMate("/dev/powermate")
+
+pm = powermate.PowerMate()
+pm.SetLEDState(led_brightness,0,0,0,0)
+
+
+def runcmd(cmd):
+#    print cmd
+    os.system(cmd)
+
+
+def volume_action(action):
+    global mode
+    global led_brightness
+    if action == "L" :
+        print "Volume down"
+
+    if action == "R" :
+        print "Volume UP"
+
+    if action == "tap" :
+        print "toggle mute"
+
+    if action == "double_tap" :
+        mode = "nav"
+        led_brightness = 500
+        print "mode is now nav"
+
+    return
+
+
+def nav_action(action):
+    global mode
+    global led_brightness
+    keycmd = '''xdotool search --name "Mythtv Frontend" key %s'''
+    if action == "L" :
+        #print "prev"
+       cmd =  keycmd %"Up"
+       runcmd(cmd)
+
+    if action == "R" :
+        #print "next"
+        cmd =  keycmd %"Down"
+        runcmd(cmd)
+
+    if action == "LP" :
+        #print "back"
+        cmd =  keycmd %"Left"
+        runcmd(cmd)
+
+    if action == "RP" :
+        #print "tab"
+        cmd =  keycmd %"Right"
+        runcmd(cmd)
+
+
+    if action == "tap" :
+        #print "select"
+        cmd =  keycmd %"Return"
+        runcmd(cmd)
+
+    if action == "double_tap" :
+        mode = "volume"
+        led_brightness = 10
+        print "mode is now vol"
+
+    if action == "button_held" :
+        cmd =  keycmd %"Escape"
+        runcmd(cmd)
+
+
+
+    return
+
+
+
+
+def act_on_event(action):
+
+    if mode == "volume":
+        volume_action(action)
+
+    elif mode == "nav":
+        nav_action(action)
+
+
+
+
+event_stack=[current_event]
+while 1:
+        #process_events = True
+        reset_lastevent = False
+	event = pm.WaitForEvent(2)
+       # print event
+        if event:
+            #event that happens after every button push, can be thrown away.
+            if event[2] == 0:
+                continue
+
+
+        elif single_tap:
+            pressed_diff = time.time() - pressed_time
+            #This takes care of the holding down the button for X amount of time
+            #print event,  button_pressed, pressed_diff, current_event
+            #if pressed_diff >= 2 and button_pressed and current_event[0] == EVENT_BUTTON_PRESS:
+            if pressed_diff >= 2 and button_pressed :
+                act_on_event("button_held")
+                single_tap = False
+                pressed_time = 0
+                button_pressed = False
+                last_event_time = time.time()
+                last_event=(0,0,time.time())
+                pm.SetLEDState(led_brightness,0,0,0,0)
+                #print "resetting last event:",  last_event
+
+
+           #This handles a single press of the button.
+            elif not button_pressed :
+                act_on_event("tap")
+                single_tap = False
+                last_event_time = time.time()
+                last_event=(0,0,time.time())
+
+        if event:
+            #print "processing:" , event
+            current_event = event[2],event[4],time.time()
+
+            last_event_time = last_event[2]
+            current_event_time = current_event[2]
+
+            last_event_only = last_event[0], last_event[1]
+            current_event_only = current_event[0], current_event[1]
+
+            repeat_rate = 10
+            repeat_time = .50
+
+            if last_event_only == current_event_only and  current_event[0] == EVENT_RELATIVE_MOTION :
+                #check for time diff
+                event_time_diff = current_event_time - last_event_time
+
+                if  skipped_event <= repeat_rate and event_time_diff <= repeat_time:
+                    skipped_event = skipped_event + 1
+                    #print "skipped a repeat event : %s" %skipped_event
+                    continue
+            #event occured, reset skip counter
+            skipped_event = 0
+
+            #rotate dial
+            if current_event[0] == EVENT_RELATIVE_MOTION:
+                reset_lastevent = True
+                single_tap = False
+                if current_event[1] >= 0 :
+                    if button_pressed :
+                        #print "dial turned Counter clockwise and button is pressed"
+                        act_on_event("RP")
+                    else:
+                        #print "dial turned Counter clockwise"
+                        act_on_event("R")
+
+                if current_event[1]  <=0 :
+                    if button_pressed :
+                        #print "dial turned clockwise and button is pressed"
+                        act_on_event("LP")
+                    else:
+                        #print "dial turned clockwise"
+                        act_on_event("L")
+
+
+            #button was pushed
+            if current_event[0] == EVENT_BUTTON_PRESS :
+                pm.SetLEDState(0,0,0,0,0)
+                if current_event[1] == 1:
+                    button_pressed = True
+                    pressed_time = current_event_time
+                    tap_diff = current_event_time - last_event_time
+                    #double tap
+                    if tap_diff < 0.2 and last_event[0] == EVENT_BUTTON_PRESS:
+                        act_on_event("double_tap")
+                        single_tap = False
+                    else:
+                        single_tap = True
+                        reset_lastevent = True
+
+                #button was released
+                elif current_event[1] == 0:
+                    button_pressed = False
+                    pressed_time  = 0
+                    pm.SetLEDState(led_brightness,0,0,0,0)
+
+        if reset_lastevent:
+            last_event = current_event
+
+
+
diff --git a/abs/core/powermate_myth/powermate.py b/abs/core/powermate_myth/powermate.py
new file mode 100644
index 0000000..8e40078
--- /dev/null
+++ b/abs/core/powermate_myth/powermate.py
@@ -0,0 +1,106 @@
+#!/usr/bin/env python
+
+import select
+import os
+import fcntl
+import struct
+import exceptions
+
+#struct input_event {
+#        struct timeval time; = {long seconds, long microseconds}
+#        unsigned short type;
+#        unsigned short code;
+#        unsigned int value;
+#};
+
+input_event_struct = "@llHHi"
+input_event_size = struct.calcsize(input_event_struct)
+
+EVENT_BUTTON_PRESS = 1
+EVENT_RELATIVE_MOTION = 2
+RELATIVE_AXES_DIAL = 7
+BUTTON_MISC = 0x100
+
+def report(x):
+    sys.stderr.write(x + "\n")
+
+class PowerMate:
+    def __init__(self, filename = None):
+        self.handle = -1
+        if filename:
+            if not self.OpenDevice(filename):
+                raise exceptions.RuntimeError, 'Unable to find powermate'
+        else:
+            ok = 0
+            for d in range(0, 16):
+                if self.OpenDevice("/dev/input/event%d" % d):
+                    ok = 1
+                    break
+            if not ok:
+                raise exceptions.RuntimeError, 'Unable to find powermate'
+        self.poll = select.poll()
+        self.poll.register(self.handle, select.POLLIN)
+        self.event_queue = [] # queue used to reduce kernel/userspace context switching
+
+    def __del__(self):
+        if self.handle >= 0:
+            self.poll.unregister(self.handle)
+            os.close(self.handle)
+            self.handle = -1
+            del self.poll
+
+    def OpenDevice(self, filename):
+        try:
+            self.handle = os.open(filename, os.O_RDWR)
+            if self.handle < 0:
+                return 0
+            name = fcntl.ioctl(self.handle, 0x80ff4506, chr(0) * 256) # read device name
+            name = name.replace(chr(0), '')
+            if name == 'Griffin PowerMate' or name == 'Griffin SoundKnob':
+                fcntl.fcntl(self.handle, fcntl.F_SETFL, os.O_NDELAY)
+                return 1
+            os.close(self.handle)
+            self.handle = -1
+            return 0
+        except exceptions.OSError:
+            return 0
+
+    def WaitForEvent(self, timeout): # timeout in seconds
+        if len(self.event_queue) > 0:
+            return self.event_queue.pop(0)
+        if self.handle < 0:
+            return None
+        r = self.poll.poll(int(timeout*100))
+        if len(r) == 0:
+            return None
+        return self.GetEvent()
+
+    def GetEvent(self): # only call when descriptor is readable
+        if self.handle < 0:
+            return None
+        try:
+            data = os.read(self.handle, input_event_size * 32)
+            while data != '':
+                self.event_queue.append(struct.unpack(input_event_struct, data[0:input_event_size]))
+                data = data[input_event_size:]
+            return self.event_queue.pop(0)
+        except exceptions.OSError, e: # Errno 11: Resource temporarily unavailable
+            #if e.errno == 19: # device has been disconnected
+            #    report("PowerMate disconnected! Urgent!");
+            return None
+
+    def SetLEDState(self, static_brightness, pulse_speed, pulse_table, pulse_on_sleep, pulse_on_wake):
+        static_brightness &= 0xff;
+        if pulse_speed < 0:
+            pulse_speed = 0
+        if pulse_speed > 510:
+            pulse_speed = 510
+        if pulse_table < 0:
+            pulse_table = 0
+        if pulse_table > 2:
+            pulse_table = 2
+        pulse_on_sleep = not not pulse_on_sleep # not not = convert to 0/1
+        pulse_on_wake  = not not pulse_on_wake
+        magic = static_brightness | (pulse_speed << 8) | (pulse_table << 17) | (pulse_on_sleep << 19) | (pulse_on_wake << 20)
+        data = struct.pack(input_event_struct, 0, 0, 0x04, 0x01, magic)
+        os.write(self.handle, data)
-- 
cgit v0.12