# -*- 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 Tim Bentley, Jonathan Corwin, Michael # # Gorven, Scott Guerrieri, Maikel Stuivenberg, Martin Thompson, Jon Tibble, # # Carsten Tinggaard # # --------------------------------------------------------------------------- # # 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 from PyQt4 import QtGui, QtCore class Renderer(object): """ Genarates a pixmap image of a array of text. The Text is formatted to make sure it fits on the screen and if not extra frames are generated. """ global log log = logging.getLogger(u'Renderer') log.info(u'Renderer Loaded') def __init__(self): """ Initialise the renderer. """ self._rect = None self._debug = 0 self._right_margin = 64 # the amount of right indent self._shadow_offset = 5 self._shadow_offset_footer = 3 self._outline_offset = 2 self.theme_name = None self._theme = None self._bg_image_filename = None self._frame = None self._frameOp = None self.bg_frame = None self.bg_image = None def set_debug(self, debug): """ Set the debug mode of the renderer. ``debug`` The debug mode. """ self._debug = debug def set_theme(self, theme): """ Set the theme to be used. ``theme`` The theme to be used. """ log.debug(u'set theme') self._theme = theme self.bg_frame = None self.bg_image = None self._bg_image_filename = None self.theme_name = theme.theme_name self._set_theme_font() if theme.background_type == u'image': if theme.background_filename: self.set_bg_image(theme.background_filename) def set_bg_image(self, filename): """ Set a background image. ``filename`` The name of the image file. """ log.debug(u'set bg image %s', filename) self._bg_image_filename = unicode(filename) if self._frame: self.scale_bg_image() def scale_bg_image(self): """ Scale the background image to fit the screen. """ assert self._frame preview = QtGui.QImage(self._bg_image_filename) width = self._frame.width() height = self._frame.height() preview = preview.scaled(width, height, QtCore.Qt.KeepAspectRatio, QtCore.Qt.SmoothTransformation) realwidth = preview.width() realheight = preview.height() # and move it to the centre of the preview space self.bg_image = QtGui.QImage(width, height, QtGui.QImage.Format_ARGB32_Premultiplied) self.bg_image.fill(QtCore.Qt.black) painter = QtGui.QPainter() painter.begin(self.bg_image) self.background_offsetx = (width - realwidth) / 2 self.background_offsety = (height - realheight) / 2 painter.drawImage(self.background_offsetx, self.background_offsety, preview) painter.end() def set_frame_dest(self, frame_width, frame_height, preview=False): """ Set the size of the slide. ``frame_width`` The width of the slide. ``frame_height`` The height of the slide. ``preview`` Defaults to *False*. Whether or not to generate a preview. """ if preview: self.bg_frame = None log.debug(u'set frame dest (frame) w %d h %d', frame_width, frame_height) self._frame = QtGui.QImage(frame_width, frame_height, QtGui.QImage.Format_ARGB32_Premultiplied) self._frameOp = QtGui.QImage(frame_width, frame_height, QtGui.QImage.Format_ARGB32_Premultiplied) if self._bg_image_filename and not self.bg_image: self.scale_bg_image() if self.bg_frame is None: self._generate_background_frame() def format_slide(self, words, footer): """ Figure out how much text can appear on a slide, using the current theme settings. ``words`` The words to be fitted on the slide. ``footer`` The footer of the slide. """ log.debug(u'format_slide - Start') words = words.replace(u'\r\n', u'\n') verses_text = words.split(u'\n') text = [] for verse in verses_text: lines = verse.split(u'\n') for line in lines: text.append(line) split_text = self.pre_render_text(text) log.debug(u'format_slide - End') return split_text def pre_render_text(self, text): metrics = QtGui.QFontMetrics(self.mainFont) #take the width work out approx how many characters and add 50% line_width = self._rect.width() - self._right_margin #number of lines on a page - adjust for rounding up. page_length = int(self._rect.height() / metrics.height() - 2 ) - 1 #Average number of characters in line ave_line_width = line_width / metrics.averageCharWidth() #Maximum size of a character max_char_width = metrics.maxWidth() #Min size of a character min_char_width = metrics.width(u'i') char_per_line = line_width / min_char_width log.debug(u'Page Length area height %s , metrics %s , lines %s' % (int(self._rect.height()), metrics.height(), page_length )) split_pages = [] page = [] split_lines = [] count = 0 for line in text: #Must be a blank line so keep it. if len(line) == 0: line = u' ' while len(line) > 0: pos = char_per_line split_text = line[:pos] #line needs splitting if metrics.width(split_text, -1) > line_width: #We have no spaces if split_text.find(u' ') == -1: #Move back 1 char at a time till it fits while metrics.width(split_text, -1) > line_width: split_text = split_text[:-1] pos = len(split_text) else: #We have spaces so split at previous one while metrics.width(split_text, -1) > line_width: pos = split_text.rfind(u' ') #no more spaces and we are still too long if pos == -1: while metrics.width(split_text, -1) > line_width: split_text = split_text[:-1] pos = len(split_text) else: split_text = line[:pos] split_lines.append(split_text) line = line[pos:].lstrip() #if we have more text add up to 10 spaces on the front. if len(line) > 0 and self._theme.font_main_indentation > 0: line = u'%s%s' % \ (u' '[:int(self._theme.font_main_indentation)], line) #Text fits in a line now for count, line in enumerate(split_lines): page.append(line) #last but one line and only 2 lines to go or end of page if (len(page) == page_length - 1 and len(split_lines) - 3 == count) or \ len(page) == page_length: split_pages.append(page) page = [] if len(page) > 0 and page != u' ': split_pages.append(page) return split_pages def set_text_rectangle(self, rect_main, rect_footer): """ Sets the rectangle within which text should be rendered. ``rect_main`` The main text block. ``rect_footer`` The footer text block. """ self._rect = rect_main self._rect_footer = rect_footer def generate_frame_from_lines(self, lines, footer_lines=None): """ Render a set of lines according to the theme, and return the block dimensions. ``lines`` The lines to be rendered. ``footer_lines`` Defaults to *None*. The footer to render. """ log.debug(u'generate_frame_from_lines - Start') bbox = self._render_lines_unaligned(lines, False) if footer_lines: bbox1 = self._render_lines_unaligned(footer_lines, True) # reset the frame. first time do not worry about what you paint on. self._frame = QtGui.QImage(self.bg_frame) self._frameOp = QtGui.QImage(self.bg_frame) x, y = self._correctAlignment(self._rect, bbox) bbox = self._render_lines_unaligned(lines, False, (x, y), True) if footer_lines: bbox = self._render_lines_unaligned(footer_lines, True, (self._rect_footer.left(), self._rect_footer.top()), True) log.debug(u'generate_frame_from_lines - Finish') if self._theme.display_slideTransition: return {u'main':self._frame, u'trans':self._frameOp} else: return {u'main':self._frame, u'trans':None} def _generate_background_frame(self): """ Generate a background frame to the same size as the frame to be used. Results are cached for performance reasons. """ assert(self._theme) self.bg_frame = QtGui.QImage(self._frame.width(), self._frame.height(), QtGui.QImage.Format_ARGB32_Premultiplied) log.debug(u'render background %s start', self._theme.background_type) painter = QtGui.QPainter() painter.begin(self.bg_frame) if self._theme.background_mode == u'transparent': painter.fillRect(self._frame.rect(), QtCore.Qt.transparent) else: if self._theme.background_type == u'solid': painter.fillRect(self._frame.rect(), QtGui.QColor(self._theme.background_color)) elif self._theme.background_type == u'gradient': # gradient gradient = None if self._theme.background_direction == u'horizontal': w = int(self._frame.width()) / 2 # vertical gradient = QtGui.QLinearGradient(w, 0, w, self._frame.height()) elif self._theme.background_direction == u'vertical': h = int(self._frame.height()) / 2 # Horizontal gradient = QtGui.QLinearGradient(0, h, self._frame.width(), h) else: w = int(self._frame.width()) / 2 h = int(self._frame.height()) / 2 # Circular gradient = QtGui.QRadialGradient(w, h, w) gradient.setColorAt(0, QtGui.QColor(self._theme.background_startColor)) gradient.setColorAt(1, QtGui.QColor(self._theme.background_endColor)) painter.setBrush(QtGui.QBrush(gradient)) rectPath = QtGui.QPainterPath() max_x = self._frame.width() max_y = self._frame.height() rectPath.moveTo(0, 0) rectPath.lineTo(0, max_y) rectPath.lineTo(max_x, max_y) rectPath.lineTo(max_x, 0) rectPath.closeSubpath() painter.drawPath(rectPath) elif self._theme.background_type == u'image': # image painter.fillRect(self._frame.rect(), QtCore.Qt.black) if self.bg_image: painter.drawImage(0, 0, self.bg_image) painter.end() log.debug(u'render background End') def _correctAlignment(self, rect, bbox): """ Corrects the vertical alignment of text. ``rect`` The block dimentions. ``bbox`` Footer dimensions? """ x = rect.left() if self._theme.display_verticalAlign == 0: # top align y = rect.top() elif self._theme.display_verticalAlign == 2: # bottom align y = rect.bottom() - bbox.height() elif self._theme.display_verticalAlign == 1: # centre align y = rect.top() + (rect.height() - bbox.height()) / 2 else: log.error(u'Invalid value for theme.VerticalAlign:%s', self._theme.display_verticalAlign) return x, y def _render_lines_unaligned(self, lines, footer, tlcorner=(0, 0), live=False): """ Given a list of lines to render, render each one in turn (using the ``_render_single_line`` fn - which may result in going off the bottom). They are expected to be pre-arranged to less than a screenful (eg. by using split_set_of_lines). Returns the bounding box of the text as QRect. ``lines`` The lines of text to render. ``footer`` The slide footer. ``tlcorner`` Defaults to *``(0, 0)``*. Co-ordinates of the top left corner. ``live`` Defaults to *False*. Whether or not this is a live screen. """ x, y = tlcorner brx = x bry = y for line in lines: # render after current bottom, but at original left edge # keep track of right edge to see which is biggest (thisx, bry) = self._render_and_wrap_single_line(line, footer, (x, bry), live) if (thisx > brx): brx = thisx retval = QtCore.QRect(x, y, brx - x, bry - y) if self._debug: painter = QtGui.QPainter() painter.begin(self._frame) painter.setPen(QtGui.QPen(QtGui.QColor(0, 0, 255))) painter.drawRect(retval) painter.end() return retval def _render_and_wrap_single_line(self, line, footer, tlcorner=(0, 0), live=False): """ Render a single line of words onto the DC, top left corner specified. If the line is too wide for the context, it wraps, but right-aligns the surplus words in the manner of song lyrics. Returns the bottom-right corner (of what was rendered) as a tuple(x, y). ``line`` Line of text to be rendered. ``footer`` The footer of the slide. ``tlcorner`` Defaults to *``(0, 0)``*. The top left corner. ``live`` Defaults to *False*. Whether or not this is a live screen. """ x, y = tlcorner maxx = self._rect.width(); maxy = self._rect.height(); lines = [] lines.append(line) startx = x starty = y rightextent = None # dont allow alignment messing with footers if footer: align = 0 shadow_offset = self._shadow_offset_footer else: align = self._theme.display_horizontalAlign shadow_offset = self._shadow_offset for linenum in range(len(lines)): line = lines[linenum] #find out how wide line is w, h = self._get_extent_and_render(line, footer, tlcorner=(x, y), draw=False) if self._theme.display_shadow: w += shadow_offset h += shadow_offset if self._theme.display_outline: # pixels either side w += 2 * self._outline_offset # pixels top/bottom h += 2 * self._outline_offset if align == 0: # left align rightextent = x + w # shift right from last line's rh edge if self._theme.display_wrapStyle == 1 and linenum != 0: rightextent = self._first_line_right_extent + \ self._right_margin if rightextent > maxx: rightextent = maxx x = rightextent - w # right align elif align == 1: rightextent = maxx x = maxx - w # centre elif align == 2: x = (maxx - w) / 2; rightextent = x + w if live: # now draw the text, and any outlines/shadows if self._theme.display_shadow: self._get_extent_and_render(line, footer, tlcorner=(x + shadow_offset, y + shadow_offset), draw=True, color = self._theme.display_shadow_color) if self._theme.display_outline: self._get_extent_and_render(line, footer, (x + self._outline_offset, y), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer, (x, y + self._outline_offset), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer, (x, y - self._outline_offset), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer, (x - self._outline_offset, y), draw=True, color = self._theme.display_outline_color) if self._outline_offset > 1: self._get_extent_and_render(line, footer, (x + self._outline_offset, y + self._outline_offset), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer, (x - self._outline_offset, y + self._outline_offset), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer, (x + self._outline_offset, y - self._outline_offset), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer, (x - self._outline_offset, y - self._outline_offset), draw=True, color = self._theme.display_outline_color) self._get_extent_and_render(line, footer,tlcorner=(x, y), draw=True) y += h if linenum == 0: self._first_line_right_extent = rightextent # draw a box around the text - debug only if self._debug: painter = QtGui.QPainter() painter.begin(self._frame) painter.setPen(QtGui.QPen(QtGui.QColor(0,255,0))) painter.drawRect(startx, starty, rightextent-startx, y-starty) painter.end() brcorner = (rightextent, y) return brcorner def _set_theme_font(self): """ Set the fonts from the current theme settings. """ footer_weight = 50 if self._theme.font_footer_weight == u'Bold': footer_weight = 75 self.footerFont = QtGui.QFont(self._theme.font_footer_name, self._theme.font_footer_proportion, # size footer_weight, # weight self._theme.font_footer_italics) # italic self.footerFont.setPixelSize(self._theme.font_footer_proportion) main_weight = 50 if self._theme.font_main_weight == u'Bold': main_weight = 75 self.mainFont = QtGui.QFont(self._theme.font_main_name, self._theme.font_main_proportion, # size main_weight, # weight self._theme.font_main_italics)# italic self.mainFont.setPixelSize(self._theme.font_main_proportion) def _get_extent_and_render(self, line, footer, tlcorner=(0, 0), draw=False, color=None): """ Find bounding box of text - as render_single_line. If draw is set, actually draw the text to the current DC as well return width and height of text as a tuple (w, h). ``line`` The line of text to render. ``footer`` The footer text. ``tlcorner`` Defaults to *``(0, 0)``*. The top left corner co-ordinates. ``draw`` Defaults to *False*. Draw the text to the current surface. ``color`` Defaults to *None*. The colour to draw with. """ # setup defaults painter = QtGui.QPainter() painter.begin(self._frame) painter.setRenderHint(QtGui.QPainter.Antialiasing); if footer : font = self.footerFont else: font = self.mainFont painter.setFont(font) if color is None: if footer: painter.setPen(QtGui.QColor(self._theme.font_footer_color)) else: painter.setPen(QtGui.QColor(self._theme.font_main_color)) else: painter.setPen(QtGui.QColor(color)) x, y = tlcorner metrics = QtGui.QFontMetrics(font) w = metrics.width(line) h = metrics.height() - 2 if draw: painter.drawText(x, y + metrics.ascent(), line) painter.end() if self._theme.display_slideTransition: # Print 2nd image with 50% weight painter = QtGui.QPainter() painter.begin(self._frameOp) painter.setRenderHint(QtGui.QPainter.Antialiasing); painter.setOpacity(0.7) painter.setFont(font) if color is None: if footer: painter.setPen(QtGui.QColor(self._theme.font_footer_color)) else: painter.setPen(QtGui.QColor(self._theme.font_main_color)) else: painter.setPen(QtGui.QColor(color)) if draw: painter.drawText(x, y + metrics.ascent(), line) painter.end() return (w, h) def snoop_Image(self, image, image2=None): """ Debugging method to allow images to be viewed. ``image`` An image to save to disk. ``image2`` Defaults to *None*. Another image to save to disk. """ image.save(u'renderer.png', u'png') if image2: image2.save(u'renderer2.png', u'png')