198 lines
8.3 KiB
Python
198 lines
8.3 KiB
Python
import clutter
|
|
import gobject
|
|
import pygtk
|
|
import gtk
|
|
|
|
#########################################################
|
|
# The input queue controls fast user input by queing up
|
|
# signals and processing them one by one once any timelines
|
|
# are complete
|
|
#########################################################
|
|
|
|
class InputQueue(gobject.GObject):
|
|
NORTH, EAST, SOUTH, WEST = range(4)
|
|
|
|
#Setup signals
|
|
__gsignals__ = {
|
|
"queue-flushed": (
|
|
gobject.SIGNAL_RUN_LAST, gobject.TYPE_NONE, []),
|
|
"entering-queue": (
|
|
gobject.SIGNAL_RUN_LAST, gobject.TYPE_NONE, [])
|
|
}
|
|
|
|
accelerating = False
|
|
current_acceleration_factor = 1
|
|
acceleration_factor_base = 10 #Timelines will run at regular speed times this when accelerated
|
|
acceleration_threshold = 3 #The queue size at which accleration kicks in (Make this higher to increase the delay before acceleration takes place)
|
|
acceleration_time = 1000 #Time (in ms) it takes for the queue to accelerate/decelerate from nothing to full
|
|
acceleration_steps = 5 #The number of steps in the acceleration process
|
|
|
|
queue_max_size = 10
|
|
release_timeout = 500 #The time (in ms) that the queue waits for key-events before timing out
|
|
|
|
def __init__(self):
|
|
gobject.GObject.__init__(self)
|
|
|
|
self.queue_north = 0
|
|
self.queue_east = 0
|
|
self.queue_south = 0
|
|
self.queue_west = 0
|
|
|
|
self.action_north = None
|
|
self.action_east = None
|
|
self.action_south = None
|
|
self.action_west = None
|
|
|
|
self.current_acceleration_step = 0
|
|
self.stage = clutter.stage_get_default()
|
|
self.poll_time = None
|
|
self.release_timeout_id = None
|
|
self.timeline = None
|
|
#gtk.settings_get_default().set_long_property("gtk-timeout-repeat", 5000, "gloss")
|
|
|
|
def set_timeline(self, timeline):
|
|
if not self.timeline is None: self.timeline.disconnect(self.flush_id)
|
|
|
|
self.timeline = timeline
|
|
self.base_fps = self.timeline.get_speed()
|
|
if self.accelerating:
|
|
fps = self.timeline.get_speed() * self.current_acceleration_factor
|
|
if fps < 1: fps = self.base_fps
|
|
self.timeline.set_speed(fps)
|
|
self.flush_id = self.timeline.connect('completed', self.flush_queue)
|
|
|
|
def input(self, event):
|
|
self.actor = event.source
|
|
if not self.timeline.is_playing():
|
|
if (event.keyval == clutter.keysyms.Left) and (not self.action_west is None): self.action_west()
|
|
if (event.keyval == clutter.keysyms.Right) and (not self.action_east is None): self.action_east()
|
|
if (event.keyval == clutter.keysyms.Up) and (not self.action_north is None): self.action_north()
|
|
if (event.keyval == clutter.keysyms.Down) and (not self.action_south is None): self.action_south()
|
|
|
|
return True
|
|
|
|
self.emit("entering-queue")
|
|
|
|
#Poll for key release
|
|
if self.accelerating:
|
|
if not self.release_timeout_id is None:
|
|
gobject.source_remove(self.release_timeout_id)
|
|
self.release_timeout_id = gobject.timeout_add(self.release_timeout, self.decelerate)
|
|
|
|
if event.keyval == clutter.keysyms.Left:
|
|
self.queue_west += 1
|
|
return True
|
|
elif event.keyval == clutter.keysyms.Right:
|
|
self.queue_east += 1
|
|
return True
|
|
elif event.keyval == clutter.keysyms.Down:
|
|
self.queue_south += 1
|
|
return True
|
|
elif event.keyval == clutter.keysyms.Up:
|
|
self.queue_north += 1
|
|
return True
|
|
|
|
#If we get to this point, we haven't handled the input, so return False
|
|
return False
|
|
|
|
def set_action(self, direction, function):
|
|
if (direction == self.NORTH): self.action_north = function
|
|
if (direction == self.EAST): self.action_east = function
|
|
if (direction == self.SOUTH): self.action_south = function
|
|
if (direction == self.WEST): self.action_west = function
|
|
|
|
def flush_queue(self, data):
|
|
if self.queue_north > self.queue_max_size: self.queue_north = self.queue_max_size
|
|
if self.queue_east > self.queue_max_size: self.queue_east = self.queue_max_size
|
|
if self.queue_south > self.queue_max_size: self.queue_south = self.queue_max_size
|
|
if self.queue_west > self.queue_max_size: self.queue_west = self.queue_max_size
|
|
|
|
#Consolodate north/south, east/west volumes
|
|
if self.queue_north > self.queue_south:
|
|
self.queue_north = self.queue_north - self.queue_south
|
|
self.queue_south = 0
|
|
elif self.queue_south > self.queue_north:
|
|
self.queue_south = self.queue_south - self.queue_north
|
|
self.queue_north = 0
|
|
if self.queue_east > self.queue_west:
|
|
self.queue_east = self.queue_east - self.queue_west
|
|
self.queue_west = 0
|
|
elif self.queue_west > self.queue_east:
|
|
self.queue_west = self.queue_west - self.queue_east
|
|
self.queue_east = 0
|
|
|
|
if (self.queue_north > 0) or (self.queue_east > 0) or (self.queue_south > 0) or (self.queue_west > 0):
|
|
self.timeline.connect('completed', self.flush_queue)
|
|
|
|
absolute_queue_size = self.queue_north + self.queue_east + self.queue_south + self.queue_west
|
|
if absolute_queue_size > self.acceleration_threshold:
|
|
if not self.accelerating:
|
|
self.accelerate()
|
|
#print "Queue Size: N=%s E=%s S=%s W=%s" % (self.queue_north, self.queue_east, self.queue_south, self.queue_west)
|
|
|
|
if self.queue_north > 0:
|
|
self.queue_north -= 1
|
|
self.action_north()
|
|
if self.queue_east > 0:
|
|
self.queue_east -= 1
|
|
self.action_east()
|
|
if self.queue_south > 0:
|
|
self.queue_south -= 1
|
|
self.action_south()
|
|
if self.queue_west > 0:
|
|
self.queue_west -= 1
|
|
self.action_west()
|
|
|
|
return
|
|
else:
|
|
self.queue_east = 0
|
|
self.queue_south = 0
|
|
self.queue_west = 0
|
|
self.queue_north = 0
|
|
|
|
if not self.accelerating:
|
|
self.emit("queue-flushed")
|
|
self.accelerating = False
|
|
|
|
|
|
def is_in_queue(self):
|
|
if self.accelerating: return True
|
|
|
|
if (self.queue_north > 0) or (self.queue_south > 0) or (self.queue_east > 0) or (self.queue_west > 0):
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
def accelerate(self):
|
|
self.accelerating = True
|
|
if self.current_acceleration_step < self.acceleration_steps:
|
|
self.current_acceleration_step +=1
|
|
#print "accelerating: %s" % str(self.current_acceleration_step)
|
|
self.current_acceleration_factor = self.current_acceleration_step * (self.acceleration_factor_base / self.acceleration_steps)
|
|
|
|
fps = self.base_fps * self.current_acceleration_factor
|
|
if fps < 1: fps = self.base_fps
|
|
self.timeline.set_speed(fps)
|
|
if self.current_acceleration_step == 1: gobject.timeout_add( (self.acceleration_time / self.acceleration_steps), self.accelerate)
|
|
return True
|
|
|
|
#print "Acceleration finished"
|
|
return False
|
|
|
|
def decelerate(self, actor = None, event = None):
|
|
#print "Key released: %s" % str(gtk.gdk.keyval_name(event.keyval))
|
|
if self.current_acceleration_step > 0:
|
|
self.current_acceleration_step -= 1
|
|
#print "decelerating: %s" % str(self.current_acceleration_step)
|
|
self.current_acceleration_factor = self.current_acceleration_step * (self.acceleration_factor_base / self.acceleration_steps)
|
|
|
|
fps = self.base_fps * self.current_acceleration_factor
|
|
if fps < 1: fps = self.base_fps
|
|
self.timeline.set_speed(fps)
|
|
if self.current_acceleration_step == (self.acceleration_steps-1): gobject.timeout_add( (self.acceleration_time / self.acceleration_steps), self.decelerate)
|
|
return True
|
|
|
|
self.accelerating = False
|
|
self.emit("queue-flushed")
|
|
#print "Deceleration finished"
|
|
return False |