gloss-mc/ui_elements/image_row.py

271 lines
12 KiB
Python

import pygtk
import gtk
import pygst
import gst
import gobject
import pango
import clutter
import os
from modules.video_player.elements.CoverItem import cover_item
from utils.InputQueue import InputQueue
class ImageRow(clutter.Group):
DIRECTION_LEFT, DIRECTION_RIGHT = range(2)
scaleFactor = 1.4
inactiveOpacity = 150
images_size_percent = 1 #0.90 #This is the percentage of the total group size that the covers will take
fps = 25
frames = 15
def __init__(self, glossMgr, width, height, columns):
clutter.Group.__init__(self)
self.glossMgr = glossMgr
self.stage = glossMgr.stage
self.textureLibrary = []
self.images_group = clutter.Group()
self.images_width = int(width * self.images_size_percent)
self.images_height = int(height * self.images_size_percent)
#Columns must be an odd number
if (columns % 2) == 0: columns += 1
self.num_columns = columns
self.image_size = int(self.images_width / self.num_columns) #A cover will be cover_size * cover_size (X * Y)
self.center = int(columns / 2) + 1
"""
#The viewer actually sits within another group that clips its size
self.images_group_clip = clutter.Group()
self.images_group_clip.add(self.images_group)
#Nasty hack to get around centering problem
pos_x = int(self.image_size/2) + (self.image_size * (self.center-1))
self.images_group.set_position(pos_x, int(self.image_size/2))
#And setup the clip size and position
scale_amount = int(self.image_size * self.scaleFactor - self.image_size)
clip_width = (self.image_size*columns) + scale_amount #Width is the cover size by the number of colums, plus the additional amount required for scaling
clip_height = (height) + scale_amount
self.images_group_clip.set_clip(-(scale_amount/2), -(scale_amount/2), clip_width, clip_height)
"""
pos_x = int(self.image_size/2) + (self.image_size * (self.center-1))
self.images_group.set_position(pos_x, int(self.image_size/2))
#self.current_video_details = video_details_group(self.covers_width)
self.num_images = 0
self.image_gap = 1
#Setup input queue controller
self.input_queue = InputQueue()
self.input_queue.set_action(InputQueue.EAST, self.move_right)
self.input_queue.set_action(InputQueue.WEST, self.move_left)
#Setup the current min and max viewable rows
self.min_visible_columns = 0
self.max_visible_columns = columns
self.currentSelection = 0
#self.add(self.images_group_clip)
self.add(self.images_group)
covers_x = int(width * (1-self.images_size_percent)/2)
covers_y = int(height * (1-self.images_size_percent)/2)
#self.images_group.set_position(covers_x, covers_y)
#self.images_group.set_depth(1) #self.image_size)
self.images_group.show()
#self.images_group_clip.show()
def add_texture_group(self, tempGroup):
tempGroup.set_opacity(self.inactiveOpacity)
#tempGroup.set_position( (self.num_covers * self.image_size), 0)
tempGroup.set_depth(1)
self.textureLibrary.append(tempGroup)
x = self.num_images * self.image_size + ( self.num_images * self.image_gap)
y = 0#(self.cover_gap + self.image_size) * (self.num_covers/self.num_columns)
tempGroup.set_position(x, y)
#If we're past the maximum rows, make the pics invistible
if self.num_images > (self.num_columns-self.center):
tempGroup.set_opacity(0)
tempGroup.hide()
else:
self.images_group.add(tempGroup)
#Set the gravity to the center
gap = (self.image_size - tempGroup.get_width())/2
anchor_x = (self.image_size - self.image_gap)/2
gap = (self.image_size - tempGroup.get_height())/2
anchor_y = (self.image_size - self.image_gap)/2
tempGroup.set_anchor_point(anchor_x, anchor_y)
tempGroup.show()
self.num_images += 1
def select_item(self, incomingItem, outgoingItem):
self.timeline = clutter.Timeline(self.frames, self.fps)
alpha = clutter.Alpha(self.timeline, clutter.smoothstep_inc_func)
self.input_queue.set_timeline(self.timeline)
outgoingTexture = self.textureLibrary[outgoingItem]
incomingTexture = self.textureLibrary[incomingItem]
#Do the stuff for edge cases (ie the textures coming in and going out)
if incomingItem > outgoingItem:
direction = self.DIRECTION_RIGHT
edge_texture_incoming_no = outgoingItem + (self.center)
edge_texture_outgoing_no = outgoingItem - (self.center-1)
else:
direction = self.DIRECTION_LEFT
edge_texture_incoming_no = outgoingItem - (self.center)
edge_texture_outgoing_no = outgoingItem + (self.center-1)
if (edge_texture_incoming_no < len(self.textureLibrary)) and (edge_texture_incoming_no >= 0):
edge_texture_incoming = self.textureLibrary[edge_texture_incoming_no]
edge_texture_incoming.show()
if edge_texture_incoming.get_parent() is None: self.images_group.add(edge_texture_incoming)
self.behaviourEdgeIncomingOpacity = clutter.BehaviourOpacity(opacity_start=0, opacity_end=self.inactiveOpacity, alpha=alpha)
self.behaviourEdgeIncomingOpacity.apply(edge_texture_incoming)
if edge_texture_outgoing_no >= 0:
edge_texture_outgoing = self.textureLibrary[edge_texture_outgoing_no]
self.timeline.connect('completed', self.remove_item, edge_texture_outgoing_no)
self.behaviourEdgeOutgoingOpacity = clutter.BehaviourOpacity(opacity_start=self.inactiveOpacity, opacity_end=0, alpha=alpha)
self.behaviourEdgeOutgoingOpacity.apply(edge_texture_outgoing)
#Now do the stuff for selecting the image in the middle
self.behaviourNew_scale = clutter.BehaviourScale(x_scale_start=1, y_scale_start=1, x_scale_end=self.scaleFactor, y_scale_end=self.scaleFactor, alpha=alpha) #clutter.GRAVITY_CENTER)
self.behaviourNew_z = clutter.BehaviourDepth(depth_start=1, depth_end=2, alpha=alpha)
self.behaviourNew_opacity = clutter.BehaviourOpacity(opacity_start=self.inactiveOpacity, opacity_end=255, alpha=alpha)
self.behaviourOld_scale = clutter.BehaviourScale(x_scale_start=self.scaleFactor, y_scale_start=self.scaleFactor, x_scale_end=1, y_scale_end=1, alpha=alpha)
self.behaviourOld_z = clutter.BehaviourDepth(depth_start=2, depth_end=1, alpha=alpha)
self.behaviourOld_opacity = clutter.BehaviourOpacity(opacity_start=255, opacity_end=self.inactiveOpacity, alpha=alpha)
self.behaviourNew_scale.apply(incomingTexture)
self.behaviourNew_z.apply(incomingTexture)
self.behaviourNew_opacity.apply(incomingTexture)
self.behaviourOld_scale.apply(outgoingTexture)
self.behaviourOld_z.apply(outgoingTexture)
self.behaviourOld_opacity.apply(outgoingTexture)
#And finally, moving the whole thing left/right
(x, y) = self.images_group.get_position()
if direction == self.DIRECTION_RIGHT:
(new_x, new_y) = ( (x-(self.image_size+self.image_gap)), y)
else:
(new_x, new_y) = ( (x+(self.image_size+self.image_gap)), y)
knots = (\
(x, y),\
(new_x, new_y)\
)
self.behaviour_path = clutter.BehaviourPath(alpha, knots)
self.behaviour_path.apply(self.images_group)
self.currentSelection = incomingItem
self.timeline.start()
def select_first(self):
#Sanity check
if len(self.textureLibrary) == 0: return
#Make sure the image group is at the beginning
pos_x = int(self.image_size/2) + (self.image_size * (self.center-1))
self.images_group.set_position(pos_x, int(self.image_size/2))
self.timeline = clutter.Timeline(self.frames, self.fps)
self.input_queue.set_timeline(self.timeline)
incomingItem = 0
incomingTexture = self.textureLibrary[incomingItem]
alpha = clutter.Alpha(self.timeline, clutter.ramp_inc_func)
#self.behaviourNew_scale = clutter.BehaviourScale(scale_start=1, scale_end=self.scaleFactor, alpha=alpha)
self.behaviourNew_scale = clutter.BehaviourScale(x_scale_start=1, y_scale_start=1, x_scale_end=self.scaleFactor, y_scale_end=self.scaleFactor, alpha=alpha)
self.behaviourNew_z = clutter.BehaviourDepth(depth_start=1, depth_end=2, alpha=alpha)
self.behaviourNew_opacity = clutter.BehaviourOpacity(opacity_start=self.inactiveOpacity, opacity_end=255, alpha=alpha)
self.behaviourNew_scale.apply(incomingTexture)
self.behaviourNew_z.apply(incomingTexture)
self.behaviourNew_opacity.apply(incomingTexture)
self.currentSelection = incomingItem
self.timeline.start()
def select_none(self):
if self.currentSelection is None:
return
self.timeline = clutter.Timeline(self.frames, self.fps)
alpha = clutter.Alpha(self.timeline, clutter.smoothstep_inc_func)
self.behaviourOld_scale = clutter.BehaviourScale(x_scale_start=self.scaleFactor, y_scale_start=self.scaleFactor, x_scale_end=1, y_scale_end=1, alpha=alpha)
self.behaviourOld_z = clutter.BehaviourDepth(depth_start=2, depth_end=1, alpha=alpha)
self.behaviourOld_opacity = clutter.BehaviourOpacity(opacity_start=255, opacity_end=self.inactiveOpacity, alpha=alpha)
current_cover = self.textureLibrary[self.currentSelection]
self.behaviourOld_scale.apply(current_cover)
self.behaviourOld_z.apply(current_cover)
self.behaviourOld_opacity.apply(current_cover)
self.timeline.start()
def remove_item(self, timeline = None, itemNo = None):
self.textureLibrary[itemNo].set_opacity(0)
self.textureLibrary[itemNo].hide()
self.images_group.remove(self.textureLibrary[itemNo])
def get_current_texture(self):
return self.textureLibrary[self.currentSelection]
def get_current_item(self):
return self.itemLibrary[(self.currentSelection-len(self.folderLibrary))]
#Was get_item_x()
def get_texture_x(self, itemNo):
return self.textureLibrary[itemNo]
def get_item_library(self):
return self.textureLibrary
def set_details_update(self, on_off, details):
self.update_details = on_off
self.details_group = details
def on_key_press_event(self, event):
self.input_queue.input(event)
return self.timeline
#These are the basic movement functions
def move_left(self):
#Make sure we're not already on the first cover
if not self.currentSelection == 0:
newItem = self.currentSelection - 1
self.move_common(newItem)
def move_right(self):
#This check makes sure that we're not on the last cover already
if not self.currentSelection == (self.num_images-1):
newItem = self.currentSelection + 1
self.move_common(newItem)
def move_common(self, newItem):
#Final sanity check
if (newItem < 0) and (not newItem == None):
newItem = self.currentSelection
#If there is movement, make the scale happen
if not newItem == None:
self.select_item(newItem, self.currentSelection)