2009-04-22 19:46:10 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
|
2009-07-14 13:51:27 +00:00
|
|
|
|
2009-09-08 19:58:05 +00:00
|
|
|
###############################################################################
|
|
|
|
# OpenLP - Open Source Lyrics Projection #
|
|
|
|
# --------------------------------------------------------------------------- #
|
2009-12-31 12:52:01 +00:00
|
|
|
# Copyright (c) 2008-2010 Raoul Snyman #
|
|
|
|
# Portions copyright (c) 2008-2010 Tim Bentley, Jonathan Corwin, Michael #
|
2010-03-21 23:58:01 +00:00
|
|
|
# Gorven, Scott Guerrieri, Christian Richter, Maikel Stuivenberg, Martin #
|
|
|
|
# Thompson, Jon Tibble, Carsten Tinggaard #
|
2009-09-08 19:58:05 +00:00
|
|
|
# --------------------------------------------------------------------------- #
|
|
|
|
# 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 #
|
|
|
|
###############################################################################
|
2009-07-14 13:51:27 +00:00
|
|
|
|
2009-04-22 19:46:10 +00:00
|
|
|
import logging
|
2009-05-11 05:09:43 +00:00
|
|
|
|
2010-01-24 23:16:15 +00:00
|
|
|
from PyQt4 import QtCore
|
2009-04-22 19:46:10 +00:00
|
|
|
|
2010-07-23 05:05:34 +00:00
|
|
|
from openlp.core.lib import Renderer, ThemeLevel, ServiceItem
|
|
|
|
from openlp.core.ui import MainDisplay
|
2010-04-03 07:10:31 +00:00
|
|
|
|
2010-02-27 15:31:23 +00:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2009-07-10 13:16:15 +00:00
|
|
|
class RenderManager(object):
|
2009-04-22 19:46:10 +00:00
|
|
|
"""
|
2009-07-10 13:16:15 +00:00
|
|
|
Class to pull all Renderer interactions into one place. The plugins will
|
|
|
|
call helper methods to do the rendering but this class will provide
|
|
|
|
display defense code.
|
2009-09-04 22:50:19 +00:00
|
|
|
|
|
|
|
``theme_manager``
|
|
|
|
The ThemeManager instance, used to get the current theme details.
|
|
|
|
|
2010-01-16 07:22:50 +00:00
|
|
|
``screens``
|
|
|
|
Contains information about the Screens.
|
2009-09-04 22:50:19 +00:00
|
|
|
|
|
|
|
``screen_number``
|
|
|
|
Defaults to *0*. The index of the output/display screen.
|
2009-04-22 19:46:10 +00:00
|
|
|
"""
|
|
|
|
log.info(u'RenderManager Loaded')
|
|
|
|
|
2010-04-02 18:12:54 +00:00
|
|
|
def __init__(self, theme_manager, screens):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
Initialise the render manager.
|
|
|
|
"""
|
2009-04-22 19:46:10 +00:00
|
|
|
log.debug(u'Initilisation started')
|
2010-01-16 07:22:50 +00:00
|
|
|
self.screens = screens
|
2010-07-23 05:05:34 +00:00
|
|
|
self.display = self.display = MainDisplay(self, screens, False)
|
|
|
|
self.display.setup()
|
2009-04-25 06:11:15 +00:00
|
|
|
self.theme_manager = theme_manager
|
2009-05-16 10:24:03 +00:00
|
|
|
self.renderer = Renderer()
|
2010-01-16 07:22:50 +00:00
|
|
|
self.calculate_default(self.screens.current[u'size'])
|
2009-05-18 19:04:25 +00:00
|
|
|
self.theme = u''
|
2009-05-20 20:17:20 +00:00
|
|
|
self.service_theme = u''
|
2009-11-27 21:37:01 +00:00
|
|
|
self.theme_level = u''
|
2009-09-26 06:46:26 +00:00
|
|
|
self.override_background = None
|
2009-10-24 07:22:44 +00:00
|
|
|
self.themedata = None
|
2009-05-18 19:04:25 +00:00
|
|
|
|
2010-04-02 18:12:54 +00:00
|
|
|
def update_display(self):
|
2009-06-05 18:53:50 +00:00
|
|
|
"""
|
|
|
|
Updates the render manager's information about the current screen.
|
|
|
|
"""
|
2009-06-10 15:54:46 +00:00
|
|
|
log.debug(u'Update Display')
|
2010-01-16 07:22:50 +00:00
|
|
|
self.calculate_default(self.screens.current[u'size'])
|
|
|
|
self.renderer.bg_frame = None
|
2009-06-05 18:53:50 +00:00
|
|
|
|
2009-11-27 21:37:01 +00:00
|
|
|
def set_global_theme(self, global_theme, theme_level=ThemeLevel.Global):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
Set the global-level theme and the theme level.
|
|
|
|
|
|
|
|
``global_theme``
|
|
|
|
The global-level theme to be set.
|
|
|
|
|
2009-11-27 21:37:01 +00:00
|
|
|
``theme_level``
|
|
|
|
Defaults to *``ThemeLevel.Global``*. The theme level, can be
|
|
|
|
``ThemeLevel.Global``, ``ThemeLevel.Service`` or
|
|
|
|
``ThemeLevel.Song``.
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
2009-05-18 19:04:25 +00:00
|
|
|
self.global_theme = global_theme
|
2009-11-27 21:37:01 +00:00
|
|
|
self.theme_level = theme_level
|
2009-05-18 19:04:25 +00:00
|
|
|
|
|
|
|
def set_service_theme(self, service_theme):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
Set the service-level theme.
|
|
|
|
|
|
|
|
``service_theme``
|
|
|
|
The service-level theme to be set.
|
|
|
|
"""
|
2009-05-18 19:04:25 +00:00
|
|
|
self.service_theme = service_theme
|
2009-04-25 06:11:15 +00:00
|
|
|
|
2009-04-29 19:07:13 +00:00
|
|
|
def set_override_theme(self, theme):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
Set the appropriate theme depending on the theme level.
|
2010-07-12 16:49:38 +00:00
|
|
|
Called by the service item when building a display frame
|
2009-07-10 13:16:15 +00:00
|
|
|
|
|
|
|
``theme``
|
2010-07-12 16:49:38 +00:00
|
|
|
The name of the song-level theme. None means the service
|
|
|
|
item wants to use the given value.
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
log.debug(u'set override theme to %s', theme)
|
2009-11-27 21:37:01 +00:00
|
|
|
if self.theme_level == ThemeLevel.Global:
|
2009-05-18 19:04:25 +00:00
|
|
|
self.theme = self.global_theme
|
2009-11-27 21:37:01 +00:00
|
|
|
elif self.theme_level == ThemeLevel.Service:
|
2009-05-18 19:04:25 +00:00
|
|
|
if self.service_theme == u'':
|
|
|
|
self.theme = self.global_theme
|
|
|
|
else:
|
|
|
|
self.theme = self.service_theme
|
2009-04-29 19:07:13 +00:00
|
|
|
else:
|
2009-11-03 18:14:25 +00:00
|
|
|
if theme:
|
2009-05-18 19:04:25 +00:00
|
|
|
self.theme = theme
|
2009-11-27 21:37:01 +00:00
|
|
|
elif self.theme_level == ThemeLevel.Song or \
|
|
|
|
self.theme_level == ThemeLevel.Service:
|
2009-05-18 19:04:25 +00:00
|
|
|
if self.service_theme == u'':
|
|
|
|
self.theme = self.global_theme
|
|
|
|
else:
|
|
|
|
self.theme = self.service_theme
|
2009-05-22 05:14:55 +00:00
|
|
|
else:
|
|
|
|
self.theme = self.global_theme
|
2009-10-24 07:22:44 +00:00
|
|
|
if self.theme != self.renderer.theme_name or self.themedata is None:
|
2009-09-21 17:56:36 +00:00
|
|
|
log.debug(u'theme is now %s', self.theme)
|
2009-05-16 10:24:03 +00:00
|
|
|
self.themedata = self.theme_manager.getThemeData(self.theme)
|
2010-01-16 07:22:50 +00:00
|
|
|
self.calculate_default(self.screens.current[u'size'])
|
2009-05-16 10:24:03 +00:00
|
|
|
self.renderer.set_theme(self.themedata)
|
|
|
|
self.build_text_rectangle(self.themedata)
|
2010-07-14 17:36:48 +00:00
|
|
|
self.renderer.set_frame_dest(self.width, self.height)
|
2010-07-17 08:59:15 +00:00
|
|
|
return self.renderer._rect, self.renderer._rect_footer
|
2009-04-30 21:02:28 +00:00
|
|
|
|
|
|
|
def build_text_rectangle(self, theme):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
2010-01-10 11:49:04 +00:00
|
|
|
Builds a text block using the settings in ``theme``
|
|
|
|
and the size of the display screen.height.
|
2009-07-10 13:16:15 +00:00
|
|
|
|
|
|
|
``theme``
|
|
|
|
The theme to build a text block for.
|
|
|
|
"""
|
|
|
|
log.debug(u'build_text_rectangle')
|
2009-04-30 21:02:28 +00:00
|
|
|
main_rect = None
|
|
|
|
footer_rect = None
|
2009-11-03 15:13:52 +00:00
|
|
|
if not theme.font_main_override:
|
2010-01-10 11:49:04 +00:00
|
|
|
main_rect = QtCore.QRect(10, 0,
|
2010-03-13 20:50:52 +00:00
|
|
|
self.width - 20, self.footer_start)
|
2009-04-30 21:02:28 +00:00
|
|
|
else:
|
2009-11-04 01:16:15 +00:00
|
|
|
main_rect = QtCore.QRect(theme.font_main_x, theme.font_main_y,
|
|
|
|
theme.font_main_width - 1, theme.font_main_height - 1)
|
2009-11-03 15:13:52 +00:00
|
|
|
if not theme.font_footer_override:
|
2010-01-10 11:49:04 +00:00
|
|
|
footer_rect = QtCore.QRect(10, self.footer_start,
|
2010-03-13 20:50:52 +00:00
|
|
|
self.width - 20, self.height - self.footer_start)
|
2009-04-30 21:02:28 +00:00
|
|
|
else:
|
2009-11-04 01:16:15 +00:00
|
|
|
footer_rect = QtCore.QRect(theme.font_footer_x,
|
|
|
|
theme.font_footer_y, theme.font_footer_width - 1,
|
|
|
|
theme.font_footer_height - 1)
|
2009-08-31 06:53:55 +00:00
|
|
|
self.renderer.set_text_rectangle(main_rect, footer_rect)
|
2009-04-22 19:46:10 +00:00
|
|
|
|
2009-04-29 19:07:13 +00:00
|
|
|
def generate_preview(self, themedata):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
Generate a preview of a theme.
|
|
|
|
|
|
|
|
``themedata``
|
|
|
|
The theme to generated a preview for.
|
|
|
|
"""
|
2009-05-11 05:09:43 +00:00
|
|
|
log.debug(u'generate preview')
|
2010-01-10 11:49:04 +00:00
|
|
|
#set the default image size for previews
|
2010-01-16 07:22:50 +00:00
|
|
|
self.calculate_default(self.screens.preview[u'size'])
|
2009-04-29 19:07:13 +00:00
|
|
|
self.renderer.set_theme(themedata)
|
2009-04-30 21:02:28 +00:00
|
|
|
self.build_text_rectangle(themedata)
|
2009-05-16 10:24:03 +00:00
|
|
|
self.renderer.set_frame_dest(self.width, self.height, True)
|
2010-01-09 09:34:06 +00:00
|
|
|
#Reset the real screen size for subsequent render requests
|
2010-01-16 07:22:50 +00:00
|
|
|
self.calculate_default(self.screens.current[u'size'])
|
2009-11-15 07:07:40 +00:00
|
|
|
verse = u'Amazing Grace!\n'\
|
|
|
|
'How sweet the sound\n'\
|
|
|
|
'To save a wretch like me;\n'\
|
|
|
|
'I once was lost but now am found,\n'\
|
|
|
|
'Was blind, but now I see.'
|
2009-07-10 13:16:15 +00:00
|
|
|
footer = []
|
|
|
|
footer.append(u'Amazing Grace (John Newton)' )
|
|
|
|
footer.append(u'Public Domain')
|
2009-11-27 21:37:01 +00:00
|
|
|
footer.append(u'CCLI 123456')
|
2009-11-15 07:07:40 +00:00
|
|
|
formatted = self.renderer.format_slide(verse, False)
|
2010-01-09 09:34:06 +00:00
|
|
|
#Only Render the first slide page returned
|
2010-07-23 05:05:34 +00:00
|
|
|
serviceItem = ServiceItem()
|
|
|
|
serviceItem.add_from_text(u'', verse, u'')
|
|
|
|
serviceItem.render_manager = self
|
|
|
|
serviceItem.render()
|
|
|
|
serviceItem.raw_footer = footer
|
|
|
|
self.display.buildHtml(serviceItem)
|
2010-07-25 07:21:10 +00:00
|
|
|
frame, raw_html = serviceItem.get_rendered_frame(0)
|
|
|
|
frame = self.display.text(raw_html)
|
|
|
|
return frame
|
2009-05-16 10:24:03 +00:00
|
|
|
|
2009-06-01 17:50:37 +00:00
|
|
|
def format_slide(self, words):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
2009-08-31 06:53:55 +00:00
|
|
|
Calculate how much text can fit on a slide.
|
2009-07-10 13:16:15 +00:00
|
|
|
|
|
|
|
``words``
|
|
|
|
The words to go on the slides.
|
|
|
|
"""
|
2009-05-04 13:48:12 +00:00
|
|
|
log.debug(u'format slide')
|
2009-05-17 08:13:22 +00:00
|
|
|
self.build_text_rectangle(self.themedata)
|
2009-05-17 15:24:02 +00:00
|
|
|
return self.renderer.format_slide(words, False)
|
2009-04-25 06:11:15 +00:00
|
|
|
|
2009-07-10 13:16:15 +00:00
|
|
|
def generate_slide(self, main_text, footer_text):
|
|
|
|
"""
|
|
|
|
Generate the actual slide image.
|
|
|
|
|
|
|
|
``main_text``
|
|
|
|
The text for the main area of the slide.
|
|
|
|
|
|
|
|
``footer_text``
|
|
|
|
The text for the slide footer.
|
|
|
|
"""
|
2009-05-04 13:48:12 +00:00
|
|
|
log.debug(u'generate slide')
|
2010-07-12 16:49:38 +00:00
|
|
|
self.build_text_rectangle(self.themedata)
|
|
|
|
self.renderer.set_frame_dest(self.width, self.height)
|
|
|
|
image = self.previewDisplay.preview(self.renderer.bg_frame,
|
|
|
|
main_text[0], self.themedata)
|
2010-07-12 19:36:42 +00:00
|
|
|
return image
|
2009-04-22 19:46:10 +00:00
|
|
|
|
|
|
|
def calculate_default(self, screen):
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
Calculate the default dimentions of the screen.
|
|
|
|
|
|
|
|
``screen``
|
2010-01-09 09:34:06 +00:00
|
|
|
The QSize of the screen.
|
2009-07-10 13:16:15 +00:00
|
|
|
"""
|
|
|
|
log.debug(u'calculate default %s', screen)
|
2010-01-09 09:34:06 +00:00
|
|
|
self.width = screen.width()
|
|
|
|
self.height = screen.height()
|
2009-09-20 07:12:47 +00:00
|
|
|
self.screen_ratio = float(self.height) / float(self.width)
|
2009-09-21 17:56:36 +00:00
|
|
|
log.debug(u'calculate default %d, %d, %f',
|
|
|
|
self.width, self.height, self.screen_ratio )
|
2009-06-17 05:11:16 +00:00
|
|
|
# 90% is start of footer
|
2010-07-25 07:21:10 +00:00
|
|
|
self.footer_start = int(self.height * 0.90)
|