openlp/openlp/core/lib/renderer.py

568 lines
27 KiB
Python
Raw Normal View History

2010-03-21 23:58:01 +00:00
# -*- coding: utf-8 -*-
2012-12-28 22:06:43 +00:00
# vim: autoindent shiftwidth=4 expandtab textwidth=120 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2015-12-31 22:46:06 +00:00
# Copyright (c) 2008-2016 OpenLP Developers #
# --------------------------------------------------------------------------- #
# 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 #
###############################################################################
2013-08-31 18:17:38 +00:00
2011-03-28 18:12:46 +00:00
2015-11-07 00:49:40 +00:00
from PyQt5 import QtGui, QtCore, QtWebKitWidgets
2011-03-28 18:12:46 +00:00
2014-03-16 21:25:23 +00:00
from openlp.core.common import Registry, RegistryProperties, OpenLPMixin, RegistryMixin, Settings
2013-12-13 17:44:05 +00:00
from openlp.core.lib import FormattingTags, ImageSource, ItemCapabilities, ScreenList, ServiceItem, expand_tags, \
build_lyrics_format_css, build_lyrics_outline_css
2013-10-13 20:36:42 +00:00
from openlp.core.common import ThemeLevel
from openlp.core.ui import MainDisplay
2011-03-28 18:12:46 +00:00
2013-08-31 18:17:38 +00:00
VERSE = 'The Lord said to {r}Noah{/r}: \n' \
2011-03-28 18:12:46 +00:00
'There\'s gonna be a {su}floody{/su}, {sb}floody{/sb}\n' \
'The Lord said to {g}Noah{/g}:\n' \
'There\'s gonna be a {st}floody{/st}, {it}floody{/it}\n' \
'Get those children out of the muddy, muddy \n' \
'{r}C{/r}{b}h{/b}{bl}i{/bl}{y}l{/y}{g}d{/g}{pk}' \
'r{/pk}{o}e{/o}{pp}n{/pp} of the Lord\n'
2013-08-31 18:17:38 +00:00
VERSE_FOR_LINE_COUNT = '\n'.join(map(str, range(100)))
FOOTER = ['Arky Arky (Unknown)', 'Public Domain', 'CCLI 123456']
2011-03-28 18:12:46 +00:00
2013-01-10 20:47:10 +00:00
2014-03-16 21:25:23 +00:00
class Renderer(OpenLPMixin, RegistryMixin, RegistryProperties):
2011-03-28 18:12:46 +00:00
"""
2013-03-01 10:46:46 +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.
2011-03-28 18:12:46 +00:00
"""
2013-01-22 21:09:43 +00:00
def __init__(self):
2011-03-28 18:12:46 +00:00
"""
2011-10-16 14:40:46 +00:00
Initialise the renderer.
2011-03-28 18:12:46 +00:00
"""
2014-01-05 17:09:51 +00:00
super(Renderer, self).__init__(None)
2014-01-11 17:52:01 +00:00
# Need live behaviour if this is also working as a pseudo MainDisplay.
self.screens = ScreenList()
2012-05-23 16:46:54 +00:00
self.theme_level = ThemeLevel.Global
2013-08-31 18:17:38 +00:00
self.global_theme_name = ''
self.service_theme_name = ''
self.item_theme_name = ''
2011-03-28 18:12:46 +00:00
self.force_page = False
2012-05-20 19:11:27 +00:00
self._theme_dimensions = {}
self._calculate_default()
2015-11-07 00:49:40 +00:00
self.web = QtWebKitWidgets.QWebView()
2012-06-30 22:20:16 +00:00
self.web.setVisible(False)
self.web_frame = self.web.page().mainFrame()
2014-01-04 20:58:18 +00:00
Registry().register_function('theme_update_global', self.set_global_theme)
def bootstrap_initialise(self):
2014-01-05 17:09:51 +00:00
"""
2014-01-05 17:12:42 +00:00
Initialise functions
2014-01-05 17:09:51 +00:00
"""
2014-01-11 17:52:01 +00:00
self.display = MainDisplay(self)
2014-01-04 20:58:18 +00:00
self.display.setup()
2011-03-28 18:12:46 +00:00
def update_display(self):
"""
2011-10-16 14:40:46 +00:00
Updates the renderer's information about the current screen.
2011-03-28 18:12:46 +00:00
"""
2012-05-24 19:05:15 +00:00
self._calculate_default()
if self.display:
self.display.close()
2014-01-11 17:52:01 +00:00
self.display = MainDisplay(self)
2011-03-28 18:12:46 +00:00
self.display.setup()
2012-05-24 18:00:43 +00:00
self._theme_dimensions = {}
2012-05-23 21:02:34 +00:00
def update_theme(self, theme_name, old_theme_name=None, only_delete=False):
2012-05-23 16:14:03 +00:00
"""
2013-03-01 10:46:46 +00:00
This method updates the theme in ``_theme_dimensions`` when a theme has been edited or renamed.
2012-05-23 16:14:03 +00:00
2014-01-05 17:09:51 +00:00
:param theme_name: The current theme name.
:param old_theme_name: The old theme name. Has only to be passed, when the theme has been renamed.
2015-09-08 19:13:59 +00:00
Defaults to *None*.
2014-01-05 17:09:51 +00:00
:param only_delete: Only remove the given ``theme_name`` from the ``_theme_dimensions`` list. This can be
2015-09-08 19:13:59 +00:00
used when a theme is permanently deleted.
2012-05-23 16:14:03 +00:00
"""
2012-12-28 22:06:43 +00:00
if old_theme_name is not None and old_theme_name in self._theme_dimensions:
2012-05-23 16:14:03 +00:00
del self._theme_dimensions[old_theme_name]
if theme_name in self._theme_dimensions:
del self._theme_dimensions[theme_name]
if not only_delete and theme_name:
2012-05-23 21:02:34 +00:00
self._set_theme(theme_name)
2012-05-20 19:11:27 +00:00
def _set_theme(self, theme_name):
"""
2012-05-20 19:11:27 +00:00
Helper method to save theme names and theme data.
2012-05-23 16:14:03 +00:00
2014-01-05 17:09:51 +00:00
:param theme_name: The theme name
"""
2014-01-11 17:52:01 +00:00
self.log_debug("_set_theme with theme %s" % theme_name)
2012-05-20 19:11:27 +00:00
if theme_name not in self._theme_dimensions:
theme_data = self.theme_manager.get_theme_data(theme_name)
main_rect = self.get_main_rectangle(theme_data)
footer_rect = self.get_footer_rectangle(theme_data)
2012-12-28 22:06:43 +00:00
self._theme_dimensions[theme_name] = [theme_data, main_rect, footer_rect]
else:
2012-12-28 22:06:43 +00:00
theme_data, main_rect, footer_rect = self._theme_dimensions[theme_name]
# if No file do not update cache
if theme_data.background_filename:
self.image_manager.add_image(theme_data.background_filename,
2014-01-05 17:12:42 +00:00
ImageSource.Theme, QtGui.QColor(theme_data.background_border_color))
2011-03-28 18:12:46 +00:00
2012-05-23 16:14:03 +00:00
def pre_render(self, override_theme_data=None):
"""
2012-05-23 16:14:03 +00:00
Set up the theme to be used before rendering an item.
2014-01-05 17:09:51 +00:00
:param override_theme_data: The theme data should be passed, when we want to use our own theme data, regardless
of the theme level. This should for example be used in the theme manager. **Note**, this is **not** to
be mixed up with the ``set_item_theme`` method.
"""
# Just assume we use the global theme.
2012-05-23 16:14:03 +00:00
theme_to_use = self.global_theme_name
2013-03-01 10:46:46 +00:00
# The theme level is either set to Service or Item. Use the service theme if one is set. We also have to use the
# service theme, even when the theme level is set to Item, because the item does not necessarily have to have a
# theme.
2012-05-24 18:00:43 +00:00
if self.theme_level != ThemeLevel.Global:
2013-03-01 10:46:46 +00:00
# When the theme level is at Service and we actually have a service theme then use it.
2012-05-23 16:14:03 +00:00
if self.service_theme_name:
theme_to_use = self.service_theme_name
2012-05-24 18:00:43 +00:00
# If we have Item level and have an item theme then use it.
if self.theme_level == ThemeLevel.Song and self.item_theme_name:
2012-05-23 16:14:03 +00:00
theme_to_use = self.item_theme_name
2012-05-20 19:11:27 +00:00
if override_theme_data is None:
2012-05-24 18:00:43 +00:00
if theme_to_use not in self._theme_dimensions:
self._set_theme(theme_to_use)
2012-12-28 22:06:43 +00:00
theme_data, main_rect, footer_rect = self._theme_dimensions[theme_to_use]
2012-05-20 19:11:27 +00:00
else:
# Ignore everything and use own theme data.
theme_data = override_theme_data
2012-05-20 19:11:27 +00:00
main_rect = self.get_main_rectangle(override_theme_data)
footer_rect = self.get_footer_rectangle(override_theme_data)
self._set_text_rectangle(theme_data, main_rect, footer_rect)
return theme_data, self._rect, self._rect_footer
def set_theme_level(self, theme_level):
"""
2012-05-20 19:11:27 +00:00
Sets the theme level.
2014-01-05 17:09:51 +00:00
:param theme_level: The theme level to be used.
"""
self.theme_level = theme_level
2013-03-14 20:21:04 +00:00
def set_global_theme(self):
2011-03-28 18:12:46 +00:00
"""
2012-05-20 19:11:27 +00:00
Set the global-level theme name.
2011-03-28 18:12:46 +00:00
"""
2013-08-31 18:17:38 +00:00
global_theme_name = Settings().value('themes/global theme')
2012-05-20 19:11:27 +00:00
self._set_theme(global_theme_name)
2012-05-23 16:14:03 +00:00
self.global_theme_name = global_theme_name
2011-03-28 18:12:46 +00:00
def set_service_theme(self, service_theme_name):
2011-03-28 18:12:46 +00:00
"""
Set the service-level theme.
2014-01-05 17:09:51 +00:00
:param service_theme_name: The service level theme's name.
2011-03-28 18:12:46 +00:00
"""
2012-05-20 19:11:27 +00:00
self._set_theme(service_theme_name)
2012-05-23 16:14:03 +00:00
self.service_theme_name = service_theme_name
2011-03-28 18:12:46 +00:00
2012-05-23 16:14:03 +00:00
def set_item_theme(self, item_theme_name):
2011-03-28 18:12:46 +00:00
"""
2013-03-01 10:46:46 +00:00
Set the item-level theme. **Note**, this has to be done for each item we are rendering.
2011-03-28 18:12:46 +00:00
2014-01-05 17:09:51 +00:00
:param item_theme_name: The item theme's name.
2011-03-28 18:12:46 +00:00
"""
2014-01-11 17:52:01 +00:00
self.log_debug("set_item_theme with theme %s" % item_theme_name)
2012-05-23 16:14:03 +00:00
self._set_theme(item_theme_name)
self.item_theme_name = item_theme_name
2011-03-28 18:12:46 +00:00
def generate_preview(self, theme_data, force_page=False):
"""
Generate a preview of a theme.
2014-01-05 17:09:51 +00:00
:param theme_data: The theme to generated a preview for.
:param force_page: Flag to tell message lines per page need to be generated.
2011-03-28 18:12:46 +00:00
"""
# save value for use in format_slide
self.force_page = force_page
# build a service item to generate preview
2013-03-06 22:23:01 +00:00
service_item = ServiceItem()
2011-03-28 18:12:46 +00:00
if self.force_page:
# make big page for theme edit dialog to get line count
2013-03-06 22:23:01 +00:00
service_item.add_from_text(VERSE_FOR_LINE_COUNT)
2011-03-28 18:12:46 +00:00
else:
2013-03-06 22:23:01 +00:00
service_item.add_from_text(VERSE)
service_item.raw_footer = FOOTER
2012-05-20 19:11:27 +00:00
# if No file do not update cache
if theme_data.background_filename:
2013-03-01 10:46:46 +00:00
self.image_manager.add_image(
theme_data.background_filename, ImageSource.Theme, QtGui.QColor(theme_data.background_border_color))
2012-05-23 21:02:34 +00:00
theme_data, main, footer = self.pre_render(theme_data)
2013-12-26 08:56:53 +00:00
service_item.theme_data = theme_data
2013-03-06 22:23:01 +00:00
service_item.main = main
service_item.footer = footer
service_item.render(True)
2011-03-28 18:12:46 +00:00
if not self.force_page:
2013-03-06 22:23:01 +00:00
self.display.build_html(service_item)
raw_html = service_item.get_rendered_frame(0)
2012-08-27 19:04:53 +00:00
self.display.text(raw_html, False)
preview = self.display.preview()
2011-03-28 18:12:46 +00:00
return preview
2011-06-13 09:47:18 +00:00
self.force_page = False
2011-03-28 18:12:46 +00:00
2011-06-13 08:16:50 +00:00
def format_slide(self, text, item):
2011-03-28 18:12:46 +00:00
"""
Calculate how much text can fit on a slide.
2014-01-05 17:09:51 +00:00
:param text: The words to go on the slides.
:param item: The :class:`~openlp.core.lib.serviceitem.ServiceItem` item object.
2011-03-28 18:12:46 +00:00
"""
2014-01-05 17:09:51 +00:00
self.log_debug('format slide')
2011-06-13 08:16:50 +00:00
# Add line endings after each line of text used for bibles.
2013-08-31 18:17:38 +00:00
line_end = '<br>'
2011-06-13 08:16:50 +00:00
if item.is_capable(ItemCapabilities.NoLineBreaks):
2013-08-31 18:17:38 +00:00
line_end = ' '
# Bibles
if item.is_capable(ItemCapabilities.CanWordSplit):
2013-08-31 18:17:38 +00:00
pages = self._paginate_slide_words(text.split('\n'), line_end)
# Songs and Custom
elif item.is_capable(ItemCapabilities.CanSoftBreak):
pages = []
2013-08-31 18:17:38 +00:00
if '[---]' in text:
# Remove two or more option slide breaks next to each other (causing infinite loop).
2014-04-29 11:04:19 +00:00
while '\n[---]\n[---]\n' in text:
text = text.replace('\n[---]\n[---]\n', '\n[---]\n')
2014-11-20 06:58:33 +00:00
while ' [---]' in text:
text = text.replace(' [---]', '[---]')
while '[---] ' in text:
text = text.replace('[---] ', '[---]')
count = 0
# only loop 5 times as there will never be more than 5 incorrect logical splits on a single slide.
while True and count < 5:
2013-08-31 18:17:38 +00:00
slides = text.split('\n[---]\n', 2)
2013-03-01 10:46:46 +00:00
# If there are (at least) two occurrences of [---] we use the first two slides (and neglect the last
# for now).
2011-08-30 13:35:06 +00:00
if len(slides) == 3:
2013-08-31 18:17:38 +00:00
html_text = expand_tags('\n'.join(slides[:2]))
2013-03-01 10:46:46 +00:00
# We check both slides to determine if the optional split is needed (there is only one optional
# split).
2011-08-30 13:35:06 +00:00
else:
2013-08-31 18:17:38 +00:00
html_text = expand_tags('\n'.join(slides))
html_text = html_text.replace('\n', '<br>')
2011-08-28 15:53:27 +00:00
if self._text_fits_on_slide(html_text):
2013-03-01 10:46:46 +00:00
# The first two optional slides fit (as a whole) on one slide. Replace the first occurrence
# of [---].
2013-08-31 18:17:38 +00:00
text = text.replace('\n[---]', '', 1)
2011-08-28 15:53:27 +00:00
else:
2013-03-01 10:46:46 +00:00
# The first optional slide fits, which means we have to render the first optional slide.
2013-08-31 18:17:38 +00:00
text_contains_split = '[---]' in text
if text_contains_split:
2011-11-02 18:58:46 +00:00
try:
2013-08-31 18:17:38 +00:00
text_to_render, text = text.split('\n[---]\n', 1)
2012-06-09 15:46:01 +00:00
except ValueError:
2013-08-31 18:17:38 +00:00
text_to_render = text.split('\n[---]\n')[0]
text = ''
2012-12-28 22:06:43 +00:00
text_to_render, raw_tags, html_tags = self._get_start_tags(text_to_render)
if text:
text = raw_tags + text
else:
2011-08-30 15:12:16 +00:00
text_to_render = text
2013-08-31 18:17:38 +00:00
text = ''
lines = text_to_render.strip('\n').split('\n')
2011-08-30 15:12:16 +00:00
slides = self._paginate_slide(lines, line_end)
if len(slides) > 1 and text:
2011-08-30 15:15:39 +00:00
# Add all slides apart from the last one the list.
2011-08-30 15:12:16 +00:00
pages.extend(slides[:-1])
2013-12-24 08:56:50 +00:00
if text_contains_split:
2013-08-31 18:17:38 +00:00
text = slides[-1] + '\n[---]\n' + text
else:
2013-08-31 18:17:38 +00:00
text = slides[-1] + '\n' + text
text = text.replace('<br>', '\n')
2011-08-30 15:12:16 +00:00
else:
pages.extend(slides)
2013-08-31 18:17:38 +00:00
if '[---]' not in text:
lines = text.strip('\n').split('\n')
pages.extend(self._paginate_slide(lines, line_end))
2011-08-28 15:53:27 +00:00
break
2014-11-26 09:04:09 +00:00
count += 1
else:
# Clean up line endings.
2013-08-31 18:17:38 +00:00
pages = self._paginate_slide(text.split('\n'), line_end)
2011-09-26 16:57:32 +00:00
else:
2013-08-31 18:17:38 +00:00
pages = self._paginate_slide(text.split('\n'), line_end)
2011-06-13 09:05:22 +00:00
new_pages = []
for page in pages:
2013-08-31 18:17:38 +00:00
while page.endswith('<br>'):
page = page[:-4]
2011-06-13 09:05:22 +00:00
new_pages.append(page)
return new_pages
2011-03-28 18:12:46 +00:00
def _calculate_default(self):
2011-03-28 18:12:46 +00:00
"""
2012-03-21 22:27:04 +00:00
Calculate the default dimensions of the screen.
2011-03-28 18:12:46 +00:00
"""
2013-08-31 18:17:38 +00:00
screen_size = self.screens.current['size']
2011-08-05 08:47:31 +00:00
self.width = screen_size.width()
self.height = screen_size.height()
2013-04-24 19:05:34 +00:00
self.screen_ratio = self.height / self.width
2014-01-05 17:09:51 +00:00
self.log_debug('_calculate default %s, %f' % (screen_size, self.screen_ratio))
2011-03-28 18:12:46 +00:00
# 90% is start of footer
self.footer_start = int(self.height * 0.90)
def get_main_rectangle(self, theme_data):
2011-09-28 17:39:26 +00:00
"""
2011-10-03 16:53:54 +00:00
Calculates the placement and size of the main rectangle.
2011-09-28 17:39:26 +00:00
2014-01-05 17:09:51 +00:00
:param theme_data: The theme information
2011-09-28 17:39:26 +00:00
"""
if not theme_data.font_main_override:
return QtCore.QRect(10, 0, self.width - 20, self.footer_start)
2011-03-28 18:12:46 +00:00
else:
return QtCore.QRect(theme_data.font_main_x, theme_data.font_main_y,
2013-12-24 08:56:50 +00:00
theme_data.font_main_width - 1, theme_data.font_main_height - 1)
2011-09-28 17:39:26 +00:00
2012-05-23 16:46:54 +00:00
def get_footer_rectangle(self, theme_data):
2011-09-28 17:39:26 +00:00
"""
2011-10-03 16:53:54 +00:00
Calculates the placement and size of the footer rectangle.
2011-09-28 17:39:26 +00:00
2014-01-05 17:09:51 +00:00
:param theme_data: The theme data.
2011-09-28 17:39:26 +00:00
"""
2012-05-23 16:46:54 +00:00
if not theme_data.font_footer_override:
2013-03-01 10:46:46 +00:00
return QtCore.QRect(10, self.footer_start, self.width - 20, self.height - self.footer_start)
2011-03-28 18:12:46 +00:00
else:
2012-05-23 16:46:54 +00:00
return QtCore.QRect(theme_data.font_footer_x,
2013-12-24 08:56:50 +00:00
theme_data.font_footer_y, theme_data.font_footer_width - 1,
theme_data.font_footer_height - 1)
2011-03-28 18:12:46 +00:00
def _set_text_rectangle(self, theme_data, rect_main, rect_footer):
2011-03-28 18:12:46 +00:00
"""
Sets the rectangle within which text should be rendered.
2014-01-05 17:09:51 +00:00
:param theme_data: The theme data.
:param rect_main: The main text block.
:param rect_footer: The footer text block.
2011-03-28 18:12:46 +00:00
"""
2014-01-05 17:09:51 +00:00
self.log_debug('_set_text_rectangle %s , %s' % (rect_main, rect_footer))
2011-03-28 18:12:46 +00:00
self._rect = rect_main
self._rect_footer = rect_footer
self.page_width = self._rect.width()
self.page_height = self._rect.height()
if theme_data.font_main_shadow:
self.page_width -= int(theme_data.font_main_shadow_size)
self.page_height -= int(theme_data.font_main_shadow_size)
2013-03-01 10:46:46 +00:00
# For the life of my I don't know why we have to completely kill the QWebView in order for the display to work
# properly, but we do. See bug #1041366 for an example of what happens if we take this out.
self.web = None
2015-11-07 00:49:40 +00:00
self.web = QtWebKitWidgets.QWebView()
self.web.setVisible(False)
2011-03-28 18:12:46 +00:00
self.web.resize(self.page_width, self.page_height)
self.web_frame = self.web.page().mainFrame()
2012-04-16 07:02:24 +00:00
# Adjust width and height to account for shadow. outline done in css.
2013-08-31 18:17:38 +00:00
html = """<!DOCTYPE html><html><head><script>
2011-07-30 06:33:51 +00:00
function show_text(newtext) {
var main = document.getElementById('main');
main.innerHTML = newtext;
2011-08-05 08:10:53 +00:00
// We need to be sure that the page is loaded, that is why we
// return the element's height (even though we do not use the
// returned value).
return main.offsetHeight;
2011-07-30 06:33:51 +00:00
}
2011-08-05 07:41:20 +00:00
</script><style>*{margin: 0; padding: 0; border: 0;}
#main {position: absolute; top: 0px; %s %s}</style></head><body>
2011-07-30 06:33:51 +00:00
<div id="main"></div></body></html>""" % \
2013-12-24 08:56:50 +00:00
(build_lyrics_format_css(theme_data, self.page_width, self.page_height),
build_lyrics_outline_css(theme_data))
2011-07-30 06:33:51 +00:00
self.web.setHtml(html)
2012-03-21 22:27:04 +00:00
self.empty_height = self.web_frame.contentsSize().height()
2011-03-28 18:12:46 +00:00
2011-06-13 08:16:50 +00:00
def _paginate_slide(self, lines, line_end):
2011-03-28 18:12:46 +00:00
"""
2013-03-01 10:46:46 +00:00
Figure out how much text can appear on a slide, using the current theme settings.
2014-01-05 17:09:51 +00:00
2013-03-01 10:46:46 +00:00
**Note:** The smallest possible "unit" of text for a slide is one line. If the line is too long it will be cut
off when displayed.
2011-03-28 18:12:46 +00:00
2014-01-05 17:09:51 +00:00
:param lines: The text to be fitted on the slide split into lines.
2014-04-29 11:04:19 +00:00
:param line_end: The text added after each line. Either ``' '`` or ``'<br>``.
2011-03-28 18:12:46 +00:00
"""
formatted = []
2013-08-31 18:17:38 +00:00
previous_html = ''
previous_raw = ''
separator = '<br>'
html_lines = list(map(expand_tags, lines))
2011-06-12 18:38:04 +00:00
# Text too long so go to next page.
2011-08-26 15:22:57 +00:00
if not self._text_fits_on_slide(separator.join(html_lines)):
2013-03-01 10:46:46 +00:00
html_text, previous_raw = self._binary_chop(
2013-08-31 18:17:38 +00:00
formatted, previous_html, previous_raw, html_lines, lines, separator, '')
2011-06-12 18:38:04 +00:00
else:
previous_raw = separator.join(lines)
2012-08-13 19:25:28 +00:00
formatted.append(previous_raw)
2011-03-28 18:12:46 +00:00
return formatted
2011-08-05 08:10:53 +00:00
def _paginate_slide_words(self, lines, line_end):
2011-04-01 04:48:09 +00:00
"""
2013-03-01 10:46:46 +00:00
Figure out how much text can appear on a slide, using the current theme settings.
2014-01-05 17:09:51 +00:00
2013-03-01 10:46:46 +00:00
**Note:** The smallest possible "unit" of text for a slide is one word. If one line is too long it will be
processed word by word. This is sometimes need for **bible** verses.
2011-04-01 04:48:09 +00:00
2014-01-05 17:09:51 +00:00
:param lines: The text to be fitted on the slide split into lines.
2014-04-29 11:04:19 +00:00
:param line_end: The text added after each line. Either ``' '`` or ``'<br>``. This is needed for **bibles**.
2011-04-01 04:48:09 +00:00
"""
formatted = []
2013-08-31 18:17:38 +00:00
previous_html = ''
previous_raw = ''
2011-04-01 04:48:09 +00:00
for line in lines:
2011-05-23 17:28:52 +00:00
line = line.strip()
2011-06-13 09:05:22 +00:00
html_line = expand_tags(line)
2011-05-23 18:41:49 +00:00
# Text too long so go to next page.
2011-08-26 15:22:57 +00:00
if not self._text_fits_on_slide(previous_html + html_line):
2013-03-01 10:46:46 +00:00
# Check if there was a verse before the current one and append it, when it fits on the page.
if previous_html:
2011-08-26 15:22:57 +00:00
if self._text_fits_on_slide(previous_html):
formatted.append(previous_raw)
2013-08-31 18:17:38 +00:00
previous_html = ''
previous_raw = ''
2013-03-01 10:46:46 +00:00
# Now check if the current verse will fit, if it does not we have to start to process the verse
# word by word.
2011-08-26 15:22:57 +00:00
if self._text_fits_on_slide(html_line):
2011-06-13 09:05:22 +00:00
previous_html = html_line + line_end
previous_raw = line + line_end
continue
2013-03-01 10:46:46 +00:00
# Figure out how many words of the line will fit on screen as the line will not fit as a whole.
2011-05-23 17:28:52 +00:00
raw_words = self._words_split(line)
2013-08-31 18:17:38 +00:00
html_words = list(map(expand_tags, raw_words))
2012-12-28 22:06:43 +00:00
previous_html, previous_raw = \
2013-08-31 18:17:38 +00:00
self._binary_chop(formatted, previous_html, previous_raw, html_words, raw_words, ' ', line_end)
2011-04-09 08:11:46 +00:00
else:
2011-06-13 09:05:22 +00:00
previous_html += html_line + line_end
2011-04-26 17:03:19 +00:00
previous_raw += line + line_end
formatted.append(previous_raw)
2011-04-01 04:48:09 +00:00
return formatted
2011-09-18 10:26:47 +00:00
def _get_start_tags(self, raw_text):
"""
2013-03-01 10:46:46 +00:00
Tests the given text for not closed formatting tags and returns a tuple consisting of three unicode strings::
2011-09-18 10:26:47 +00:00
2014-04-29 11:04:19 +00:00
('{st}{r}Text text text{/r}{/st}', '{st}{r}', '<strong><span style="-webkit-text-fill-color:red">')
2011-09-18 10:26:47 +00:00
2013-03-01 10:46:46 +00:00
The first unicode string is the text, with correct closing tags. The second unicode string are OpenLP's opening
formatting tags and the third unicode string the html opening formatting tags.
2011-09-18 10:26:47 +00:00
2014-01-05 17:09:51 +00:00
:param raw_text: The text to test. The text must **not** contain html tags, only OpenLP formatting tags
are allowed::
2011-09-19 16:00:27 +00:00
{st}{r}Text text text
2011-09-18 10:26:47 +00:00
"""
raw_tags = []
html_tags = []
2011-09-06 11:56:52 +00:00
for tag in FormattingTags.get_html_tags():
2013-08-31 18:17:38 +00:00
if tag['start tag'] == '{br}':
2011-09-06 11:56:52 +00:00
continue
2013-08-31 18:17:38 +00:00
if raw_text.count(tag['start tag']) != raw_text.count(tag['end tag']):
raw_tags.append((raw_text.find(tag['start tag']), tag['start tag'], tag['end tag']))
html_tags.append((raw_text.find(tag['start tag']), tag['start html']))
2013-03-01 10:46:46 +00:00
# Sort the lists, so that the tags which were opened first on the first slide (the text we are checking) will be
# opened first on the next slide as well.
2011-09-19 16:00:27 +00:00
raw_tags.sort(key=lambda tag: tag[0])
html_tags.sort(key=lambda tag: tag[0])
# Create a list with closing tags for the raw_text.
2012-05-02 18:25:37 +00:00
end_tags = []
start_tags = []
for tag in raw_tags:
start_tags.append(tag[1])
end_tags.append(tag[2])
2011-09-19 16:00:27 +00:00
end_tags.reverse()
2011-09-18 10:26:47 +00:00
# Remove the indexes.
2011-09-19 16:00:27 +00:00
html_tags = [tag[1] for tag in html_tags]
2013-08-31 18:17:38 +00:00
return raw_text + ''.join(end_tags), ''.join(start_tags), ''.join(html_tags)
2011-09-06 11:56:52 +00:00
2012-12-28 22:06:43 +00:00
def _binary_chop(self, formatted, previous_html, previous_raw, html_list, raw_list, separator, line_end):
2011-06-12 18:38:04 +00:00
"""
2013-03-01 10:46:46 +00:00
This implements the binary chop algorithm for faster rendering. This algorithm works line based (line by line)
and word based (word by word). It is assumed that this method is **only** called, when the lines/words to be
rendered do **not** fit as a whole.
2011-06-12 18:38:04 +00:00
2014-01-05 17:09:51 +00:00
:param formatted: The list to append any slides.
:param previous_html: The html text which is know to fit on a slide, but is not yet added to the list of
slides. (unicode string)
:param previous_raw: The raw text (with formatting tags) which is know to fit on a slide, but is not yet added
to the list of slides. (unicode string)
:param html_list: The elements which do not fit on a slide and needs to be processed using the binary chop.
The text contains html.
:param raw_list: The elements which do not fit on a slide and needs to be processed using the binary chop.
The elements can contain formatting tags.
2014-04-29 11:04:19 +00:00
:param separator: The separator for the elements. For lines this is ``'<br>'`` and for words this is ``' '``.
:param line_end: The text added after each "element line". Either ``' '`` or ``'<br>``. This is needed for
2014-01-05 17:09:51 +00:00
bibles.
2011-06-12 18:38:04 +00:00
"""
smallest_index = 0
2011-06-12 19:27:19 +00:00
highest_index = len(html_list) - 1
2013-04-24 19:05:34 +00:00
index = highest_index // 2
2011-06-12 18:38:04 +00:00
while True:
2013-03-01 10:46:46 +00:00
if not self._text_fits_on_slide(previous_html + separator.join(html_list[:index + 1]).strip()):
# We know that it does not fit, so change/calculate the new index and highest_index accordingly.
2011-06-12 18:38:04 +00:00
highest_index = index
2013-04-24 19:05:34 +00:00
index = index - (index - smallest_index) // 2
2011-06-12 18:38:04 +00:00
else:
smallest_index = index
2013-04-24 19:05:34 +00:00
index = index + (highest_index - index) // 2
2011-06-12 18:38:04 +00:00
# We found the number of words which will fit.
if smallest_index == index or highest_index == index:
index = smallest_index
2013-08-31 18:17:38 +00:00
text = previous_raw.rstrip('<br>') + separator.join(raw_list[:index + 1])
2011-09-19 16:00:27 +00:00
text, raw_tags, html_tags = self._get_start_tags(text)
2011-09-06 11:56:52 +00:00
formatted.append(text)
2013-08-31 18:17:38 +00:00
previous_html = ''
previous_raw = ''
2011-06-13 08:16:50 +00:00
# Stop here as the theme line count was requested.
if self.force_page:
2013-08-31 18:17:38 +00:00
Registry().execute('theme_line_count', index + 1)
2011-06-13 08:16:50 +00:00
break
2011-06-12 18:38:04 +00:00
else:
continue
2011-06-13 12:59:52 +00:00
# Check if the remaining elements fit on the slide.
2013-03-01 10:46:46 +00:00
if self._text_fits_on_slide(html_tags + separator.join(html_list[index + 1:]).strip()):
2012-12-28 22:06:43 +00:00
previous_html = html_tags + separator.join(html_list[index + 1:]).strip() + line_end
previous_raw = raw_tags + separator.join(raw_list[index + 1:]).strip() + line_end
2011-06-12 18:38:04 +00:00
break
else:
2013-03-01 10:46:46 +00:00
# The remaining elements do not fit, thus reset the indexes, create a new list and continue.
2011-06-12 19:27:19 +00:00
raw_list = raw_list[index + 1:]
2011-09-06 11:56:52 +00:00
raw_list[0] = raw_tags + raw_list[0]
2011-06-12 19:27:19 +00:00
html_list = html_list[index + 1:]
2011-09-06 11:56:52 +00:00
html_list[0] = html_tags + html_list[0]
2011-06-12 18:38:04 +00:00
smallest_index = 0
2011-06-12 19:27:19 +00:00
highest_index = len(html_list) - 1
2013-04-24 19:05:34 +00:00
index = highest_index // 2
2011-06-13 08:16:50 +00:00
return previous_html, previous_raw
2011-06-12 18:38:04 +00:00
2011-07-30 06:33:51 +00:00
def _text_fits_on_slide(self, text):
"""
2013-03-01 10:46:46 +00:00
Checks if the given ``text`` fits on a slide. If it does ``True`` is returned, otherwise ``False``.
2011-07-30 06:33:51 +00:00
2014-01-05 17:09:51 +00:00
:param text: The text to check. It may contain HTML tags.
2011-07-30 06:33:51 +00:00
"""
2013-08-31 18:17:38 +00:00
self.web_frame.evaluateJavaScript('show_text("%s")' % text.replace('\\', '\\\\').replace('\"', '\\\"'))
2012-03-21 22:27:04 +00:00
return self.web_frame.contentsSize().height() <= self.empty_height
2011-07-30 06:33:51 +00:00
2011-04-26 19:09:56 +00:00
def _words_split(self, line):
2011-03-28 18:12:46 +00:00
"""
Split the slide up by word so can wrap better
2014-01-05 17:09:51 +00:00
:param line: Line to be split
2011-03-28 18:12:46 +00:00
"""
2011-04-08 15:35:33 +00:00
# this parse we are to be wordy
2013-08-31 18:17:38 +00:00
line = line.replace('\n', ' ')
return line.split(' ')