diff --git a/openlp/core/common/__init__.py b/openlp/core/common/__init__.py
index f11b13727..1f1ddb1d1 100644
--- a/openlp/core/common/__init__.py
+++ b/openlp/core/common/__init__.py
@@ -45,7 +45,7 @@ log = logging.getLogger(__name__ + '.__init__')
FIRST_CAMEL_REGEX = re.compile('(.)([A-Z][a-z]+)')
SECOND_CAMEL_REGEX = re.compile('([a-z0-9])([A-Z])')
CONTROL_CHARS = re.compile(r'[\x00-\x08\x0B\x0C\x0E-\x1F\x7F-\x9F]')
-INVALID_FILE_CHARS = re.compile(r'[\\/:\*\?"<>\|\+\[\]%]')
+INVALID_FILE_CHARS = re.compile(r'[\\/:*?"<>|+\[\]%]')
IMAGES_FILTER = None
REPLACMENT_CHARS_MAP = str.maketrans({'\u2018': '\'', '\u2019': '\'', '\u201c': '"', '\u201d': '"', '\u2026': '...',
'\u2013': '-', '\u2014': '-', '\v': '\n\n', '\f': '\n\n'})
@@ -103,21 +103,21 @@ def trace_error_handler(logger):
logger.error(log_string)
-def extension_loader(glob_pattern, excluded_files=[]):
+def extension_loader(glob_pattern, excluded_files=None):
"""
A utility function to find and load OpenLP extensions, such as plugins, presentation and media controllers and
importers.
:param str glob_pattern: A glob pattern used to find the extension(s) to be imported. Should be relative to the
application directory. i.e. plugins/*/*plugin.py
- :param list[str] excluded_files: A list of file names to exclude that the glob pattern may find.
+ :param list[str] | None excluded_files: A list of file names to exclude that the glob pattern may find.
:rtype: None
"""
from openlp.core.common.applocation import AppLocation
app_dir = AppLocation.get_directory(AppLocation.AppDir)
for extension_path in app_dir.glob(glob_pattern):
extension_path = extension_path.relative_to(app_dir)
- if extension_path.name in excluded_files:
+ if extension_path.name in (excluded_files or []):
continue
log.debug('Attempting to import %s', extension_path)
module_name = path_to_module(extension_path)
@@ -172,6 +172,21 @@ class SlideLimits(object):
Next = 3
+class Singleton(type):
+ """
+ Provide a `Singleton` metaclass https://stackoverflow.com/questions/6760685/creating-a-singleton-in-python
+ """
+ _instances = {}
+
+ def __call__(cls, *args, **kwargs):
+ """
+ Create a new instance if one does not already exist.
+ """
+ if cls not in cls._instances:
+ cls._instances[cls] = super().__call__(*args, **kwargs)
+ return cls._instances[cls]
+
+
def de_hump(name):
"""
Change any Camel Case string to python string
@@ -385,7 +400,7 @@ def get_images_filter():
global IMAGES_FILTER
if not IMAGES_FILTER:
log.debug('Generating images filter.')
- formats = list(map(bytes.decode, list(map(bytes, QtGui.QImageReader.supportedImageFormats()))))
+ formats = list(map(bytes.decode, map(bytes, QtGui.QImageReader.supportedImageFormats())))
visible_formats = '(*.{text})'.format(text='; *.'.join(formats))
actual_formats = '(*.{text})'.format(text=' *.'.join(formats))
IMAGES_FILTER = '{text} {visible} {actual}'.format(text=translate('OpenLP', 'Image Files'),
diff --git a/openlp/core/common/actions.py b/openlp/core/common/actions.py
index 6d3cda905..8681888e8 100644
--- a/openlp/core/common/actions.py
+++ b/openlp/core/common/actions.py
@@ -260,7 +260,7 @@ class ActionList(object):
return
# We have to do this to ensure that the loaded shortcut list e. g. STRG+O (German) is converted to CTRL+O,
# which is only done when we convert the strings in this way (QKeySequencet -> uncode).
- shortcuts = list(map(QtGui.QKeySequence.toString, list(map(QtGui.QKeySequence, shortcuts))))
+ shortcuts = list(map(QtGui.QKeySequence.toString, map(QtGui.QKeySequence, shortcuts)))
# Check the alternate shortcut first, to avoid problems when the alternate shortcut becomes the primary shortcut
# after removing the (initial) primary shortcut due to conflicts.
if len(shortcuts) == 2:
diff --git a/openlp/core/common/i18n.py b/openlp/core/common/i18n.py
index 85fe35e2f..91a2a6f22 100644
--- a/openlp/core/common/i18n.py
+++ b/openlp/core/common/i18n.py
@@ -29,7 +29,7 @@ from collections import namedtuple
from PyQt5 import QtCore, QtWidgets
-from openlp.core.common import is_macosx, is_win
+from openlp.core.common import Singleton, is_macosx, is_win
from openlp.core.common.applocation import AppLocation
from openlp.core.common.settings import Settings
@@ -327,22 +327,11 @@ class LanguageManager(object):
return LanguageManager.__qm_list__
-class UiStrings(object):
+class UiStrings(metaclass=Singleton):
"""
Provide standard strings for objects to use.
"""
- __instance__ = None
-
- def __new__(cls):
- """
- Override the default object creation method to return a single instance.
- """
- if not cls.__instance__:
- cls.__instance__ = object.__new__(cls)
- cls.load(cls)
- return cls.__instance__
-
- def load(self):
+ def __init__(self):
"""
These strings should need a good reason to be retranslated elsewhere.
Should some/more/less of these have an & attached?
@@ -436,6 +425,7 @@ class UiStrings(object):
self.ResetLiveBG = translate('OpenLP.Ui', 'Reset live background.')
self.RequiredShowInFooter = translate('OpenLP.Ui', 'Required, this will be displayed in footer.')
self.Seconds = translate('OpenLP.Ui', 's', 'The abbreviated unit for seconds')
+ self.SaveAndClose = translate('OpenLP.ui', translate('SongsPlugin.EditSongForm', '&Save && Close'))
self.SaveAndPreview = translate('OpenLP.Ui', 'Save && Preview')
self.Search = translate('OpenLP.Ui', 'Search')
self.SearchThemes = translate('OpenLP.Ui', 'Search Themes...', 'Search bar place holder text ')
@@ -503,7 +493,7 @@ def format_time(text, local_time):
"""
return local_time.strftime(match.group())
- return re.sub(r'\%[a-zA-Z]', match_formatting, text)
+ return re.sub(r'%[a-zA-Z]', match_formatting, text)
def get_locale_key(string, numeric=False):
diff --git a/openlp/core/common/registry.py b/openlp/core/common/registry.py
index 10992d6fe..7c30ddc0d 100644
--- a/openlp/core/common/registry.py
+++ b/openlp/core/common/registry.py
@@ -23,29 +23,19 @@
Provide Registry Services
"""
import logging
-import sys
-from openlp.core.common import de_hump, trace_error_handler
+from openlp.core.common import Singleton, de_hump, trace_error_handler
log = logging.getLogger(__name__)
-class Registry(object):
+class Registry(metaclass=Singleton):
"""
This is the Component Registry. It is a singleton object and is used to provide a look up service for common
objects.
"""
log.info('Registry loaded')
- __instance__ = None
-
- def __new__(cls):
- """
- Re-implement the __new__ method to make sure we create a true singleton.
- """
- if not cls.__instance__:
- cls.__instance__ = object.__new__(cls)
- return cls.__instance__
@classmethod
def create(cls):
@@ -57,20 +47,9 @@ class Registry(object):
registry.service_list = {}
registry.functions_list = {}
registry.working_flags = {}
- # Allow the tests to remove Registry entries but not the live system
- registry.running_under_test = 'nose' in sys.argv[0] or 'pytest' in sys.argv[0]
registry.initialising = True
return registry
- @classmethod
- def destroy(cls):
- """
- Destroy the Registry.
- """
- if cls.__instance__.running_under_test:
- del cls.__instance__
- cls.__instance__ = None
-
def get(self, key):
"""
Extracts the registry value from the list based on the key passed in
diff --git a/openlp/core/common/settings.py b/openlp/core/common/settings.py
index 6fbb8a8ed..bbe0b731a 100644
--- a/openlp/core/common/settings.py
+++ b/openlp/core/common/settings.py
@@ -90,12 +90,13 @@ def upgrade_screens(number, x_position, y_position, height, width, can_override,
number: {
'number': number,
geometry_key: {
- 'x': x_position,
- 'y': y_position,
- 'height': height,
- 'width': width
+ 'x': int(x_position),
+ 'y': int(y_position),
+ 'height': int(height),
+ 'width': int(width)
},
- 'is_display': is_display_screen
+ 'is_display': is_display_screen,
+ 'is_primary': can_override
}
}
@@ -309,7 +310,7 @@ class Settings(QtCore.QSettings):
('songuasge/db hostname', 'songusage/db hostname', []),
('songuasge/db database', 'songusage/db database', []),
('presentations / Powerpoint Viewer', '', []),
- (['core/monitor', 'core/x position', 'core/y position', 'core/height', 'core/width', 'core/override',
+ (['core/monitor', 'core/x position', 'core/y position', 'core/height', 'core/width', 'core/override position',
'core/display on monitor'], 'core/screens', [(upgrade_screens, [1, 0, 0, None, None, False, False])]),
('bibles/proxy name', '', []), # Just remove these bible proxy settings. They weren't used in 2.4!
('bibles/proxy address', '', []),
diff --git a/openlp/core/display/html/display.js b/openlp/core/display/html/display.js
index e198eeee6..56e3c0a48 100644
--- a/openlp/core/display/html/display.js
+++ b/openlp/core/display/html/display.js
@@ -281,8 +281,9 @@ var Display = {
* Checks if the present slide content fits within the slide
*/
doesContentFit: function () {
- console.debug("scrollHeight: " + $(".slides")[0].scrollHeight + ", clientHeight: " + $(".slides")[0].clientHeight);
- return $(".slides")[0].clientHeight >= $(".slides")[0].scrollHeight;
+ var currSlide = $(".slides")[0];
+ console.debug("scrollHeight: " + currSlide.scrollHeight + ", clientHeight: " + currSlide.clientHeight);
+ return currSlide.clientHeight >= currSlide.scrollHeight;
},
/**
* Generate the OpenLP startup splashscreen
@@ -333,7 +334,7 @@ var Display = {
/**
* Set fullscreen image from base64 data
* @param {string} bg_color - The background color
- * @param {string} image - Path to the image
+ * @param {string} image_data - base64 encoded image data
*/
setFullscreenImageFromData: function(bg_color, image_data) {
Display.clearSlides();
@@ -372,7 +373,6 @@ var Display = {
* @param {string} verse - The verse number, e.g. "v1"
* @param {string} text - The HTML for the verse, e.g. "line1
line2"
* @param {string} footer_text - The HTML for the footer"
- * @param {bool} [reinit=true] - Re-initialize Reveal. Defaults to true.
*/
addTextSlide: function (verse, text, footer_text) {
var html = _prepareText(text);
@@ -476,25 +476,28 @@ var Display = {
* Play a video
*/
playVideo: function () {
- if ($("#video").length == 1) {
- $("#video")[0].play();
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].play();
}
},
/**
* Pause a video
*/
pauseVideo: function () {
- if ($("#video").length == 1) {
- $("#video")[0].pause();
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].pause();
}
},
/**
* Stop a video
*/
stopVideo: function () {
- if ($("#video").length == 1) {
- $("#video")[0].pause();
- $("#video")[0].currentTime = 0.0;
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].pause();
+ videoElem[0].currentTime = 0.0;
}
},
/**
@@ -502,8 +505,9 @@ var Display = {
* @param seconds The position in seconds to seek to
*/
seekVideo: function (seconds) {
- if ($("#video").length == 1) {
- $("#video")[0].currentTime = seconds;
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].currentTime = seconds;
}
},
/**
@@ -511,8 +515,9 @@ var Display = {
* @param rate A Double of the rate. 1.0 => 100% speed, 0.75 => 75% speed, 1.25 => 125% speed, etc.
*/
setPlaybackRate: function (rate) {
- if ($("#video").length == 1) {
- $("#video")[0].playbackRate = rate;
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].playbackRate = rate;
}
},
/**
@@ -520,24 +525,27 @@ var Display = {
* @param level The volume level from 0 to 100.
*/
setVideoVolume: function (level) {
- if ($("#video").length == 1) {
- $("#video")[0].volume = level / 100.0;
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].volume = level / 100.0;
}
},
/**
* Mute the volume
*/
toggleVideoMute: function () {
- if ($("#video").length == 1) {
- $("#video")[0].muted = !$("#video")[0].muted;
+ var videoElem = $("#video");
+ if (videoElem.length == 1) {
+ videoElem[0].muted = !videoElem[0].muted;
}
},
/**
* Clear the background audio playlist
*/
clearPlaylist: function () {
- if ($("#background-audio").length == 1) {
- var audio = $("#background-audio")[0];
+ var backgroundAudoElem = $("#background-audio");
+ if (backgroundAudoElem.length == 1) {
+ var audio = backgroundAudoElem[0];
/* audio.playList */
}
},
@@ -619,7 +627,6 @@ var Display = {
},
setTheme: function (theme) {
this._theme = theme;
- var slidesDiv = $(".slides")
// Set the background
var globalBackground = $("#global-background")[0];
var backgroundStyle = {};
diff --git a/openlp/core/display/render.py b/openlp/core/display/render.py
index 1097b5ac0..5e41c8174 100644
--- a/openlp/core/display/render.py
+++ b/openlp/core/display/render.py
@@ -47,7 +47,7 @@ log = logging.getLogger(__name__)
SLIM_CHARS = 'fiíIÍjlĺľrtť.,;/ ()|"\'!:\\'
CHORD_LINE_MATCH = re.compile(r'\[(.*?)\]([\u0080-\uFFFF,\w]*)'
- r'([\u0080-\uFFFF,\w,\s,\.,\,,\!,\?,\;,\:,\|,\",\',\-,\_]*)(\Z)?')
+ r'([\u0080-\uFFFF\w\s\.\,\!\?\;\:\|\"\'\-\_]*)(\Z)?')
CHORD_TEMPLATE = '{chord}'
FIRST_CHORD_TEMPLATE = '{chord}'
CHORD_LINE_TEMPLATE = '{chord}{tail}{whitespace}{remainder}'
@@ -482,6 +482,7 @@ class ThemePreviewRenderer(LogMixin, DisplayWindow):
:param theme_data: The theme to generated a preview for.
:param force_page: Flag to tell message lines per page need to be generated.
+ :param generate_screenshot: Do I need to generate a screen shot?
:rtype: QtGui.QPixmap
"""
# save value for use in format_slide
diff --git a/openlp/core/display/screens.py b/openlp/core/display/screens.py
index fccb9741f..ef9af023c 100644
--- a/openlp/core/display/screens.py
+++ b/openlp/core/display/screens.py
@@ -28,6 +28,7 @@ from functools import cmp_to_key
from PyQt5 import QtCore, QtWidgets
+from openlp.core.common import Singleton
from openlp.core.common.i18n import translate
from openlp.core.common.registry import Registry
from openlp.core.common.settings import Settings
@@ -133,8 +134,13 @@ class Screen(object):
self.number = int(screen_dict['number'])
self.is_display = screen_dict['is_display']
self.is_primary = screen_dict['is_primary']
- self.geometry = QtCore.QRect(screen_dict['geometry']['x'], screen_dict['geometry']['y'],
- screen_dict['geometry']['width'], screen_dict['geometry']['height'])
+ try:
+ self.geometry = QtCore.QRect(screen_dict['geometry']['x'], screen_dict['geometry']['y'],
+ screen_dict['geometry']['width'], screen_dict['geometry']['height'])
+ except KeyError:
+ # Preserve the current values as this has come from the settings update which does not have
+ # the geometry information
+ pass
if 'custom_geometry' in screen_dict:
self.custom_geometry = QtCore.QRect(screen_dict['custom_geometry']['x'],
screen_dict['custom_geometry']['y'],
@@ -142,24 +148,15 @@ class Screen(object):
screen_dict['custom_geometry']['height'])
-class ScreenList(object):
+class ScreenList(metaclass=Singleton):
"""
Wrapper to handle the parameters of the display screen.
To get access to the screen list call ``ScreenList()``.
"""
log.info('Screen loaded')
- __instance__ = None
screens = []
- def __new__(cls):
- """
- Re-implement __new__ to create a true singleton.
- """
- if not cls.__instance__:
- cls.__instance__ = object.__new__(cls)
- return cls.__instance__
-
def __iter__(self):
"""
Convert this object into an iterable, so that we can iterate over it instead of the inner list
diff --git a/openlp/core/lib/db.py b/openlp/core/lib/db.py
index a0445730f..e2ef2e33e 100644
--- a/openlp/core/lib/db.py
+++ b/openlp/core/lib/db.py
@@ -265,7 +265,7 @@ def upgrade_db(url, upgrade):
"""
if not database_exists(url):
log.warning("Database {db} doesn't exist - skipping upgrade checks".format(db=url))
- return (0, 0)
+ return 0, 0
log.debug('Checking upgrades for DB {db}'.format(db=url))
diff --git a/openlp/core/lib/formattingtags.py b/openlp/core/lib/formattingtags.py
index 560cbb5f2..dc58cafd0 100644
--- a/openlp/core/lib/formattingtags.py
+++ b/openlp/core/lib/formattingtags.py
@@ -59,97 +59,98 @@ class FormattingTags(object):
"""
temporary_tags = [tag for tag in FormattingTags.html_expands if tag.get('temporary')]
FormattingTags.html_expands = []
- base_tags = []
+ base_tags = [
+ {
+ 'desc': translate('OpenLP.FormattingTags', 'Red'),
+ 'start tag': '{r}',
+ 'start html': '',
+ 'end tag': '{/r}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Black'),
+ 'start tag': '{b}',
+ 'start html': '',
+ 'end tag': '{/b}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Blue'),
+ 'start tag': '{bl}',
+ 'start html': '',
+ 'end tag': '{/bl}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Yellow'),
+ 'start tag': '{y}',
+ 'start html': '',
+ 'end tag': '{/y}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Green'),
+ 'start tag': '{g}',
+ 'start html': '',
+ 'end tag': '{/g}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Pink'),
+ 'start tag': '{pk}',
+ 'start html': '',
+ 'end tag': '{/pk}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Orange'),
+ 'start tag': '{o}',
+ 'start html': '',
+ 'end tag': '{/o}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Purple'),
+ 'start tag': '{pp}',
+ 'start html': '',
+ 'end tag': '{/pp}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'White'),
+ 'start tag': '{w}',
+ 'start html': '',
+ 'end tag': '{/w}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Superscript'),
+ 'start tag': '{su}', 'start html': '',
+ 'end tag': '{/su}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Subscript'),
+ 'start tag': '{sb}', 'start html': '',
+ 'end tag': '{/sb}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Paragraph'),
+ 'start tag': '{p}', 'start html': '
', 'end tag': '{/p}',
+ 'end html': '
', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Bold'),
+ 'start tag': '{st}', 'start html': '',
+ 'end tag': '{/st}', 'end html': '',
+ 'protected': True, 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Italics'),
+ 'start tag': '{it}', 'start html': '', 'end tag': '{/it}',
+ 'end html': '', 'protected': True, 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Underline'),
+ 'start tag': '{u}',
+ 'start html': '',
+ 'end tag': '{/u}', 'end html': '', 'protected': True,
+ 'temporary': False
+ }, {
+ 'desc': translate('OpenLP.FormattingTags', 'Break'),
+ 'start tag': '{br}', 'start html': '
', 'end tag': '',
+ 'end html': '', 'protected': True,
+ 'temporary': False
+ }]
# Append the base tags.
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Red'),
- 'start tag': '{r}',
- 'start html': '',
- 'end tag': '{/r}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Black'),
- 'start tag': '{b}',
- 'start html': '',
- 'end tag': '{/b}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Blue'),
- 'start tag': '{bl}',
- 'start html': '',
- 'end tag': '{/bl}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Yellow'),
- 'start tag': '{y}',
- 'start html': '',
- 'end tag': '{/y}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Green'),
- 'start tag': '{g}',
- 'start html': '',
- 'end tag': '{/g}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Pink'),
- 'start tag': '{pk}',
- 'start html': '',
- 'end tag': '{/pk}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Orange'),
- 'start tag': '{o}',
- 'start html': '',
- 'end tag': '{/o}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Purple'),
- 'start tag': '{pp}',
- 'start html': '',
- 'end tag': '{/pp}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'White'),
- 'start tag': '{w}',
- 'start html': '',
- 'end tag': '{/w}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Superscript'),
- 'start tag': '{su}', 'start html': '',
- 'end tag': '{/su}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Subscript'),
- 'start tag': '{sb}', 'start html': '',
- 'end tag': '{/sb}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Paragraph'),
- 'start tag': '{p}', 'start html': '', 'end tag': '{/p}',
- 'end html': '
', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Bold'),
- 'start tag': '{st}', 'start html': '',
- 'end tag': '{/st}', 'end html': '',
- 'protected': True, 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Italics'),
- 'start tag': '{it}', 'start html': '', 'end tag': '{/it}',
- 'end html': '', 'protected': True, 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Underline'),
- 'start tag': '{u}',
- 'start html': '',
- 'end tag': '{/u}', 'end html': '', 'protected': True,
- 'temporary': False})
- base_tags.append({
- 'desc': translate('OpenLP.FormattingTags', 'Break'),
- 'start tag': '{br}', 'start html': '
', 'end tag': '',
- 'end html': '', 'protected': True,
- 'temporary': False})
FormattingTags.add_html_tags(base_tags)
FormattingTags.add_html_tags(temporary_tags)
user_expands_string = str(Settings().value('formattingTags/html_tags'))
diff --git a/openlp/core/lib/theme.py b/openlp/core/lib/theme.py
index bc5ba69b8..1a1f7f1cb 100644
--- a/openlp/core/lib/theme.py
+++ b/openlp/core/lib/theme.py
@@ -170,6 +170,7 @@ class Theme(object):
jsn = get_text_file_string(json_path)
self.load_theme(jsn)
self.background_filename = None
+ self.version = 2
def expand_json(self, var, prev=None):
"""
diff --git a/openlp/core/projectors/manager.py b/openlp/core/projectors/manager.py
index 02b1c71f2..91b604b14 100644
--- a/openlp/core/projectors/manager.py
+++ b/openlp/core/projectors/manager.py
@@ -344,14 +344,14 @@ class ProjectorManager(QtWidgets.QWidget, RegistryBase, UiProjectorManager, LogM
"""
log.debug('Checking for UDP port {port} listener deletion'.format(port=port))
if port not in self.pjlink_udp:
- log.warn('UDP listener for port {port} not there - skipping delete'.format(port=port))
+ log.warning('UDP listener for port {port} not there - skipping delete'.format(port=port))
return
keep_port = False
for item in self.projector_list:
if port == item.link.port:
keep_port = True
if keep_port:
- log.warn('UDP listener for port {port} needed for other projectors - skipping delete'.format(port=port))
+ log.warning('UDP listener for port {port} needed for other projectors - skipping delete'.format(port=port))
return
Registry().execute('udp_broadcast_remove', port=port)
del self.pjlink_udp[port]
diff --git a/openlp/core/projectors/pjlink.py b/openlp/core/projectors/pjlink.py
index 5ca3d9ba3..403b96b7c 100644
--- a/openlp/core/projectors/pjlink.py
+++ b/openlp/core/projectors/pjlink.py
@@ -552,7 +552,7 @@ class PJLink(QtNetwork.QTcpSocket):
data = data_in.strip()
self.receive_data_signal()
# Initial packet checks
- if (len(data) < 7):
+ if len(data) < 7:
self._trash_buffer(msg='get_data(): Invalid packet - length')
return
elif len(data) > self.max_size:
diff --git a/openlp/core/projectors/sourceselectform.py b/openlp/core/projectors/sourceselectform.py
index 1d40436a2..f34439d33 100644
--- a/openlp/core/projectors/sourceselectform.py
+++ b/openlp/core/projectors/sourceselectform.py
@@ -177,7 +177,7 @@ class FingerTabBarWidget(QtWidgets.QTabBar):
:param height: Remove default height parameter in kwargs
"""
self.tabSize = QtCore.QSize(kwargs.pop('width', 100), kwargs.pop('height', 25))
- QtWidgets.QTabBar.__init__(self, parent, *args, **kwargs)
+ super().__init__(parent)
def paintEvent(self, event):
"""
@@ -215,11 +215,11 @@ class FingerTabWidget(QtWidgets.QTabWidget):
Based on thread discussion
http://www.riverbankcomputing.com/pipermail/pyqt/2005-December/011724.html
"""
- def __init__(self, parent, *args):
+ def __init__(self, parent):
"""
Initialize FingerTabWidget instance
"""
- QtWidgets.QTabWidget.__init__(self, parent, *args)
+ super().__init__(parent)
self.setTabBar(FingerTabBarWidget(self))
diff --git a/openlp/core/state.py b/openlp/core/state.py
index c76644a57..b0474c8c2 100644
--- a/openlp/core/state.py
+++ b/openlp/core/state.py
@@ -28,6 +28,7 @@ contained within the openlp.core module.
"""
import logging
+from openlp.core.common import Singleton
from openlp.core.common.registry import Registry
from openlp.core.common.mixins import LogMixin
from openlp.core.lib.plugin import PluginStatus
@@ -52,17 +53,7 @@ class StateModule(LogMixin):
self.text = None
-class State(LogMixin):
-
- __instance__ = None
-
- def __new__(cls):
- """
- Re-implement the __new__ method to make sure we create a true singleton.
- """
- if not cls.__instance__:
- cls.__instance__ = object.__new__(cls)
- return cls.__instance__
+class State(LogMixin, metaclass=Singleton):
def load_settings(self):
self.modules = {}
diff --git a/openlp/core/threading.py b/openlp/core/threading.py
index 9a4043292..38c68638a 100644
--- a/openlp/core/threading.py
+++ b/openlp/core/threading.py
@@ -24,10 +24,11 @@ The :mod:`openlp.core.threading` module contains some common threading code
"""
from PyQt5 import QtCore
+from openlp.core.common.mixins import LogMixin
from openlp.core.common.registry import Registry
-class ThreadWorker(QtCore.QObject):
+class ThreadWorker(QtCore.QObject, LogMixin):
"""
The :class:`~openlp.core.threading.ThreadWorker` class provides a base class for all worker objects
"""
diff --git a/openlp/core/ui/__init__.py b/openlp/core/ui/__init__.py
index 220bda7ce..2e9a7a48c 100644
--- a/openlp/core/ui/__init__.py
+++ b/openlp/core/ui/__init__.py
@@ -93,4 +93,4 @@ class SingleColumnTableWidget(QtWidgets.QTableWidget):
self.resizeRowsToContents()
-__all__ = ['SingleColumnTableWidget', 'DisplayControllerType']
+__all__ = ['AlertLocation', 'DisplayControllerType', 'HideMode', 'SingleColumnTableWidget']
diff --git a/openlp/core/ui/advancedtab.py b/openlp/core/ui/advancedtab.py
index 72b9133a0..38542e59f 100644
--- a/openlp/core/ui/advancedtab.py
+++ b/openlp/core/ui/advancedtab.py
@@ -81,7 +81,7 @@ class AdvancedTab(SettingsTab):
self.ui_layout.addRow(self.media_plugin_check_box)
self.hide_mouse_check_box = QtWidgets.QCheckBox(self.ui_group_box)
self.hide_mouse_check_box.setObjectName('hide_mouse_check_box')
- self.ui_layout.addWidget(self.hide_mouse_check_box)
+ self.ui_layout.addRow(self.hide_mouse_check_box)
self.double_click_live_check_box = QtWidgets.QCheckBox(self.ui_group_box)
self.double_click_live_check_box.setObjectName('double_click_live_check_box')
self.ui_layout.addRow(self.double_click_live_check_box)
diff --git a/openlp/core/ui/formattingtagcontroller.py b/openlp/core/ui/formattingtagcontroller.py
index 5b26851b5..0b88702aa 100644
--- a/openlp/core/ui/formattingtagcontroller.py
+++ b/openlp/core/ui/formattingtagcontroller.py
@@ -40,7 +40,7 @@ class FormattingTagController(object):
"""
self.html_tag_regex = re.compile(
r'<(?:(?P/(?=[^\s/>]+>))?'
- r'(?P[^\s/!\?>]+)(?:\s+[^\s=]+="[^"]*")*\s*(?P/)?'
+ r'(?P[^\s/!?>]+)(?:\s+[^\s=]+="[^"]*")*\s*(?P/)?'
r'|(?P!\[CDATA\[(?:(?!\]\]>).)*\]\])'
r'|(?P\?(?:(?!\?>).)*\?)'
r'|(?P!--(?:(?!-->).)*--))>')
diff --git a/openlp/core/ui/icons.py b/openlp/core/ui/icons.py
index 925becf1f..b70d10996 100644
--- a/openlp/core/ui/icons.py
+++ b/openlp/core/ui/icons.py
@@ -27,6 +27,7 @@ import logging
import qtawesome as qta
from PyQt5 import QtGui, QtWidgets
+from openlp.core.common import Singleton
from openlp.core.common.applocation import AppLocation
from openlp.core.lib import build_icon
@@ -34,22 +35,11 @@ from openlp.core.lib import build_icon
log = logging.getLogger(__name__)
-class UiIcons(object):
+class UiIcons(metaclass=Singleton):
"""
Provide standard icons for objects to use.
"""
- __instance__ = None
-
- def __new__(cls):
- """
- Override the default object creation method to return a single instance.
- """
- if not cls.__instance__:
- cls.__instance__ = object.__new__(cls)
- cls.load(cls)
- return cls.__instance__
-
- def load(self):
+ def __init__(self):
"""
These are the font icons used in the code.
"""
@@ -164,7 +154,8 @@ class UiIcons(object):
'video': {'icon': 'fa.file-video-o'},
'volunteer': {'icon': 'fa.group'}
}
- self.load_icons(self, icon_list)
+ self.load_icons(icon_list)
+ self.main_icon = build_icon(':/icon/openlp-logo.svg')
def load_icons(self, icon_list):
"""
@@ -184,7 +175,6 @@ class UiIcons(object):
setattr(self, key, qta.icon('fa.plus-circle', color='red'))
except Exception:
setattr(self, key, qta.icon('fa.plus-circle', color='red'))
- self.main_icon = build_icon(':/icon/openlp-logo.svg')
@staticmethod
def _print_icons():
diff --git a/openlp/core/ui/mainwindow.py b/openlp/core/ui/mainwindow.py
index 2313e4705..da8b72856 100644
--- a/openlp/core/ui/mainwindow.py
+++ b/openlp/core/ui/mainwindow.py
@@ -635,7 +635,10 @@ class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow, LogMixin, RegistryPropert
# if self.live_controller.display.isVisible():
# self.live_controller.display.setFocus()
self.activateWindow()
- if self.application.args:
+ # We have -disable-web-security added by our code.
+ # If a file is passed in we will have that as well so count of 2
+ # If not we need to see if we want to use the previous file.so count of 1
+ if self.application.args and len(self.application.args) > 1:
self.open_cmd_line_files(self.application.args)
elif Settings().value(self.general_settings_section + '/auto open'):
self.service_manager_contents.load_last_file()
diff --git a/openlp/core/ui/printserviceform.py b/openlp/core/ui/printserviceform.py
index 0f8ea064b..e81f82f9a 100644
--- a/openlp/core/ui/printserviceform.py
+++ b/openlp/core/ui/printserviceform.py
@@ -219,13 +219,13 @@ class PrintServiceForm(QtWidgets.QDialog, Ui_PrintServiceDialog, RegistryPropert
verse_def = None
verse_html = None
for slide in item.get_frames():
- if not verse_def or verse_def != slide['verseTag'] or verse_html == slide['printing_html']:
+ if not verse_def or verse_def != slide['verse'] or verse_html == slide['text']:
text_div = self._add_element('div', parent=div, class_id='itemText')
- elif 'chordspacing' not in slide['printing_html']:
+ elif 'chordspacing' not in slide['text']:
self._add_element('br', parent=text_div)
- self._add_element('span', slide['printing_html'], text_div)
- verse_def = slide['verseTag']
- verse_html = slide['printing_html']
+ self._add_element('span', slide['text'], text_div)
+ verse_def = slide['verse']
+ verse_html = slide['text']
# Break the page before the div element.
if index != 0 and self.page_break_after_text.isChecked():
div.set('class', 'item newPage')
diff --git a/openlp/core/ui/servicemanager.py b/openlp/core/ui/servicemanager.py
index 92b34badd..202a926dd 100644
--- a/openlp/core/ui/servicemanager.py
+++ b/openlp/core/ui/servicemanager.py
@@ -34,6 +34,7 @@ from tempfile import NamedTemporaryFile
from PyQt5 import QtCore, QtGui, QtWidgets
+from openlp.core.state import State
from openlp.core.common import ThemeLevel, delete_file
from openlp.core.common.actions import ActionList, CategoryOrder
from openlp.core.common.applocation import AppLocation
@@ -828,7 +829,7 @@ class ServiceManager(QtWidgets.QWidget, RegistryBase, Ui_ServiceManager, LogMixi
self.auto_start_action.setIcon(UiIcons().inactive)
self.auto_start_action.setText(translate('OpenLP.ServiceManager', '&Auto Start - inactive'))
if service_item['service_item'].is_text():
- for plugin in self.plugin_manager.plugins:
+ for plugin in State().list_plugins():
if plugin.name == 'custom' and plugin.status == PluginStatus.Active:
self.create_custom_action.setVisible(True)
break
diff --git a/openlp/core/ui/shortcutlistdialog.py b/openlp/core/ui/shortcutlistdialog.py
index efc795501..a6440b63f 100644
--- a/openlp/core/ui/shortcutlistdialog.py
+++ b/openlp/core/ui/shortcutlistdialog.py
@@ -101,7 +101,7 @@ class Ui_ShortcutListDialog(object):
self.primary_push_button = CaptureShortcutButton(shortcutListDialog)
self.primary_push_button.setObjectName('primary_push_button')
self.primary_push_button.setMinimumSize(QtCore.QSize(84, 0))
- self.primary_push_button.setIcon(UiIcons.shortcuts)
+ self.primary_push_button.setIcon(UiIcons().shortcuts)
self.primary_layout.addWidget(self.primary_push_button)
self.clear_primary_button = QtWidgets.QToolButton(shortcutListDialog)
self.clear_primary_button.setObjectName('clear_primary_button')
diff --git a/openlp/core/ui/shortcutlistform.py b/openlp/core/ui/shortcutlistform.py
index 360c631c0..6d37130ab 100644
--- a/openlp/core/ui/shortcutlistform.py
+++ b/openlp/core/ui/shortcutlistform.py
@@ -34,7 +34,7 @@ from openlp.core.common.settings import Settings
from openlp.core.ui.shortcutlistdialog import Ui_ShortcutListDialog
-REMOVE_AMPERSAND = re.compile(r'&{1}')
+REMOVE_AMPERSAND = re.compile(r'&')
log = logging.getLogger(__name__)
diff --git a/openlp/core/ui/thememanager.py b/openlp/core/ui/thememanager.py
index 4a3aa5073..06a76372d 100644
--- a/openlp/core/ui/thememanager.py
+++ b/openlp/core/ui/thememanager.py
@@ -30,6 +30,7 @@ from xml.etree.ElementTree import XML, ElementTree
from PyQt5 import QtCore, QtWidgets
+from openlp.core.state import State
from openlp.core.common import delete_file
from openlp.core.common.applocation import AppLocation
from openlp.core.common.i18n import UiStrings, get_locale_key, translate
@@ -293,7 +294,7 @@ class ThemeManager(QtWidgets.QWidget, RegistryBase, Ui_ThemeManager, LogMixin, R
old_theme_data = self.get_theme_data(old_theme_name)
self.clone_theme_data(old_theme_data, new_theme_name)
self.delete_theme(old_theme_name)
- for plugin in self.plugin_manager.plugins:
+ for plugin in State().list_plugins():
if plugin.uses_theme(old_theme_name):
plugin.rename_theme(old_theme_name, new_theme_name)
self.renderer.set_theme(self.get_theme_data(new_theme_name))
@@ -771,7 +772,7 @@ class ThemeManager(QtWidgets.QWidget, RegistryBase, Ui_ThemeManager, LogMixin, R
# check for use in the system else where.
if test_plugin:
plugin_usage = ""
- for plugin in self.plugin_manager.plugins:
+ for plugin in State().list_plugins():
used_count = plugin.uses_theme(theme)
if used_count:
plugin_usage = "{plug}{text}".format(plug=plugin_usage,
diff --git a/openlp/core/widgets/views.py b/openlp/core/widgets/views.py
index b5c07993c..c3f4be0ca 100644
--- a/openlp/core/widgets/views.py
+++ b/openlp/core/widgets/views.py
@@ -80,7 +80,7 @@ class ListPreviewWidget(QtWidgets.QTableWidget, RegistryProperties):
An empty ``ServiceItem`` is used by default. replace_service_manager_item() needs to be called to make this
widget display something.
"""
- super(QtWidgets.QTableWidget, self).__init__(parent)
+ super().__init__(parent)
self._setup(screen_ratio)
def _setup(self, screen_ratio):
@@ -124,7 +124,7 @@ class ListPreviewWidget(QtWidgets.QTableWidget, RegistryProperties):
max_img_row_height = Settings().value('advanced/slide max height')
# Adjust for row height cap if in use.
if isinstance(max_img_row_height, int):
- if max_img_row_height > 0 and height > max_img_row_height:
+ if 0 < max_img_row_height < height:
height = max_img_row_height
elif max_img_row_height < 0:
# If auto setting, show that number of slides, or if the resulting slides too small, 100px.
@@ -219,7 +219,7 @@ class ListPreviewWidget(QtWidgets.QTableWidget, RegistryProperties):
slide_height = width // self.screen_ratio
max_img_row_height = Settings().value('advanced/slide max height')
if isinstance(max_img_row_height, int):
- if max_img_row_height > 0 and slide_height > max_img_row_height:
+ if 0 < max_img_row_height < slide_height:
slide_height = max_img_row_height
elif max_img_row_height < 0:
# If auto setting, show that number of slides, or if the resulting slides too small, 100px.
diff --git a/openlp/core/widgets/widgets.py b/openlp/core/widgets/widgets.py
index 0415d73a5..42f83511d 100644
--- a/openlp/core/widgets/widgets.py
+++ b/openlp/core/widgets/widgets.py
@@ -194,11 +194,11 @@ class ScreenButton(QtWidgets.QPushButton):
class ScreenSelectionWidget(QtWidgets.QWidget):
- def __init__(self, parent=None, screens=[]):
+ def __init__(self, parent=None, screens=None):
super().__init__(parent)
self.current_screen = None
self.identify_labels = []
- self.screens = screens
+ self.screens = screens or []
self.timer = QtCore.QTimer()
self.timer.setSingleShot(True)
self.timer.setInterval(3000)
diff --git a/openlp/plugins/bibles/lib/__init__.py b/openlp/plugins/bibles/lib/__init__.py
index e0a06fa2d..b0d5189e1 100644
--- a/openlp/plugins/bibles/lib/__init__.py
+++ b/openlp/plugins/bibles/lib/__init__.py
@@ -26,6 +26,7 @@ plugin.
import logging
import re
+from openlp.core.common import Singleton
from openlp.core.common.i18n import translate
from openlp.core.common.settings import Settings
@@ -64,20 +65,10 @@ class LanguageSelection(object):
English = 2
-class BibleStrings(object):
+class BibleStrings(metaclass=Singleton):
"""
Provide standard strings for objects to use.
"""
- __instance__ = None
-
- def __new__(cls):
- """
- Override the default object creation method to return a single instance.
- """
- if not cls.__instance__:
- cls.__instance__ = object.__new__(cls)
- return cls.__instance__
-
def __init__(self):
"""
These strings should need a good reason to be retranslated elsewhere.
@@ -336,11 +327,13 @@ def parse_reference(reference, bible, language_selection, book_ref_id=False):
log.debug('Matched reference {text}'.format(text=reference))
book = match.group('book')
if not book_ref_id:
- book_ref_id = bible.get_book_ref_id_by_localised_name(book, language_selection)
+ book_ref_ids = bible.get_book_ref_id_by_localised_name(book, language_selection)
elif not bible.get_book_by_book_ref_id(book_ref_id):
return []
+ else:
+ book_ref_ids = [book_ref_id]
# We have not found the book so do not continue
- if not book_ref_id:
+ if not book_ref_ids:
return []
ranges = match.group('ranges')
range_list = get_reference_match('range_separator').split(ranges)
@@ -381,22 +374,23 @@ def parse_reference(reference, bible, language_selection, book_ref_id=False):
to_chapter = to_verse
to_verse = None
# Append references to the list
- if has_range:
- if not from_verse:
- from_verse = 1
- if not to_verse:
- to_verse = -1
- if to_chapter and to_chapter > from_chapter:
- ref_list.append((book_ref_id, from_chapter, from_verse, -1))
- for i in range(from_chapter + 1, to_chapter):
- ref_list.append((book_ref_id, i, 1, -1))
- ref_list.append((book_ref_id, to_chapter, 1, to_verse))
- elif to_verse >= from_verse or to_verse == -1:
- ref_list.append((book_ref_id, from_chapter, from_verse, to_verse))
- elif from_verse:
- ref_list.append((book_ref_id, from_chapter, from_verse, from_verse))
- else:
- ref_list.append((book_ref_id, from_chapter, 1, -1))
+ for book_ref_id in book_ref_ids:
+ if has_range:
+ if not from_verse:
+ from_verse = 1
+ if not to_verse:
+ to_verse = -1
+ if to_chapter and to_chapter > from_chapter:
+ ref_list.append((book_ref_id, from_chapter, from_verse, -1))
+ for i in range(from_chapter + 1, to_chapter):
+ ref_list.append((book_ref_id, i, 1, -1))
+ ref_list.append((book_ref_id, to_chapter, 1, to_verse))
+ elif to_verse >= from_verse or to_verse == -1:
+ ref_list.append((book_ref_id, from_chapter, from_verse, to_verse))
+ elif from_verse:
+ ref_list.append((book_ref_id, from_chapter, from_verse, from_verse))
+ else:
+ ref_list.append((book_ref_id, from_chapter, 1, -1))
return ref_list
else:
log.debug('Invalid reference: {text}'.format(text=reference))
diff --git a/openlp/plugins/bibles/lib/db.py b/openlp/plugins/bibles/lib/db.py
index 3f48ca7b2..5a629d14e 100644
--- a/openlp/plugins/bibles/lib/db.py
+++ b/openlp/plugins/bibles/lib/db.py
@@ -281,13 +281,14 @@ class BibleDB(Manager):
log.debug('BibleDB.get_book("{book}")'.format(book=book))
return self.get_object_filtered(Book, Book.name.like(book + '%'))
- def get_books(self):
+ def get_books(self, book=None):
"""
A wrapper so both local and web bibles have a get_books() method that
manager can call. Used in the media manager advanced search tab.
"""
- log.debug('BibleDB.get_books()')
- return self.get_all_objects(Book, order_by_ref=Book.id)
+ log.debug('BibleDB.get_books("{book}")'.format(book=book))
+ filter = Book.name.like(book + '%') if book else None
+ return self.get_all_objects(Book, filter_clause=filter, order_by_ref=Book.id)
def get_book_by_book_ref_id(self, ref_id):
"""
@@ -300,39 +301,35 @@ class BibleDB(Manager):
def get_book_ref_id_by_localised_name(self, book, language_selection):
"""
- Return the id of a named book.
+ Return the ids of a matching named book.
:param book: The name of the book, according to the selected language.
:param language_selection: The language selection the user has chosen in the settings section of the Bible.
+ :rtype: list[int]
"""
log.debug('get_book_ref_id_by_localised_name("{book}", "{lang}")'.format(book=book, lang=language_selection))
from openlp.plugins.bibles.lib import LanguageSelection, BibleStrings
book_names = BibleStrings().BookNames
# escape reserved characters
- book_escaped = book
for character in RESERVED_CHARACTERS:
- book_escaped = book_escaped.replace(character, '\\' + character)
+ book_escaped = book.replace(character, '\\' + character)
regex_book = re.compile('\\s*{book}\\s*'.format(book='\\s*'.join(book_escaped.split())), re.IGNORECASE)
if language_selection == LanguageSelection.Bible:
- db_book = self.get_book(book)
- if db_book:
- return db_book.book_reference_id
- elif language_selection == LanguageSelection.Application:
- books = [key for key in list(book_names.keys()) if regex_book.match(str(book_names[key]))]
- books = [_f for _f in map(BiblesResourcesDB.get_book, books) if _f]
- for value in books:
- if self.get_book_by_book_ref_id(value['id']):
- return value['id']
- elif language_selection == LanguageSelection.English:
- books = BiblesResourcesDB.get_books_like(book)
- if books:
- book_list = [value for value in books if regex_book.match(value['name'])]
- if not book_list:
- book_list = books
- for value in book_list:
- if self.get_book_by_book_ref_id(value['id']):
- return value['id']
- return False
+ db_books = self.get_books(book)
+ return [db_book.book_reference_id for db_book in db_books]
+ else:
+ book_list = []
+ if language_selection == LanguageSelection.Application:
+ books = [key for key in list(book_names.keys()) if regex_book.match(book_names[key])]
+ book_list = [_f for _f in map(BiblesResourcesDB.get_book, books) if _f]
+ elif language_selection == LanguageSelection.English:
+ books = BiblesResourcesDB.get_books_like(book)
+ if books:
+ book_list = [value for value in books if regex_book.match(value['name'])]
+ if not book_list:
+ book_list = books
+ return [value['id'] for value in book_list if self.get_book_by_book_ref_id(value['id'])]
+ return []
def get_verses(self, reference_list, show_error=True):
"""
diff --git a/openlp/plugins/bibles/lib/manager.py b/openlp/plugins/bibles/lib/manager.py
index 79f66cef4..44d18b92d 100644
--- a/openlp/plugins/bibles/lib/manager.py
+++ b/openlp/plugins/bibles/lib/manager.py
@@ -240,8 +240,10 @@ class BibleManager(LogMixin, RegistryProperties):
book=book,
chapter=chapter))
language_selection = self.get_language_selection(bible)
- book_ref_id = self.db_cache[bible].get_book_ref_id_by_localised_name(book, language_selection)
- return self.db_cache[bible].get_verse_count(book_ref_id, chapter)
+ book_ref_ids = self.db_cache[bible].get_book_ref_id_by_localised_name(book, language_selection)
+ if book_ref_ids:
+ return self.db_cache[bible].get_verse_count(book_ref_ids[0], chapter)
+ return 0
def get_verse_count_by_book_ref_id(self, bible, book_ref_id, chapter):
"""
@@ -401,4 +403,4 @@ class BibleManager(LogMixin, RegistryProperties):
self.db_cache[bible].finalise()
-__all__ = ['BibleFormat']
+__all__ = ['BibleFormat', 'BibleManager']
diff --git a/openlp/plugins/bibles/lib/mediaitem.py b/openlp/plugins/bibles/lib/mediaitem.py
index 185a6832e..b239defdd 100755
--- a/openlp/plugins/bibles/lib/mediaitem.py
+++ b/openlp/plugins/bibles/lib/mediaitem.py
@@ -109,7 +109,7 @@ class BibleMediaItem(MediaManagerItem):
:param kwargs: Keyword arguments to pass to the super method. (dict)
"""
self.clear_icon = UiIcons().square
- self.save_results_icon = UiIcons.save
+ self.save_results_icon = UiIcons().save
self.sort_icon = UiIcons().sort
self.bible = None
self.second_bible = None
diff --git a/openlp/plugins/custom/forms/editcustomdialog.py b/openlp/plugins/custom/forms/editcustomdialog.py
index 6193cd017..470eccba7 100644
--- a/openlp/plugins/custom/forms/editcustomdialog.py
+++ b/openlp/plugins/custom/forms/editcustomdialog.py
@@ -97,6 +97,7 @@ class Ui_CustomEditDialog(object):
self.preview_button = QtWidgets.QPushButton()
self.button_box = create_button_box(custom_edit_dialog, 'button_box', ['cancel', 'save'],
[self.preview_button])
+ self.save_button = self.button_box.button(QtWidgets.QDialogButtonBox.Save)
self.dialog_layout.addWidget(self.button_box)
self.retranslate_ui(custom_edit_dialog)
@@ -112,3 +113,4 @@ class Ui_CustomEditDialog(object):
self.theme_label.setText(translate('CustomPlugin.EditCustomForm', 'The&me:'))
self.credit_label.setText(translate('CustomPlugin.EditCustomForm', '&Credits:'))
self.preview_button.setText(UiStrings().SaveAndPreview)
+ self.save_button.setText(UiStrings().SaveAndClose)
diff --git a/openlp/plugins/custom/lib/mediaitem.py b/openlp/plugins/custom/lib/mediaitem.py
index 3db190e8d..768bef391 100644
--- a/openlp/plugins/custom/lib/mediaitem.py
+++ b/openlp/plugins/custom/lib/mediaitem.py
@@ -349,7 +349,7 @@ class CustomMediaItem(MediaManagerItem):
custom.credits = ''
custom_xml = CustomXMLBuilder()
for (idx, slide) in enumerate(item.slides):
- custom_xml.add_verse_to_lyrics('custom', str(idx + 1), slide['raw_slide'])
+ custom_xml.add_verse_to_lyrics('custom', str(idx + 1), slide['text'])
custom.text = str(custom_xml.extract_xml(), 'utf-8')
self.plugin.db_manager.save_object(custom)
self.on_search_text_button_clicked()
diff --git a/openlp/plugins/presentations/lib/impresscontroller.py b/openlp/plugins/presentations/lib/impresscontroller.py
index 873e91b86..dd8ad9df4 100644
--- a/openlp/plugins/presentations/lib/impresscontroller.py
+++ b/openlp/plugins/presentations/lib/impresscontroller.py
@@ -55,7 +55,7 @@ if is_win():
class SlideShowListenerImport(XSlideShowListenerObj.__class__):
pass
except (AttributeError, pywintypes.com_error):
- class SlideShowListenerImport():
+ class SlideShowListenerImport(object):
pass
# Declare an empty exception to match the exception imported from UNO
@@ -76,7 +76,7 @@ else:
except ImportError:
uno_available = False
- class SlideShowListenerImport():
+ class SlideShowListenerImport(object):
pass
log = logging.getLogger(__name__)
@@ -233,9 +233,7 @@ class ImpressController(PresentationController):
self.conf_provider = self.manager.createInstanceWithContext(
'com.sun.star.configuration.ConfigurationProvider', uno.getComponentContext())
# Setup lookup properties to get Impress settings
- properties = []
- properties.append(self.create_property('nodepath', 'org.openoffice.Office.Impress'))
- properties = tuple(properties)
+ properties = tuple(self.create_property('nodepath', 'org.openoffice.Office.Impress'))
try:
# Get an updateable configuration view
impress_conf_props = self.conf_provider.createInstanceWithArguments(
@@ -311,9 +309,7 @@ class ImpressDocument(PresentationDocument):
if desktop is None:
return False
self.desktop = desktop
- properties = []
- properties.append(self.controller.create_property('Hidden', True))
- properties = tuple(properties)
+ properties = tuple(self.controller.create_property('Hidden', True))
try:
self.document = desktop.loadComponentFromURL(url, '_blank', 0, properties)
except Exception:
@@ -338,9 +334,7 @@ class ImpressDocument(PresentationDocument):
return
temp_folder_path = self.get_temp_folder()
thumb_dir_url = temp_folder_path.as_uri()
- properties = []
- properties.append(self.controller.create_property('FilterName', 'impress_png_Export'))
- properties = tuple(properties)
+ properties = tuple(self.controller.create_property('FilterName', 'impress_png_Export'))
doc = self.document
pages = doc.getDrawPages()
if not pages:
diff --git a/openlp/plugins/songs/forms/editsongdialog.py b/openlp/plugins/songs/forms/editsongdialog.py
index bf5910cf3..1fa042c37 100644
--- a/openlp/plugins/songs/forms/editsongdialog.py
+++ b/openlp/plugins/songs/forms/editsongdialog.py
@@ -291,6 +291,7 @@ class Ui_EditSongDialog(object):
self.warning_label.setObjectName('warning_label')
self.bottom_layout.addWidget(self.warning_label)
self.button_box = create_button_box(edit_song_dialog, 'button_box', ['cancel', 'save'])
+ self.save_button = self.button_box.button(QtWidgets.QDialogButtonBox.Save)
self.bottom_layout.addWidget(self.button_box)
self.dialog_layout.addLayout(self.bottom_layout)
self.retranslate_ui(edit_song_dialog)
@@ -341,6 +342,7 @@ class Ui_EditSongDialog(object):
translate('SongsPlugin.EditSongForm', 'Warning: Not all of the verses are in use.')
self.no_verse_order_entered_warning = \
translate('SongsPlugin.EditSongForm', 'Warning: You have not entered a verse order.')
+ self.save_button.setText(UiStrings().SaveAndPreview)
def create_combo_box(parent, name, editable=True):
diff --git a/openlp/plugins/songs/forms/editsongform.py b/openlp/plugins/songs/forms/editsongform.py
index 62d0eb7d7..899e62122 100644
--- a/openlp/plugins/songs/forms/editsongform.py
+++ b/openlp/plugins/songs/forms/editsongform.py
@@ -29,6 +29,7 @@ from shutil import copyfile
from PyQt5 import QtCore, QtWidgets
+from openlp.core.state import State
from openlp.core.common.applocation import AppLocation
from openlp.core.common.i18n import UiStrings, get_natural_key, translate
from openlp.core.common.mixins import RegistryProperties
@@ -416,7 +417,7 @@ class EditSongForm(QtWidgets.QDialog, Ui_EditSongDialog, RegistryProperties):
Load the media files into a combobox.
"""
self.from_media_button.setVisible(False)
- for plugin in self.plugin_manager.plugins:
+ for plugin in State().list_plugins():
if plugin.name == 'media' and plugin.status == PluginStatus.Active:
self.from_media_button.setVisible(True)
self.media_form.populate_files(plugin.media_item.get_list(MediaType.Audio))
diff --git a/openlp/plugins/songs/lib/__init__.py b/openlp/plugins/songs/lib/__init__.py
index e79dbc95a..243eedcc8 100644
--- a/openlp/plugins/songs/lib/__init__.py
+++ b/openlp/plugins/songs/lib/__init__.py
@@ -615,14 +615,16 @@ def transpose_chord(chord, transpose_value, notation):
:return: The transposed chord.
"""
# See https://en.wikipedia.org/wiki/Musical_note#12-tone_chromatic_scale
- notes_sharp_notation = {}
- notes_flat_notation = {}
- notes_sharp_notation['german'] = ['C', 'C#', 'D', 'D#', 'E', 'F', 'F#', 'G', 'G#', 'A', 'A#', 'H']
- notes_flat_notation['german'] = ['C', 'Db', 'D', 'Eb', 'Fb', 'F', 'Gb', 'G', 'Ab', 'A', 'B', 'H']
- notes_sharp_notation['english'] = ['C', 'C#', 'D', 'D#', 'E', 'F', 'F#', 'G', 'G#', 'A', 'A#', 'B']
- notes_flat_notation['english'] = ['C', 'Db', 'D', 'Eb', 'Fb', 'F', 'Gb', 'G', 'Ab', 'A', 'Bb', 'B']
- notes_sharp_notation['neo-latin'] = ['Do', 'Do#', 'Re', 'Re#', 'Mi', 'Fa', 'Fa#', 'Sol', 'Sol#', 'La', 'La#', 'Si']
- notes_flat_notation['neo-latin'] = ['Do', 'Reb', 'Re', 'Mib', 'Fab', 'Fa', 'Solb', 'Sol', 'Lab', 'La', 'Sib', 'Si']
+ notes_sharp_notation = {
+ 'german': ['C', 'C#', 'D', 'D#', 'E', 'F', 'F#', 'G', 'G#', 'A', 'A#', 'H'],
+ 'english': ['C', 'C#', 'D', 'D#', 'E', 'F', 'F#', 'G', 'G#', 'A', 'A#', 'B'],
+ 'neo-latin': ['Do', 'Do#', 'Re', 'Re#', 'Mi', 'Fa', 'Fa#', 'Sol', 'Sol#', 'La', 'La#', 'Si']
+ }
+ notes_flat_notation = {
+ 'german': ['C', 'Db', 'D', 'Eb', 'Fb', 'F', 'Gb', 'G', 'Ab', 'A', 'B', 'H'],
+ 'english': ['C', 'Db', 'D', 'Eb', 'Fb', 'F', 'Gb', 'G', 'Ab', 'A', 'Bb', 'B'],
+ 'neo-latin': ['Do', 'Reb', 'Re', 'Mib', 'Fab', 'Fa', 'Solb', 'Sol', 'Lab', 'La', 'Sib', 'Si']
+ }
chord_split = chord.replace('♭', 'b').split('/')
transposed_chord = ''
last_chord = ''
diff --git a/openlp/plugins/songs/lib/db.py b/openlp/plugins/songs/lib/db.py
index 3dd3b643f..ef6641a9b 100644
--- a/openlp/plugins/songs/lib/db.py
+++ b/openlp/plugins/songs/lib/db.py
@@ -374,7 +374,9 @@ def init_schema(url):
mapper(SongBookEntry, songs_songbooks_table, properties={
'songbook': relation(Book)
})
- mapper(Book, song_books_table)
+ mapper(Book, song_books_table, properties={
+ 'songs': relation(Song, secondary=songs_songbooks_table)
+ })
mapper(MediaFile, media_files_table)
mapper(Song, songs_table, properties={
# Use the authors_songs relation when you need access to the 'author_type' attribute
diff --git a/openlp/plugins/songs/lib/importers/cclifile.py b/openlp/plugins/songs/lib/importers/cclifile.py
index 48d536265..72c2eafec 100644
--- a/openlp/plugins/songs/lib/importers/cclifile.py
+++ b/openlp/plugins/songs/lib/importers/cclifile.py
@@ -146,7 +146,9 @@ class CCLIFileImport(SongImport):
"""
log.debug('USR file text: {text}'.format(text=text_list))
song_author = ''
+ song_fields = ''
song_topics = ''
+ song_words = ''
for line in text_list:
if line.startswith('[S '):
ccli, line = line.split(']', 1)
diff --git a/openlp/plugins/songs/lib/importers/dreambeam.py b/openlp/plugins/songs/lib/importers/dreambeam.py
index e591fe379..8e76e2f0c 100644
--- a/openlp/plugins/songs/lib/importers/dreambeam.py
+++ b/openlp/plugins/songs/lib/importers/dreambeam.py
@@ -87,6 +87,7 @@ class DreamBeamImport(SongImport):
if self.stop_import_flag:
return
self.set_defaults()
+ author_copyright = ''
parser = etree.XMLParser(remove_blank_text=True)
try:
with file_path.open('r') as xml_file:
@@ -142,7 +143,7 @@ class DreamBeamImport(SongImport):
author_copyright = song_xml.Text2.Text.text
if author_copyright:
author_copyright = str(author_copyright)
- if author_copyright.find(str(SongStrings.CopyrightSymbol)) >= 0:
+ if author_copyright.find(SongStrings.CopyrightSymbol) >= 0:
self.add_copyright(author_copyright)
else:
self.parse_author(author_copyright)
diff --git a/openlp/plugins/songs/lib/importers/easyslides.py b/openlp/plugins/songs/lib/importers/easyslides.py
index 7fd5b25c1..7f03afb5a 100644
--- a/openlp/plugins/songs/lib/importers/easyslides.py
+++ b/openlp/plugins/songs/lib/importers/easyslides.py
@@ -137,9 +137,11 @@ class EasySlidesImport(SongImport):
except UnicodeDecodeError:
log.exception('Unicode decode error while decoding Contents')
self._success = False
+ return
except AttributeError:
log.exception('no Contents')
self._success = False
+ return
lines = lyrics.split('\n')
# we go over all lines first, to determine information,
# which tells us how to parse verses later
diff --git a/openlp/plugins/songs/lib/importers/easyworship.py b/openlp/plugins/songs/lib/importers/easyworship.py
index 9468031fa..eeb9feb11 100644
--- a/openlp/plugins/songs/lib/importers/easyworship.py
+++ b/openlp/plugins/songs/lib/importers/easyworship.py
@@ -118,8 +118,8 @@ class EasyWorshipSongImport(SongImport):
# 40/48/56 Entry count int32le 4 Number of items in the schedule
# 44/52/60 Entry length int16le 2 Length of schedule entries: 0x0718 = 1816
# Get file version
- type, = struct.unpack('<38s', self.ews_file.read(38))
- version = type.decode()[-3:]
+ file_type, = struct.unpack('<38s', self.ews_file.read(38))
+ version = file_type.decode()[-3:]
# Set fileposition based on filetype/version
file_pos = 0
if version == ' 5':
@@ -268,13 +268,13 @@ class EasyWorshipSongImport(SongImport):
self.db_set_record_struct(field_descriptions)
# Pick out the field description indexes we will need
try:
- success = True
fi_title = self.db_find_field(b'Title')
fi_author = self.db_find_field(b'Author')
fi_copy = self.db_find_field(b'Copyright')
fi_admin = self.db_find_field(b'Administrator')
fi_words = self.db_find_field(b'Words')
fi_ccli = self.db_find_field(b'Song Number')
+ success = True
except IndexError:
# This is the wrong table
success = False
diff --git a/openlp/plugins/songs/lib/importers/foilpresenter.py b/openlp/plugins/songs/lib/importers/foilpresenter.py
index da16c9b9e..ed64210b6 100644
--- a/openlp/plugins/songs/lib/importers/foilpresenter.py
+++ b/openlp/plugins/songs/lib/importers/foilpresenter.py
@@ -302,7 +302,7 @@ class FoilPresenter(object):
break
author_temp = []
for author in strings:
- temp = re.split(r',(?=\D{2})|(?<=\D),|\/(?=\D{3,})|(?<=\D);', author)
+ temp = re.split(r',(?=\D{2})|(?<=\D),|/(?=\D{3,})|(?<=\D);', author)
for tempx in temp:
author_temp.append(tempx)
for author in author_temp:
diff --git a/openlp/plugins/songs/lib/importers/openoffice.py b/openlp/plugins/songs/lib/importers/openoffice.py
index 2346dc039..0fc057e82 100644
--- a/openlp/plugins/songs/lib/importers/openoffice.py
+++ b/openlp/plugins/songs/lib/importers/openoffice.py
@@ -145,9 +145,7 @@ class OpenOfficeImport(SongImport):
"""
self.file_path = file_path
url = file_path.as_uri()
- properties = []
- properties.append(self.create_property('Hidden', True))
- properties = tuple(properties)
+ properties = tuple(self.create_property('Hidden', True))
try:
self.document = self.desktop.loadComponentFromURL(url, '_blank', 0, properties)
if not self.document.supportsService("com.sun.star.presentation.PresentationDocument") and not \
diff --git a/openlp/plugins/songs/lib/importers/opspro.py b/openlp/plugins/songs/lib/importers/opspro.py
index ece7dc41e..2ad14487e 100644
--- a/openlp/plugins/songs/lib/importers/opspro.py
+++ b/openlp/plugins/songs/lib/importers/opspro.py
@@ -136,8 +136,8 @@ class OPSProImport(SongImport):
verse_text = re.sub(r'^\d+\r\n', '', verse_text)
verse_def = 'v' + verse_number.group(1)
# Detect verse tags
- elif re.match(r'^.+?\:\r\n', verse_text):
- tag_match = re.match(r'^(.+?)\:\r\n(.*)', verse_text, flags=re.DOTALL)
+ elif re.match(r'^.+?:\r\n', verse_text):
+ tag_match = re.match(r'^(.+?):\r\n(.*)', verse_text, flags=re.DOTALL)
tag = tag_match.group(1).lower()
tag = tag.split(' ')[0]
verse_text = tag_match.group(2)
diff --git a/openlp/plugins/songs/lib/importers/propresenter.py b/openlp/plugins/songs/lib/importers/propresenter.py
index 0d666e449..05ac8aff5 100644
--- a/openlp/plugins/songs/lib/importers/propresenter.py
+++ b/openlp/plugins/songs/lib/importers/propresenter.py
@@ -80,7 +80,7 @@ class ProPresenterImport(SongImport):
self.parse_author(author)
# ProPresenter 4
- if(self.version >= 400 and self.version < 500):
+ if 400 <= self.version < 500:
self.copyright = root.get('CCLICopyrightInfo')
self.ccli_number = root.get('CCLILicenseNumber')
count = 0
@@ -95,7 +95,7 @@ class ProPresenterImport(SongImport):
self.add_verse(words, "v{count}".format(count=count))
# ProPresenter 5
- elif(self.version >= 500 and self.version < 600):
+ elif 500 <= self.version < 600:
self.copyright = root.get('CCLICopyrightInfo')
self.ccli_number = root.get('CCLILicenseNumber')
count = 0
@@ -111,7 +111,7 @@ class ProPresenterImport(SongImport):
self.add_verse(words, "v{count:d}".format(count=count))
# ProPresenter 6
- elif(self.version >= 600 and self.version < 700):
+ elif 600 <= self.version < 700:
self.copyright = root.get('CCLICopyrightYear')
self.ccli_number = root.get('CCLISongNumber')
count = 0
@@ -128,7 +128,7 @@ class ProPresenterImport(SongImport):
b64Data = contents.text
data = base64.standard_b64decode(b64Data)
words = None
- if(contents.get('rvXMLIvarName') == "RTFData"):
+ if contents.get('rvXMLIvarName') == "RTFData":
words, encoding = strip_rtf(data.decode())
break
if words:
diff --git a/openlp/plugins/songs/lib/importers/songbeamer.py b/openlp/plugins/songs/lib/importers/songbeamer.py
index 3b5bf4aac..7f19086ac 100644
--- a/openlp/plugins/songs/lib/importers/songbeamer.py
+++ b/openlp/plugins/songs/lib/importers/songbeamer.py
@@ -128,7 +128,7 @@ class SongBeamerImport(SongImport):
# The encoding should only be ANSI (cp1252), UTF-8, Unicode, Big-Endian-Unicode.
# So if it doesn't start with 'u' we default to cp1252. See:
# https://forum.songbeamer.com/viewtopic.php?p=419&sid=ca4814924e37c11e4438b7272a98b6f2
- if not self.input_file_encoding.lower().startswith('u'):
+ if self.input_file_encoding and not self.input_file_encoding.lower().startswith('u'):
self.input_file_encoding = 'cp1252'
with file_path.open(encoding=self.input_file_encoding) as song_file:
song_data = song_file.readlines()
diff --git a/openlp/plugins/songs/lib/importers/videopsalm.py b/openlp/plugins/songs/lib/importers/videopsalm.py
index bc0e06093..ff3eec52b 100644
--- a/openlp/plugins/songs/lib/importers/videopsalm.py
+++ b/openlp/plugins/songs/lib/importers/videopsalm.py
@@ -75,7 +75,7 @@ class VideoPsalmImport(SongImport):
c = next(file_content_it)
processed_content += '"' + c
# Remove control characters
- elif (c < chr(32)):
+ elif c < chr(32):
processed_content += ' '
# Handle escaped characters
elif c == '\\':
diff --git a/openlp/plugins/songs/lib/openlyricsxml.py b/openlp/plugins/songs/lib/openlyricsxml.py
index efa3103d2..7cb1f7a38 100644
--- a/openlp/plugins/songs/lib/openlyricsxml.py
+++ b/openlp/plugins/songs/lib/openlyricsxml.py
@@ -221,7 +221,7 @@ class OpenLyrics(object):
"""
IMPLEMENTED_VERSION = '0.8'
START_TAGS_REGEX = re.compile(r'\{(\w+)\}')
- END_TAGS_REGEX = re.compile(r'\{\/(\w+)\}')
+ END_TAGS_REGEX = re.compile(r'\{/(\w+)\}')
VERSE_TAG_SPLITTER = re.compile('([a-zA-Z]+)([0-9]*)([a-zA-Z]?)')
def __init__(self, manager):
diff --git a/tests/functional/openlp_core/common/test_common.py b/tests/functional/openlp_core/common/test_common.py
index 21f246a75..6624a4819 100644
--- a/tests/functional/openlp_core/common/test_common.py
+++ b/tests/functional/openlp_core/common/test_common.py
@@ -26,7 +26,7 @@ from pathlib import Path
from unittest import TestCase
from unittest.mock import MagicMock, call, patch
-from openlp.core.common import clean_button_text, de_hump, extension_loader, is_linux, is_macosx, is_win, \
+from openlp.core.common import Singleton, clean_button_text, de_hump, extension_loader, is_linux, is_macosx, is_win, \
normalize_str, path_to_module, trace_error_handler
@@ -163,6 +163,48 @@ class TestCommonFunctions(TestCase):
mocked_logger.error.assert_called_with(
'OpenLP Error trace\n File openlp.fake at line 56 \n\t called trace_error_handler_test')
+ def test_singleton_metaclass_multiple_init(self):
+ """
+ Test that a class using the Singleton Metaclass is only initialised once despite being called several times and
+ that the same instance is returned each time..
+ """
+ # GIVEN: The Singleton Metaclass and a test class using it
+ class SingletonClass(metaclass=Singleton):
+ def __init__(self):
+ pass
+
+ with patch.object(SingletonClass, '__init__', return_value=None) as patched_init:
+
+ # WHEN: Initialising the class multiple times
+ inst_1 = SingletonClass()
+ inst_2 = SingletonClass()
+
+ # THEN: The __init__ method of the SingletonClass should have only been called once, and both returned values
+ # should be the same instance.
+ assert inst_1 is inst_2
+ assert patched_init.call_count == 1
+
+ def test_singleton_metaclass_multiple_classes(self):
+ """
+ Test that multiple classes using the Singleton Metaclass return the different an appropriate instances.
+ """
+ # GIVEN: Two different classes using the Singleton Metaclass
+ class SingletonClass1(metaclass=Singleton):
+ def __init__(self):
+ pass
+
+ class SingletonClass2(metaclass=Singleton):
+ def __init__(self):
+ pass
+
+ # WHEN: Initialising both classes
+ s_c1 = SingletonClass1()
+ s_c2 = SingletonClass2()
+
+ # THEN: The instances should be an instance of the appropriate class
+ assert isinstance(s_c1, SingletonClass1)
+ assert isinstance(s_c2, SingletonClass2)
+
def test_is_win(self):
"""
Test the is_win() function
diff --git a/tests/functional/openlp_core/lib/test_theme.py b/tests/functional/openlp_core/lib/test_theme.py
index f53d4367c..800e9c584 100644
--- a/tests/functional/openlp_core/lib/test_theme.py
+++ b/tests/functional/openlp_core/lib/test_theme.py
@@ -25,7 +25,78 @@ Package to test the openlp.core.lib.theme package.
from pathlib import Path
from unittest import TestCase
-from openlp.core.lib.theme import Theme
+from openlp.core.lib.theme import BackgroundType, Theme
+
+
+class TestBackgroundType(TestCase):
+ """
+ Test the BackgroundType enum methods.
+ """
+ def test_solid_to_string(self):
+ """
+ Test the to_string method of :class:`BackgroundType`
+ """
+ # GIVEN: A BackgroundType member
+ background_type = BackgroundType.Solid
+
+ # WHEN: Calling BackgroundType.to_string
+ # THEN: The string equivalent should have been returned
+ assert BackgroundType.to_string(background_type) == 'solid'
+
+ def test_gradient_to_string(self):
+ """
+ Test the to_string method of :class:`BackgroundType`
+ """
+ # GIVEN: A BackgroundType member
+ background_type = BackgroundType.Gradient
+
+ # WHEN: Calling BackgroundType.to_string
+ # THEN: The string equivalent should have been returned
+ assert BackgroundType.to_string(background_type) == 'gradient'
+
+ def test_image_to_string(self):
+ """
+ Test the to_string method of :class:`BackgroundType`
+ """
+ # GIVEN: A BackgroundType member
+ background_type = BackgroundType.Image
+
+ # WHEN: Calling BackgroundType.to_string
+ # THEN: The string equivalent should have been returned
+ assert BackgroundType.to_string(background_type) == 'image'
+
+ def test_transparent_to_string(self):
+ """
+ Test the to_string method of :class:`BackgroundType`
+ """
+ # GIVEN: A BackgroundType member
+ background_type = BackgroundType.Transparent
+
+ # WHEN: Calling BackgroundType.to_string
+ # THEN: The string equivalent should have been returned
+ assert BackgroundType.to_string(background_type) == 'transparent'
+
+ def test_video_to_string(self):
+ """
+ Test the to_string method of :class:`BackgroundType`
+ """
+ # GIVEN: A BackgroundType member
+ background_type = BackgroundType.Video
+
+ # WHEN: Calling BackgroundType.to_string
+ # THEN: The string equivalent should have been returned
+ assert BackgroundType.to_string(background_type) == 'video'
+
+ def test_stream_to_string(self):
+ """
+ Test the to_string method of :class:`BackgroundType`
+ """
+ # GIVEN: A BackgroundType member
+ background_type = BackgroundType.Stream
+
+ # WHEN: Calling BackgroundType.to_string
+ # THEN: The string equivalent should have been returned
+ assert BackgroundType.to_string(background_type) == 'stream'
class TestTheme(TestCase):
@@ -111,4 +182,4 @@ class TestTheme(TestCase):
assert 0 == theme.display_vertical_align, 'display_vertical_align should be 0'
assert theme.font_footer_bold is False, 'font_footer_bold should be False'
assert 'Arial' == theme.font_main_name, 'font_main_name should be "Arial"'
- assert 47 == len(theme.__dict__), 'The theme should have 47 attributes'
+ assert 48 == len(theme.__dict__), 'The theme should have 48 attributes'
diff --git a/tests/functional/openlp_core/ui/test_icons.py b/tests/functional/openlp_core/ui/test_icons.py
index 9d91ec367..6bcdc641c 100644
--- a/tests/functional/openlp_core/ui/test_icons.py
+++ b/tests/functional/openlp_core/ui/test_icons.py
@@ -33,7 +33,7 @@ from tests.helpers.testmixin import TestMixin
class TestIcons(TestCase, TestMixin):
- @patch('openlp.core.ui.icons.UiIcons.load')
+ @patch('openlp.core.ui.icons.UiIcons.__init__', return_value=None)
def test_simple_icon(self, _):
# GIVEN: an basic set of icons
icons = UiIcons()