forked from openlp/openlp
641 lines
22 KiB
Python
641 lines
22 KiB
Python
# -*- coding: utf-8 -*-
|
|
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
|
|
|
|
###############################################################################
|
|
# OpenLP - Open Source Lyrics Projection #
|
|
# --------------------------------------------------------------------------- #
|
|
# Copyright (c) 2008-2011 Raoul Snyman #
|
|
# Portions copyright (c) 2008-2011 Tim Bentley, Gerald Britton, Jonathan #
|
|
# Corwin, Michael Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan, #
|
|
# Armin Köhler, Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias #
|
|
# Põldaru, Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith, #
|
|
# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund #
|
|
# --------------------------------------------------------------------------- #
|
|
# 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 #
|
|
###############################################################################
|
|
"""
|
|
Provide the theme XML and handling functions for OpenLP v2 themes.
|
|
"""
|
|
import os
|
|
import re
|
|
import logging
|
|
|
|
from xml.dom.minidom import Document
|
|
from lxml import etree, objectify
|
|
|
|
from openlp.core.lib import str_to_bool
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
BLANK_THEME_XML = \
|
|
'''<?xml version="1.0" encoding="utf-8"?>
|
|
<theme version="1.0">
|
|
<name> </name>
|
|
<background type="image">
|
|
<filename></filename>
|
|
<borderColor>#000000</borderColor>
|
|
</background>
|
|
<background type="gradient">
|
|
<startColor>#000000</startColor>
|
|
<endColor>#000000</endColor>
|
|
<direction>vertical</direction>
|
|
</background>
|
|
<background type="solid">
|
|
<color>#000000</color>
|
|
</background>
|
|
<font type="main">
|
|
<name>Arial</name>
|
|
<color>#FFFFFF</color>
|
|
<size>40</size>
|
|
<bold>False</bold>
|
|
<italics>False</italics>
|
|
<line_adjustment>0</line_adjustment>
|
|
<shadow shadowColor="#000000" shadowSize="5">True</shadow>
|
|
<outline outlineColor="#000000" outlineSize="2">False</outline>
|
|
<location override="False" x="10" y="10" width="1004" height="690"/>
|
|
</font>
|
|
<font type="footer">
|
|
<name>Arial</name>
|
|
<color>#FFFFFF</color>
|
|
<size>12</size>
|
|
<bold>False</bold>
|
|
<italics>False</italics>
|
|
<line_adjustment>0</line_adjustment>
|
|
<shadow shadowColor="#000000" shadowSize="5">True</shadow>
|
|
<outline outlineColor="#000000" outlineSize="2">False</outline>
|
|
<location override="False" x="10" y="690" width="1004" height="78"/>
|
|
</font>
|
|
<display>
|
|
<horizontalAlign>0</horizontalAlign>
|
|
<verticalAlign>0</verticalAlign>
|
|
<slideTransition>False</slideTransition>
|
|
</display>
|
|
</theme>
|
|
'''
|
|
|
|
class ThemeLevel(object):
|
|
"""
|
|
Provides an enumeration for the level a theme applies to
|
|
"""
|
|
Global = 1
|
|
Service = 2
|
|
Song = 3
|
|
|
|
|
|
class BackgroundType(object):
|
|
"""
|
|
Type enumeration for backgrounds.
|
|
"""
|
|
Solid = 0
|
|
Gradient = 1
|
|
Image = 2
|
|
|
|
@staticmethod
|
|
def to_string(background_type):
|
|
"""
|
|
Return a string representation of a background type.
|
|
"""
|
|
if background_type == BackgroundType.Solid:
|
|
return u'solid'
|
|
elif background_type == BackgroundType.Gradient:
|
|
return u'gradient'
|
|
elif background_type == BackgroundType.Image:
|
|
return u'image'
|
|
|
|
@staticmethod
|
|
def from_string(type_string):
|
|
"""
|
|
Return a background type for the given string.
|
|
"""
|
|
if type_string == u'solid':
|
|
return BackgroundType.Solid
|
|
elif type_string == u'gradient':
|
|
return BackgroundType.Gradient
|
|
elif type_string == u'image':
|
|
return BackgroundType.Image
|
|
|
|
|
|
class BackgroundGradientType(object):
|
|
"""
|
|
Type enumeration for background gradients.
|
|
"""
|
|
Horizontal = 0
|
|
Vertical = 1
|
|
Circular = 2
|
|
LeftTop = 3
|
|
LeftBottom = 4
|
|
|
|
@staticmethod
|
|
def to_string(gradient_type):
|
|
"""
|
|
Return a string representation of a background gradient type.
|
|
"""
|
|
if gradient_type == BackgroundGradientType.Horizontal:
|
|
return u'horizontal'
|
|
elif gradient_type == BackgroundGradientType.Vertical:
|
|
return u'vertical'
|
|
elif gradient_type == BackgroundGradientType.Circular:
|
|
return u'circular'
|
|
elif gradient_type == BackgroundGradientType.LeftTop:
|
|
return u'leftTop'
|
|
elif gradient_type == BackgroundGradientType.LeftBottom:
|
|
return u'leftBottom'
|
|
|
|
@staticmethod
|
|
def from_string(type_string):
|
|
"""
|
|
Return a background gradient type for the given string.
|
|
"""
|
|
if type_string == u'horizontal':
|
|
return BackgroundGradientType.Horizontal
|
|
elif type_string == u'vertical':
|
|
return BackgroundGradientType.Vertical
|
|
elif type_string == u'circular':
|
|
return BackgroundGradientType.Circular
|
|
elif type_string == u'leftTop':
|
|
return BackgroundGradientType.LeftTop
|
|
elif type_string == u'leftBottom':
|
|
return BackgroundGradientType.LeftBottom
|
|
|
|
|
|
class HorizontalType(object):
|
|
"""
|
|
Type enumeration for horizontal alignment.
|
|
"""
|
|
Left = 0
|
|
Right = 1
|
|
Center = 2
|
|
|
|
Names = [u'left', u'right', u'center']
|
|
|
|
|
|
class VerticalType(object):
|
|
"""
|
|
Type enumeration for vertical alignment.
|
|
"""
|
|
Top = 0
|
|
Middle = 1
|
|
Bottom = 2
|
|
|
|
Names = [u'top', u'middle', u'bottom']
|
|
|
|
|
|
BOOLEAN_LIST = [u'bold', u'italics', u'override', u'outline', u'shadow',
|
|
u'slide_transition']
|
|
|
|
INTEGER_LIST = [u'size', u'line_adjustment', u'x', u'height', u'y',
|
|
u'width', u'shadow_size', u'outline_size', u'horizontal_align',
|
|
u'vertical_align', u'wrap_style']
|
|
|
|
|
|
class ThemeXML(object):
|
|
"""
|
|
A class to encapsulate the Theme XML.
|
|
"""
|
|
FIRST_CAMEL_REGEX = re.compile(u'(.)([A-Z][a-z]+)')
|
|
SECOND_CAMEL_REGEX = re.compile(u'([a-z0-9])([A-Z])')
|
|
def __init__(self):
|
|
"""
|
|
Initialise the theme object.
|
|
"""
|
|
# Create the minidom document
|
|
self.theme_xml = Document()
|
|
self.parse_xml(BLANK_THEME_XML)
|
|
|
|
def extend_image_filename(self, path):
|
|
"""
|
|
Add the path name to the image name so the background can be rendered.
|
|
|
|
``path``
|
|
The path name to be added.
|
|
"""
|
|
if self.background_type == u'image':
|
|
if self.background_filename and path:
|
|
self.theme_name = self.theme_name.strip()
|
|
self.background_filename = self.background_filename.strip()
|
|
self.background_filename = os.path.join(path, self.theme_name,
|
|
self.background_filename)
|
|
|
|
def _new_document(self, name):
|
|
"""
|
|
Create a new theme XML document.
|
|
"""
|
|
self.theme_xml = Document()
|
|
self.theme = self.theme_xml.createElement(u'theme')
|
|
self.theme_xml.appendChild(self.theme)
|
|
self.theme.setAttribute(u'version', u'2.0')
|
|
self.name = self.theme_xml.createElement(u'name')
|
|
text_node = self.theme_xml.createTextNode(name)
|
|
self.name.appendChild(text_node)
|
|
self.theme.appendChild(self.name)
|
|
|
|
def add_background_transparent(self):
|
|
"""
|
|
Add a transparent background.
|
|
"""
|
|
background = self.theme_xml.createElement(u'background')
|
|
background.setAttribute(u'mode', u'transparent')
|
|
self.theme.appendChild(background)
|
|
|
|
def add_background_solid(self, bkcolor):
|
|
"""
|
|
Add a Solid background.
|
|
|
|
``bkcolor``
|
|
The color of the background.
|
|
"""
|
|
background = self.theme_xml.createElement(u'background')
|
|
background.setAttribute(u'type', u'solid')
|
|
self.theme.appendChild(background)
|
|
self.child_element(background, u'color', unicode(bkcolor))
|
|
|
|
def add_background_gradient(self, startcolor, endcolor, direction):
|
|
"""
|
|
Add a gradient background.
|
|
|
|
``startcolor``
|
|
The gradient's starting colour.
|
|
|
|
``endcolor``
|
|
The gradient's ending colour.
|
|
|
|
``direction``
|
|
The direction of the gradient.
|
|
"""
|
|
background = self.theme_xml.createElement(u'background')
|
|
background.setAttribute(u'type', u'gradient')
|
|
self.theme.appendChild(background)
|
|
# Create startColor element
|
|
self.child_element(background, u'startColor', unicode(startcolor))
|
|
# Create endColor element
|
|
self.child_element(background, u'endColor', unicode(endcolor))
|
|
# Create direction element
|
|
self.child_element(background, u'direction', unicode(direction))
|
|
|
|
def add_background_image(self, filename, borderColor):
|
|
"""
|
|
Add a image background.
|
|
|
|
``filename``
|
|
The file name of the image.
|
|
"""
|
|
background = self.theme_xml.createElement(u'background')
|
|
background.setAttribute(u'type', u'image')
|
|
self.theme.appendChild(background)
|
|
# Create Filename element
|
|
self.child_element(background, u'filename', filename)
|
|
# Create endColor element
|
|
self.child_element(background, u'borderColor', unicode(borderColor))
|
|
|
|
def add_font(self, name, color, size, override, fonttype=u'main',
|
|
bold=u'False', italics=u'False', line_adjustment=0,
|
|
xpos=0, ypos=0, width=0, height=0 , outline=u'False',
|
|
outline_color=u'#ffffff', outline_pixel=2, shadow=u'False',
|
|
shadow_color=u'#ffffff', shadow_pixel=5):
|
|
"""
|
|
Add a Font.
|
|
|
|
``name``
|
|
The name of the font.
|
|
|
|
``color``
|
|
The colour of the font.
|
|
|
|
``size``
|
|
The size of the font.
|
|
|
|
``override``
|
|
Whether or not to override the default positioning of the theme.
|
|
|
|
``fonttype``
|
|
The type of font, ``main`` or ``footer``. Defaults to ``main``.
|
|
|
|
``weight``
|
|
The weight of then font Defaults to 50 Normal
|
|
|
|
``italics``
|
|
Does the font render to italics Defaults to 0 Normal
|
|
|
|
``xpos``
|
|
The X position of the text block.
|
|
|
|
``ypos``
|
|
The Y position of the text block.
|
|
|
|
``width``
|
|
The width of the text block.
|
|
|
|
``height``
|
|
The height of the text block.
|
|
|
|
``outline``
|
|
Whether or not to show an outline.
|
|
|
|
``outline_color``
|
|
The colour of the outline.
|
|
|
|
``outline_size``
|
|
How big the Shadow is
|
|
|
|
``shadow``
|
|
Whether or not to show a shadow.
|
|
|
|
``shadow_color``
|
|
The colour of the shadow.
|
|
|
|
``shadow_size``
|
|
How big the Shadow is
|
|
|
|
"""
|
|
background = self.theme_xml.createElement(u'font')
|
|
background.setAttribute(u'type', fonttype)
|
|
self.theme.appendChild(background)
|
|
# Create Font name element
|
|
self.child_element(background, u'name', name)
|
|
# Create Font color element
|
|
self.child_element(background, u'color', unicode(color))
|
|
# Create Proportion name element
|
|
self.child_element(background, u'size', unicode(size))
|
|
# Create weight name element
|
|
self.child_element(background, u'bold', unicode(bold))
|
|
# Create italics name element
|
|
self.child_element(background, u'italics', unicode(italics))
|
|
# Create indentation name element
|
|
self.child_element(
|
|
background, u'line_adjustment', unicode(line_adjustment))
|
|
# Create Location element
|
|
element = self.theme_xml.createElement(u'location')
|
|
element.setAttribute(u'override', unicode(override))
|
|
element.setAttribute(u'x', unicode(xpos))
|
|
element.setAttribute(u'y', unicode(ypos))
|
|
element.setAttribute(u'width', unicode(width))
|
|
element.setAttribute(u'height', unicode(height))
|
|
background.appendChild(element)
|
|
# Shadow
|
|
element = self.theme_xml.createElement(u'shadow')
|
|
element.setAttribute(u'shadowColor', unicode(shadow_color))
|
|
element.setAttribute(u'shadowSize', unicode(shadow_pixel))
|
|
value = self.theme_xml.createTextNode(unicode(shadow))
|
|
element.appendChild(value)
|
|
background.appendChild(element)
|
|
# Outline
|
|
element = self.theme_xml.createElement(u'outline')
|
|
element.setAttribute(u'outlineColor', unicode(outline_color))
|
|
element.setAttribute(u'outlineSize', unicode(outline_pixel))
|
|
value = self.theme_xml.createTextNode(unicode(outline))
|
|
element.appendChild(value)
|
|
background.appendChild(element)
|
|
|
|
def add_display(self, horizontal, vertical, transition):
|
|
"""
|
|
Add a Display options.
|
|
|
|
``horizontal``
|
|
The horizontal alignment of the text.
|
|
|
|
``vertical``
|
|
The vertical alignment of the text.
|
|
|
|
``transition``
|
|
Whether the slide transition is active.
|
|
|
|
"""
|
|
background = self.theme_xml.createElement(u'display')
|
|
self.theme.appendChild(background)
|
|
# Horizontal alignment
|
|
element = self.theme_xml.createElement(u'horizontalAlign')
|
|
value = self.theme_xml.createTextNode(unicode(horizontal))
|
|
element.appendChild(value)
|
|
background.appendChild(element)
|
|
# Vertical alignment
|
|
element = self.theme_xml.createElement(u'verticalAlign')
|
|
value = self.theme_xml.createTextNode(unicode(vertical))
|
|
element.appendChild(value)
|
|
background.appendChild(element)
|
|
# Slide Transition
|
|
element = self.theme_xml.createElement(u'slideTransition')
|
|
value = self.theme_xml.createTextNode(unicode(transition))
|
|
element.appendChild(value)
|
|
background.appendChild(element)
|
|
|
|
def child_element(self, element, tag, value):
|
|
"""
|
|
Generic child element creator.
|
|
"""
|
|
child = self.theme_xml.createElement(tag)
|
|
child.appendChild(self.theme_xml.createTextNode(value))
|
|
element.appendChild(child)
|
|
return child
|
|
|
|
def dump_xml(self):
|
|
"""
|
|
Dump the XML to file used for debugging
|
|
"""
|
|
return self.theme_xml.toprettyxml(indent=u' ')
|
|
|
|
def extract_xml(self):
|
|
"""
|
|
Print out the XML string.
|
|
"""
|
|
self._build_xml_from_attrs()
|
|
return self.theme_xml.toxml(u'utf-8').decode(u'utf-8')
|
|
|
|
def extract_formatted_xml(self):
|
|
"""
|
|
Pull out the XML string formatted for human consumption
|
|
"""
|
|
self._build_xml_from_attrs()
|
|
return self.theme_xml.toprettyxml(indent=u' ', newl=u'\n',
|
|
encoding=u'utf-8')
|
|
|
|
def parse(self, xml):
|
|
"""
|
|
Read in an XML string and parse it.
|
|
|
|
``xml``
|
|
The XML string to parse.
|
|
"""
|
|
self.parse_xml(unicode(xml))
|
|
|
|
def parse_xml(self, xml):
|
|
"""
|
|
Parse an XML string.
|
|
|
|
``xml``
|
|
The XML string to parse.
|
|
"""
|
|
# remove encoding string
|
|
line = xml.find(u'?>')
|
|
if line:
|
|
xml = xml[line + 2:]
|
|
try:
|
|
theme_xml = objectify.fromstring(xml)
|
|
except etree.XMLSyntaxError:
|
|
log.exception(u'Invalid xml %s', xml)
|
|
return
|
|
xml_iter = theme_xml.getiterator()
|
|
for element in xml_iter:
|
|
parent = element.getparent()
|
|
master = u''
|
|
if parent is not None:
|
|
if element.getparent().tag == u'font':
|
|
master = element.getparent().tag + u'_' + \
|
|
element.getparent().attrib[u'type']
|
|
# set up Outline and Shadow Tags and move to font_main
|
|
if element.getparent().tag == u'display':
|
|
if element.tag.startswith(u'shadow') or \
|
|
element.tag.startswith(u'outline'):
|
|
self._create_attr(u'font_main', element.tag,
|
|
element.text)
|
|
master = element.getparent().tag
|
|
if element.getparent().tag == u'background':
|
|
master = element.getparent().tag
|
|
if element.getparent().attrib:
|
|
for attr in element.getparent().attrib:
|
|
self._create_attr(master, attr, \
|
|
element.getparent().attrib[attr])
|
|
if master:
|
|
self._create_attr(master, element.tag, element.text)
|
|
if element.attrib:
|
|
for attr in element.attrib:
|
|
base_element = attr
|
|
# correction for the shadow and outline tags
|
|
if element.tag == u'shadow' or \
|
|
element.tag == u'outline':
|
|
if not attr.startswith(element.tag):
|
|
base_element = element.tag + u'_' + attr
|
|
self._create_attr(master, base_element,
|
|
element.attrib[attr])
|
|
else:
|
|
if element.tag == u'name':
|
|
self._create_attr(u'theme', element.tag, element.text)
|
|
|
|
def _translate_tags(self, master, element, value):
|
|
"""
|
|
Clean up XML removing and redefining tags
|
|
"""
|
|
master = master.strip().lstrip()
|
|
element = element.strip().lstrip()
|
|
value = unicode(value).strip().lstrip()
|
|
if master == u'display':
|
|
if element == u'wrapStyle':
|
|
return True, None, None, None
|
|
if element.startswith(u'shadow') or element.startswith(u'outline'):
|
|
master = u'font_main'
|
|
# fix bold font
|
|
if element == u'weight':
|
|
element = u'bold'
|
|
if value == u'Normal':
|
|
value = False
|
|
else:
|
|
value = True
|
|
if element == u'proportion':
|
|
element = u'size'
|
|
return False, master, element, value
|
|
|
|
def _create_attr(self, master , element, value):
|
|
"""
|
|
Create the attributes with the correct data types and name format
|
|
"""
|
|
reject, master, element, value = \
|
|
self._translate_tags(master, element, value)
|
|
if reject:
|
|
return
|
|
field = self._de_hump(element)
|
|
tag = master + u'_' + field
|
|
if field in BOOLEAN_LIST:
|
|
setattr(self, tag, str_to_bool(value))
|
|
elif field in INTEGER_LIST:
|
|
setattr(self, tag, int(value))
|
|
else:
|
|
# make string value unicode
|
|
if not isinstance(value, unicode):
|
|
value = unicode(str(value), u'utf-8')
|
|
# None means an empty string so lets have one.
|
|
if value == u'None':
|
|
value = u''
|
|
setattr(self, tag, unicode(value).strip().lstrip())
|
|
|
|
def __str__(self):
|
|
"""
|
|
Return a string representation of this object.
|
|
"""
|
|
theme_strings = []
|
|
for key in dir(self):
|
|
if key[0:1] != u'_':
|
|
theme_strings.append(u'%30s: %s' % (key, getattr(self, key)))
|
|
return u'\n'.join(theme_strings)
|
|
|
|
def _de_hump(self, name):
|
|
"""
|
|
Change Camel Case string to python string
|
|
"""
|
|
sub_name = ThemeXML.FIRST_CAMEL_REGEX.sub(r'\1_\2', name)
|
|
return ThemeXML.SECOND_CAMEL_REGEX.sub(r'\1_\2', sub_name).lower()
|
|
|
|
def _build_xml_from_attrs(self):
|
|
"""
|
|
Build the XML from the varables in the object
|
|
"""
|
|
self._new_document(self.theme_name)
|
|
if self.background_type == \
|
|
BackgroundType.to_string(BackgroundType.Solid):
|
|
self.add_background_solid(self.background_color)
|
|
elif self.background_type == \
|
|
BackgroundType.to_string(BackgroundType.Gradient):
|
|
self.add_background_gradient(
|
|
self.background_start_color,
|
|
self.background_end_color,
|
|
self.background_direction)
|
|
else:
|
|
filename = os.path.split(self.background_filename)[1]
|
|
self.add_background_image(filename, self.background_border_color)
|
|
self.add_font(self.font_main_name,
|
|
self.font_main_color,
|
|
self.font_main_size,
|
|
self.font_main_override, u'main',
|
|
self.font_main_bold,
|
|
self.font_main_italics,
|
|
self.font_main_line_adjustment,
|
|
self.font_main_x,
|
|
self.font_main_y,
|
|
self.font_main_width,
|
|
self.font_main_height,
|
|
self.font_main_outline,
|
|
self.font_main_outline_color,
|
|
self.font_main_outline_size,
|
|
self.font_main_shadow,
|
|
self.font_main_shadow_color,
|
|
self.font_main_shadow_size)
|
|
self.add_font(self.font_footer_name,
|
|
self.font_footer_color,
|
|
self.font_footer_size,
|
|
self.font_footer_override, u'footer',
|
|
self.font_footer_bold,
|
|
self.font_footer_italics,
|
|
0, # line adjustment
|
|
self.font_footer_x,
|
|
self.font_footer_y,
|
|
self.font_footer_width,
|
|
self.font_footer_height,
|
|
self.font_footer_outline,
|
|
self.font_footer_outline_color,
|
|
self.font_footer_outline_size,
|
|
self.font_footer_shadow,
|
|
self.font_footer_shadow_color,
|
|
self.font_footer_shadow_size)
|
|
self.add_display(self.display_horizontal_align,
|
|
self.display_vertical_align,
|
|
self.display_slide_transition)
|