# -*- coding: utf-8 -*- # vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4 ############################################################################### # OpenLP - Open Source Lyrics Projection # # --------------------------------------------------------------------------- # # Copyright (c) 2008-2009 Raoul Snyman # # Portions copyright (c) 2008-2009 Martin Thompson, Tim Bentley, Carsten # # Tinggaard, Jon Tibble, Jonathan Corwin, Maikel Stuivenberg, Scott Guerrieri # # --------------------------------------------------------------------------- # # 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; version 2 of the License. # # # # 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., 59 # # Temple Place, Suite 330, Boston, MA 02111-1307 USA # ############################################################################### import logging import os import time import uuid from PyQt4 import QtGui from openlp.core.lib import buildIcon class ServiceItemType(object): """ Defines the type of service item """ Text = 1 Image = 2 Command = 3 class ServiceItem(object): """ The service item is a base class for the plugins to use to interact with the service manager, the slide controller, and the projection screen compositor. """ global log log = logging.getLogger(u'ServiceItem') log.info(u'Service Item created') def __init__(self, plugin=None): """ Set up the service item. ``plugin`` The plugin that this service item belongs to. """ if plugin: self.RenderManager = plugin.render_manager self.name = plugin.name self.title = u'' self.audit = u'' self.items = [] self.iconic_representation = None self.raw_slides = None self.display_frames = [] self.raw_footer = None self.theme = None self.service_item_path = None self.service_item_type = None self.editEnabled = False self.raw_frames = [] self.uuid = unicode(uuid.uuid1()) def addIcon(self, icon): """ Add an icon to the service item. This is used when displaying the service item in the service manager. ``icon`` An instance of QIcon or a string to an icon in the resource or on disk. """ self.icon = icon self.iconic_representation = buildIcon(icon) def render(self): """ The render method is what generates the frames for the screen. """ log.debug(u'Render called') self.display_frames = [] if self.service_item_type == ServiceItemType.Text: log.debug(u'Formatting slides') if self.theme is None: self.RenderManager.set_override_theme(None) else: self.RenderManager.set_override_theme(self.theme) for slide in self.raw_frames: before = time.time() formated = self.RenderManager.format_slide(slide[u'raw_slide']) for format in formated: lines = u'' for line in format: lines += line + u'\n' title = lines.split(u'\n')[0] self.display_frames.append({u'title': title, u'text': lines}) log.info(u'Formatting took %4s' % (time.time() - before)) elif self.service_item_type == ServiceItemType.Command: self.display_frames = self.raw_frames elif self.service_item_type == ServiceItemType.Image: for slide in self.raw_frames: slide[u'image'] = \ self.RenderManager.resize_image(slide[u'image']) else: log.error(u'Invalid value renderer :%s' % self.service_item_type) def render_individual(self, row): """ Takes an array of text and geneates an Image from the theme. It assumes the text will fit on the screen as it has generated by the render method above. """ log.debug(u'render individual') if self.theme is None: self.RenderManager.set_override_theme(None) else: self.RenderManager.set_override_theme(self.theme) format = self.display_frames[row][u'text'].split(u'\n') frame = self.RenderManager.generate_slide(format, self.raw_footer) return frame def add_from_image(self, path, title, image): """ Add an image slide to the service item. ``path`` The directory in which the image file is located. ``title`` A title for the slide in the service item. ``image`` The actual image file name. """ self.service_item_type = ServiceItemType.Image self.service_item_path = path self.raw_frames.append( {u'title': title, u'image': image}) def add_from_text(self, title, raw_slide): """ Add a text slide to the service item. ``frame_title`` The title of the slide in the service item. ``raw_slide`` The raw text of the slide. """ self.service_item_type = ServiceItemType.Text title = title.split(u'\n')[0] self.raw_frames.append( {u'title': title, u'raw_slide': raw_slide}) def add_from_command(self, path, file_name, image): """ Add a slide from a command. ``path`` The title of the slide in the service item. ``file_name`` The title of the slide in the service item. ``immage`` The command of/for the slide. """ self.service_item_type = ServiceItemType.Command self.service_item_path = path self.raw_frames.append( {u'title': file_name, u'image': image}) def get_service_repr(self): """ This method returns some text which can be saved into the service file to represent this item. """ service_header = { u'name': self.name.lower(), u'plugin': self.name, u'theme':self.theme, u'title':self.title, u'icon':self.icon, u'footer':self.raw_footer, u'type':self.service_item_type, u'audit':self.audit } service_data = [] if self.service_item_type == ServiceItemType.Text: for slide in self.raw_frames: service_data.append(slide) elif self.service_item_type == ServiceItemType.Image: for slide in self.raw_frames: service_data.append(slide[u'title']) elif self.service_item_type == ServiceItemType.Command: for slide in self.raw_frames: service_data.append({u'title':slide[u'title'], u'image':slide[u'image']}) return {u'header': service_header, u'data': service_data} def set_from_service(self, serviceitem, path=None): """ This method takes a service item from a saved service file (passed from the ServiceManager) and extracts the data actually required. ``serviceitem`` The item to extract data from. ``path`` Defaults to *None*. Any path data, usually for images. """ header = serviceitem[u'serviceitem'][u'header'] self.title = header[u'title'] self.name = header[u'name'] self.service_item_type = header[u'type'] self.shortname = header[u'plugin'] self.theme = header[u'theme'] self.addIcon(header[u'icon']) self.raw_footer = header[u'footer'] self.audit = header[u'audit'] if self.service_item_type == ServiceItemType.Text: for slide in serviceitem[u'serviceitem'][u'data']: self.raw_frames.append(slide) elif self.service_item_type == ServiceItemType.Image: for text_image in serviceitem[u'serviceitem'][u'data']: filename = os.path.join(path, text_image) real_image = QtGui.QImage(unicode(filename)) self.add_from_image(path, text_image, real_image) elif self.service_item_type == ServiceItemType.Command: for text_image in serviceitem[u'serviceitem'][u'data']: filename = os.path.join(path, text_image[u'title']) self.add_from_command(path, text_image[u'title'], text_image[u'image'] ) def merge(self, other): """ Updates the uuid with the value from the original one The uuid is unique for a give service item but this allows one to replace an original version. """ self.uuid = other.uuid def __eq__(self, other): """ Confirms the service items are for the same instance """ if not other: return False return self.uuid == other.uuid def __ne__(self, other): """ Confirms the service items are not for the same instance """ return self.uuid != other.uuid def isSong(self): return self.name == u'Songs' def isMedia(self): return self.name.lower() == u'media' def isCommand(self): return self.service_item_type == ServiceItemType.Command def isImage(self): return self.service_item_type == ServiceItemType.Image def isText(self): return self.service_item_type == ServiceItemType.Text def getFrames(self): if self.service_item_type == ServiceItemType.Text: return self.display_frames else: return self.raw_frames def get_rendered_frame(self, row): if self.service_item_type == ServiceItemType.Text: return self.render_individual(row) else: return self.raw_frames[row][u'image']