openlp/openlp/core/ui/servicemanager.py

1631 lines
78 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2012-12-10 06:15:42 +00:00
# vim: autoindent shiftwidth=4 expandtab textwidth=120 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
# Copyright (c) 2008-2014 Raoul Snyman #
# Portions copyright (c) 2008-2014 Tim Bentley, Gerald Britton, Jonathan #
# Corwin, Samuel Findlay, Michael Gorven, Scott Guerrieri, Matthias Hub, #
2012-11-11 21:16:14 +00:00
# Meinert Jordan, Armin Köhler, Erik Lundin, Edwin Lunando, Brian T. Meyer. #
2012-10-21 13:16:22 +00:00
# Joshua Miller, Stevan Pettit, Andreas Preikschat, Mattias Põldaru, #
# Christian Richter, Philip Ridout, Simon Scudder, Jeffrey Smith, #
# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Dave Warnock, #
# Frode Woldsund, Martin Zibricky, Patrick Zimmermann #
# --------------------------------------------------------------------------- #
# 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-02-01 21:34:23 +00:00
"""
The service manager sets up, loads, saves and manages services.
"""
2013-12-26 08:56:53 +00:00
import html
2011-02-04 12:57:48 +00:00
import os
import shutil
import zipfile
import json
from tempfile import mkstemp
from datetime import datetime, timedelta
from PyQt4 import QtCore, QtGui
2010-03-03 17:48:37 +00:00
from openlp.core.common import Registry, AppLocation, Settings, ThemeLevel, OpenLPMixin, RegistryMixin, \
check_directory_exists, UiStrings, translate
2013-12-13 17:44:05 +00:00
from openlp.core.lib import OpenLPToolbar, ServiceItem, ItemCapabilities, PluginStatus, build_icon
from openlp.core.lib.ui import critical_error_message_box, create_widget_action, find_and_set_in_combo_box
2013-02-02 19:49:56 +00:00
from openlp.core.ui import ServiceNoteForm, ServiceItemEditForm, StartTimeForm
2011-02-19 17:33:24 +00:00
from openlp.core.ui.printserviceform import PrintServiceForm
2013-10-13 13:51:13 +00:00
from openlp.core.utils import delete_file, split_filename, format_time
2013-02-02 19:49:56 +00:00
from openlp.core.utils.actions import ActionList, CategoryOrder
2013-02-01 21:34:23 +00:00
2013-02-02 19:49:56 +00:00
class ServiceManagerList(QtGui.QTreeWidget):
"""
Set up key bindings and mouse behaviour for the service list
"""
2013-12-26 08:56:53 +00:00
def __init__(self, service_manager, parent=None):
2013-02-01 21:34:23 +00:00
"""
Constructor
"""
2013-07-18 14:32:23 +00:00
super(ServiceManagerList, self).__init__(parent)
2013-12-26 08:56:53 +00:00
self.service_manager = service_manager
2013-02-02 19:49:56 +00:00
def keyPressEvent(self, event):
"""
Capture Key press and respond accordingly.
"""
if isinstance(event, QtGui.QKeyEvent):
# here accept the event and do something
if event.key() == QtCore.Qt.Key_Up:
2013-12-26 08:56:53 +00:00
self.service_manager.on_move_selection_up()
2013-02-02 19:49:56 +00:00
event.accept()
elif event.key() == QtCore.Qt.Key_Down:
2013-12-26 08:56:53 +00:00
self.service_manager.on_move_selection_down()
2013-02-02 19:49:56 +00:00
event.accept()
elif event.key() == QtCore.Qt.Key_Delete:
self.service_manager.on_delete_from_service()
2013-02-02 19:49:56 +00:00
event.accept()
event.ignore()
else:
event.ignore()
def mouseMoveEvent(self, event):
"""
2013-03-29 12:53:07 +00:00
Drag and drop event does not care what data is selected as the recipient will use events to request the data
move just tell it what plugin to call
2013-02-02 19:49:56 +00:00
"""
if event.buttons() != QtCore.Qt.LeftButton:
event.ignore()
return
if not self.itemAt(self.mapFromGlobal(QtGui.QCursor.pos())):
event.ignore()
return
drag = QtGui.QDrag(self)
mime_data = QtCore.QMimeData()
drag.setMimeData(mime_data)
2013-08-31 18:17:38 +00:00
mime_data.setText('ServiceManager')
2013-02-02 19:49:56 +00:00
drag.start(QtCore.Qt.CopyAction)
2013-02-01 21:34:23 +00:00
class Ui_ServiceManager(object):
2013-02-02 19:49:56 +00:00
"""
UI part of the Service Manager
"""
def setup_ui(self, widget):
"""
Define the UI
"""
# start with the layout
self.layout = QtGui.QVBoxLayout(self)
self.layout.setSpacing(0)
self.layout.setMargin(0)
2013-02-02 19:49:56 +00:00
# Create the top toolbar
self.toolbar = OpenLPToolbar(self)
2013-08-31 18:17:38 +00:00
self.toolbar.add_toolbar_action('newService', text=UiStrings().NewService, icon=':/general/general_new.png',
2013-12-24 15:55:01 +00:00
tooltip=UiStrings().CreateService, triggers=self.on_new_service_clicked)
2013-08-31 18:17:38 +00:00
self.toolbar.add_toolbar_action('openService', text=UiStrings().OpenService,
2013-12-24 15:55:01 +00:00
icon=':/general/general_open.png',
tooltip=translate('OpenLP.ServiceManager', 'Load an existing service.'),
triggers=self.on_load_service_clicked)
2013-08-31 18:17:38 +00:00
self.toolbar.add_toolbar_action('saveService', text=UiStrings().SaveService,
2013-12-24 15:55:01 +00:00
icon=':/general/general_save.png',
2013-12-24 21:24:52 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Save this service.'),
2013-12-24 15:55:01 +00:00
triggers=self.decide_save_method)
2013-02-02 19:49:56 +00:00
self.toolbar.addSeparator()
2013-08-31 18:17:38 +00:00
self.theme_label = QtGui.QLabel('%s:' % UiStrings().Theme, self)
2013-02-02 19:49:56 +00:00
self.theme_label.setMargin(3)
2013-08-31 18:17:38 +00:00
self.theme_label.setObjectName('theme_label')
2013-03-07 06:48:09 +00:00
self.toolbar.add_toolbar_widget(self.theme_label)
2013-02-02 19:49:56 +00:00
self.theme_combo_box = QtGui.QComboBox(self.toolbar)
self.theme_combo_box.setToolTip(translate('OpenLP.ServiceManager', 'Select a theme for the service.'))
self.theme_combo_box.setSizeAdjustPolicy(QtGui.QComboBox.AdjustToMinimumContentsLength)
self.theme_combo_box.setSizePolicy(QtGui.QSizePolicy.Expanding, QtGui.QSizePolicy.Fixed)
2013-08-31 18:17:38 +00:00
self.theme_combo_box.setObjectName('theme_combo_box')
2013-03-07 06:48:09 +00:00
self.toolbar.add_toolbar_widget(self.theme_combo_box)
2013-08-31 18:17:38 +00:00
self.toolbar.setObjectName('toolbar')
2013-02-02 19:49:56 +00:00
self.layout.addWidget(self.toolbar)
# Create the service manager list
self.service_manager_list = ServiceManagerList(self)
self.service_manager_list.setEditTriggers(
QtGui.QAbstractItemView.CurrentChanged |
QtGui.QAbstractItemView.DoubleClicked |
QtGui.QAbstractItemView.EditKeyPressed)
self.service_manager_list.setDragDropMode(QtGui.QAbstractItemView.DragDrop)
self.service_manager_list.setAlternatingRowColors(True)
self.service_manager_list.setHeaderHidden(True)
self.service_manager_list.setExpandsOnDoubleClick(False)
self.service_manager_list.setContextMenuPolicy(QtCore.Qt.CustomContextMenu)
2013-02-14 21:50:10 +00:00
self.service_manager_list.customContextMenuRequested.connect(self.context_menu)
2013-08-31 18:17:38 +00:00
self.service_manager_list.setObjectName('service_manager_list')
2013-02-02 19:49:56 +00:00
# enable drop
self.service_manager_list.__class__.dragEnterEvent = self.drag_enter_event
self.service_manager_list.__class__.dragMoveEvent = self.drag_enter_event
self.service_manager_list.__class__.dropEvent = self.drop_event
self.layout.addWidget(self.service_manager_list)
# Add the bottom toolbar
self.order_toolbar = OpenLPToolbar(self)
action_list = ActionList.get_instance()
action_list.add_category(UiStrings().Service, CategoryOrder.standard_toolbar)
self.service_manager_list.move_top = self.order_toolbar.add_toolbar_action(
'moveTop',
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', 'Move to &top'), icon=':/services/service_top.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Move item to the top of the service.'),
can_shortcuts=True, category=UiStrings().Service, triggers=self.on_service_top)
self.service_manager_list.move_up = self.order_toolbar.add_toolbar_action(
'moveUp',
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', 'Move &up'), icon=':/services/service_up.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Move item up one position in the service.'),
can_shortcuts=True, category=UiStrings().Service, triggers=self.on_service_up)
self.service_manager_list.move_down = self.order_toolbar.add_toolbar_action(
'moveDown',
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', 'Move &down'), icon=':/services/service_down.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Move item down one position in the service.'),
can_shortcuts=True, category=UiStrings().Service, triggers=self.on_service_down)
self.service_manager_list.move_bottom = self.order_toolbar.add_toolbar_action(
'moveBottom',
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', 'Move to &bottom'), icon=':/services/service_bottom.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Move item to the end of the service.'),
can_shortcuts=True, category=UiStrings().Service, triggers=self.on_service_end)
self.service_manager_list.down = self.order_toolbar.add_toolbar_action(
'down',
text=translate('OpenLP.ServiceManager', 'Move &down'), can_shortcuts=True,
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Moves the selection down the window.'), visible=False,
triggers=self.on_move_selection_down)
2013-02-02 19:49:56 +00:00
action_list.add_action(self.service_manager_list.down)
self.service_manager_list.up = self.order_toolbar.add_toolbar_action(
'up',
text=translate('OpenLP.ServiceManager', 'Move up'), can_shortcuts=True,
tooltip=translate('OpenLP.ServiceManager', 'Moves the selection up the window.'), visible=False,
2013-02-02 19:49:56 +00:00
triggers=self.on_move_selection_up)
action_list.add_action(self.service_manager_list.up)
self.order_toolbar.addSeparator()
self.service_manager_list.delete = self.order_toolbar.add_toolbar_action(
'delete', can_shortcuts=True,
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', '&Delete From Service'), icon=':/general/general_delete.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Delete the selected item from the service.'),
triggers=self.on_delete_from_service)
2013-02-02 19:49:56 +00:00
self.order_toolbar.addSeparator()
self.service_manager_list.expand = self.order_toolbar.add_toolbar_action(
'expand', can_shortcuts=True,
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', '&Expand all'), icon=':/services/service_expand_all.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Expand all the service items.'),
2013-12-26 08:56:53 +00:00
category=UiStrings().Service, triggers=self.on_expand_all)
self.service_manager_list.collapse = self.order_toolbar.add_toolbar_action(
'collapse', can_shortcuts=True,
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', '&Collapse all'), icon=':/services/service_collapse_all.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Collapse all the service items.'),
2013-12-26 08:56:53 +00:00
category=UiStrings().Service, triggers=self.on_collapse_all)
2013-02-02 19:49:56 +00:00
self.order_toolbar.addSeparator()
self.service_manager_list.make_live = self.order_toolbar.add_toolbar_action(
'make_live', can_shortcuts=True,
2013-08-31 18:17:38 +00:00
text=translate('OpenLP.ServiceManager', 'Go Live'), icon=':/general/general_live.png',
2013-02-02 19:49:56 +00:00
tooltip=translate('OpenLP.ServiceManager', 'Send the selected item to Live.'),
category=UiStrings().Service,
2013-02-02 19:49:56 +00:00
triggers=self.make_live)
self.layout.addWidget(self.order_toolbar)
# Connect up our signals and slots
2013-02-14 21:50:10 +00:00
self.theme_combo_box.activated.connect(self.on_theme_combo_box_selected)
self.service_manager_list.doubleClicked.connect(self.on_make_live)
self.service_manager_list.itemCollapsed.connect(self.collapsed)
self.service_manager_list.itemExpanded.connect(self.expanded)
2013-02-02 19:49:56 +00:00
# Last little bits of setting up
2013-08-31 18:17:38 +00:00
self.service_theme = Settings().value(self.main_window.service_manager_settings_section + '/service theme')
2013-12-26 08:56:53 +00:00
self.service_path = AppLocation.get_section_data_path('servicemanager')
2013-02-02 19:49:56 +00:00
# build the drag and drop context menu
2013-12-26 08:56:53 +00:00
self.dnd_menu = QtGui.QMenu()
self.new_action = self.dnd_menu.addAction(translate('OpenLP.ServiceManager', '&Add New Item'))
self.new_action.setIcon(build_icon(':/general/general_edit.png'))
self.add_to_action = self.dnd_menu.addAction(translate('OpenLP.ServiceManager', '&Add to Selected Item'))
self.add_to_action.setIcon(build_icon(':/general/general_edit.png'))
2013-02-02 19:49:56 +00:00
# build the context menu
self.menu = QtGui.QMenu()
self.edit_action = create_widget_action(self.menu, text=translate('OpenLP.ServiceManager', '&Edit Item'),
2013-12-26 08:56:53 +00:00
icon=':/general/general_edit.png', triggers=self.remote_edit)
2013-02-02 19:49:56 +00:00
self.maintain_action = create_widget_action(self.menu, text=translate('OpenLP.ServiceManager', '&Reorder Item'),
2013-12-26 08:56:53 +00:00
icon=':/general/general_edit.png',
triggers=self.on_service_item_edit_form)
2013-02-02 19:49:56 +00:00
self.notes_action = create_widget_action(self.menu, text=translate('OpenLP.ServiceManager', '&Notes'),
2013-12-26 08:56:53 +00:00
icon=':/services/service_notes.png',
triggers=self.on_service_item_note_form)
2013-02-02 19:49:56 +00:00
self.time_action = create_widget_action(self.menu, text=translate('OpenLP.ServiceManager', '&Start Time'),
2013-12-26 08:56:53 +00:00
icon=':/media/media_time.png', triggers=self.on_start_time_form)
2013-08-31 18:17:38 +00:00
self.auto_start_action = create_widget_action(self.menu, text='',
2013-12-26 08:56:53 +00:00
icon=':/media/auto-start_active.png',
triggers=self.on_auto_start)
2013-02-02 19:49:56 +00:00
# Add already existing delete action to the menu.
self.menu.addAction(self.service_manager_list.delete)
self.create_custom_action = create_widget_action(self.menu,
2013-12-26 08:56:53 +00:00
text=translate('OpenLP.ServiceManager', 'Create New &Custom '
'Slide'),
icon=':/general/general_edit.png',
triggers=self.create_custom)
2013-02-02 19:49:56 +00:00
self.menu.addSeparator()
# Add AutoPlay menu actions
self.auto_play_slides_menu = QtGui.QMenu(translate('OpenLP.ServiceManager', '&Auto play slides'))
self.menu.addMenu(self.auto_play_slides_menu)
auto_play_slides_group = QtGui.QActionGroup(self.auto_play_slides_menu)
auto_play_slides_group.setExclusive(True)
self.auto_play_slides_loop = create_widget_action(self.auto_play_slides_menu,
2013-12-26 08:56:53 +00:00
text=translate('OpenLP.ServiceManager', 'Auto play slides '
'&Loop'),
checked=False, triggers=self.toggle_auto_play_slides_loop)
auto_play_slides_group.addAction(self.auto_play_slides_loop)
self.auto_play_slides_once = create_widget_action(self.auto_play_slides_menu,
2013-12-26 08:56:53 +00:00
text=translate('OpenLP.ServiceManager', 'Auto play slides '
'&Once'),
checked=False, triggers=self.toggle_auto_play_slides_once)
auto_play_slides_group.addAction(self.auto_play_slides_once)
self.auto_play_slides_menu.addSeparator()
self.timed_slide_interval = create_widget_action(self.auto_play_slides_menu,
2013-12-26 08:56:53 +00:00
text=translate('OpenLP.ServiceManager', '&Delay between '
'slides'),
triggers=self.on_timed_slide_interval)
2013-02-02 19:49:56 +00:00
self.menu.addSeparator()
self.preview_action = create_widget_action(self.menu, text=translate('OpenLP.ServiceManager', 'Show &Preview'),
2013-12-26 08:56:53 +00:00
icon=':/general/general_preview.png', triggers=self.make_preview)
2013-02-02 19:49:56 +00:00
# Add already existing make live action to the menu.
self.menu.addAction(self.service_manager_list.make_live)
self.menu.addSeparator()
self.theme_menu = QtGui.QMenu(translate('OpenLP.ServiceManager', '&Change Item Theme'))
self.menu.addMenu(self.theme_menu)
self.service_manager_list.addActions(
2013-12-26 08:56:53 +00:00
[self.service_manager_list.move_down,
self.service_manager_list.move_up,
2013-02-02 19:49:56 +00:00
self.service_manager_list.make_live,
2013-12-26 08:56:53 +00:00
self.service_manager_list.move_top,
self.service_manager_list.move_bottom,
2013-02-02 19:49:56 +00:00
self.service_manager_list.up,
self.service_manager_list.down,
self.service_manager_list.expand,
self.service_manager_list.collapse
2013-12-31 20:29:03 +00:00
])
2013-08-31 18:17:38 +00:00
Registry().register_function('theme_update_list', self.update_theme_list)
2013-12-26 08:56:53 +00:00
Registry().register_function('config_screen_changed', self.regenerate_service_items)
2013-08-31 18:17:38 +00:00
Registry().register_function('theme_update_global', self.theme_change)
Registry().register_function('mediaitem_suffix_reset', self.reset_supported_suffixes)
2013-02-02 19:49:56 +00:00
def drag_enter_event(self, event):
"""
Accept Drag events
``event``
Handle of the event passed
2013-02-02 19:49:56 +00:00
"""
event.accept()
2013-01-27 07:36:04 +00:00
class ServiceManager(OpenLPMixin, RegistryMixin, QtGui.QWidget, Ui_ServiceManager):
2013-01-27 07:36:04 +00:00
"""
2013-03-29 12:53:07 +00:00
Manages the services. This involves taking text strings from plugins and adding them to the service. This service
can then be zipped up with all the resources used into one OSZ or oszl file for use on any OpenLP v2 installation.
Also handles the UI tasks of moving things up and down etc.
2013-01-27 07:36:04 +00:00
"""
def __init__(self, parent=None):
"""
Sets up the service manager, toolbars, list view, et al.
"""
2013-07-18 14:19:01 +00:00
super(ServiceManager, self).__init__(parent)
2013-08-31 18:17:38 +00:00
self.active = build_icon(':/media/auto-start_active.png')
self.inactive = build_icon(':/media/auto-start_inactive.png')
2013-01-27 07:36:04 +00:00
self.service_items = []
self.suffixes = []
self.drop_position = 0
self.service_id = 0
# is a new service and has not been saved
self._modified = False
2013-08-31 18:17:38 +00:00
self._file_name = ''
2013-01-27 07:36:04 +00:00
self.service_has_all_original_files = True
def bootstrap_initialise(self):
2013-01-27 07:36:04 +00:00
self.setup_ui(self)
2013-03-17 21:20:40 +00:00
# Need to use event as called across threads and UI is updated
2013-08-31 18:17:38 +00:00
QtCore.QObject.connect(self, QtCore.SIGNAL('servicemanager_set_item'), self.on_set_item)
2013-01-27 07:36:04 +00:00
def bootstrap_post_set_up(self):
"""
Can be set up as a late setup
"""
self.service_note_form = ServiceNoteForm()
self.service_item_edit_form = ServiceItemEditForm()
self.start_time_form = StartTimeForm()
2013-01-27 07:36:04 +00:00
def set_modified(self, modified=True):
"""
2013-03-29 12:53:07 +00:00
Setter for property "modified". Sets whether or not the current service has been modified.
"""
2012-04-10 19:38:25 +00:00
if modified:
2013-01-21 07:29:43 +00:00
self.service_id += 1
self._modified = modified
2013-01-27 07:36:04 +00:00
service_file = self.short_file_name() or translate('OpenLP.ServiceManager', 'Untitled Service')
2013-03-16 11:05:52 +00:00
self.main_window.set_service_modified(modified, service_file)
2013-01-27 07:36:04 +00:00
def is_modified(self):
"""
Getter for boolean property "modified".
"""
return self._modified
2013-01-27 07:36:04 +00:00
def set_file_name(self, file_name):
"""
Setter for service file.
"""
2013-08-31 18:17:38 +00:00
self._file_name = str(file_name)
2013-03-16 11:05:52 +00:00
self.main_window.set_service_modified(self.is_modified(), self.short_file_name())
2013-08-31 18:17:38 +00:00
Settings().setValue('servicemanager/last file', file_name)
self._save_lite = self._file_name.endswith('.oszl')
2013-01-27 07:36:04 +00:00
def file_name(self):
"""
Return the current file name including path.
"""
2013-01-27 07:36:04 +00:00
return self._file_name
2013-01-27 07:36:04 +00:00
def short_file_name(self):
"""
Return the current file name, excluding the path.
"""
2013-01-27 07:36:04 +00:00
return split_filename(self._file_name)[1]
2013-01-27 07:36:04 +00:00
def reset_supported_suffixes(self):
"""
2012-11-26 17:57:36 +00:00
Resets the Suffixes list.
"""
self.suffixes = []
2013-12-31 20:29:03 +00:00
def supported_suffixes(self, suffix_list):
2012-08-24 20:06:56 +00:00
"""
2013-03-29 12:53:07 +00:00
Adds Suffixes supported to the master list. Called from Plugins.
2012-08-24 20:06:56 +00:00
2013-12-31 20:29:03 +00:00
``suffix_list``
New Suffix's to be supported
2012-08-24 20:06:56 +00:00
"""
2013-12-31 20:29:03 +00:00
for suffix in suffix_list:
if not suffix in self.suffixes:
self.suffixes.append(suffix)
2010-05-05 19:14:48 +00:00
2013-12-31 07:27:07 +00:00
def on_new_service_clicked(self, field=None):
"""
Create a new service.
"""
2013-01-27 07:36:04 +00:00
if self.is_modified():
result = self.save_modified_service()
if result == QtGui.QMessageBox.Cancel:
return False
elif result == QtGui.QMessageBox.Save:
2013-01-27 07:36:04 +00:00
if not self.decide_save_method():
return False
2013-01-27 07:36:04 +00:00
self.new_file()
2013-01-27 07:36:04 +00:00
def on_load_service_clicked(self, load_file=None):
2011-08-02 18:17:07 +00:00
"""
2013-03-29 12:53:07 +00:00
Loads the service file and saves the existing one it there is one unchanged.
2011-08-02 18:17:07 +00:00
2013-01-29 17:19:19 +00:00
``load_file``
2013-03-29 12:53:07 +00:00
The service file to the loaded. Will be None is from menu so selection will be required.
2011-08-02 18:17:07 +00:00
"""
2013-01-27 07:36:04 +00:00
if self.is_modified():
result = self.save_modified_service()
if result == QtGui.QMessageBox.Cancel:
return False
elif result == QtGui.QMessageBox.Save:
2013-01-27 07:36:04 +00:00
self.decide_save_method()
if not load_file:
file_name = QtGui.QFileDialog.getOpenFileName(
self.main_window,
2011-08-02 18:17:07 +00:00
translate('OpenLP.ServiceManager', 'Open File'),
2013-08-31 18:17:38 +00:00
Settings().value(self.main_window.service_manager_settings_section + '/last directory'),
translate('OpenLP.ServiceManager', 'OpenLP Service Files (*.osz *.oszl)')
)
2013-01-27 07:36:04 +00:00
if not file_name:
2011-08-02 18:17:07 +00:00
return False
else:
2013-01-27 07:36:04 +00:00
file_name = load_file
2013-08-31 18:17:38 +00:00
Settings().setValue(self.main_window.service_manager_settings_section + '/last directory',
2013-12-24 15:55:01 +00:00
split_filename(file_name)[0])
2013-01-27 07:36:04 +00:00
self.load_file(file_name)
2013-01-27 07:36:04 +00:00
def save_modified_service(self):
2013-01-21 11:11:47 +00:00
"""
Check to see if a service needs to be saved.
"""
return QtGui.QMessageBox.question(self.main_window,
2013-12-24 21:24:52 +00:00
translate('OpenLP.ServiceManager', 'Modified Service'),
2013-12-24 15:55:01 +00:00
translate('OpenLP.ServiceManager',
2013-12-24 21:24:52 +00:00
'The current service has been modified. Would you like to save '
'this service?'),
2013-12-24 15:55:01 +00:00
QtGui.QMessageBox.Save | QtGui.QMessageBox.Discard |
2013-12-26 08:56:53 +00:00
QtGui.QMessageBox.Cancel, QtGui.QMessageBox.Save)
def on_recent_service_clicked(self, field=None):
2013-01-21 11:11:47 +00:00
"""
Load a recent file as the service triggered by mainwindow recent service list.
"""
sender = self.sender()
2013-01-27 07:36:04 +00:00
self.load_file(sender.data())
2013-01-27 07:36:04 +00:00
def new_file(self):
"""
Create a blank new service file.
"""
2013-01-27 07:36:04 +00:00
self.service_manager_list.clear()
self.service_items = []
2013-08-31 18:17:38 +00:00
self.set_file_name('')
2013-01-21 07:29:43 +00:00
self.service_id += 1
2013-01-27 07:36:04 +00:00
self.set_modified(False)
2013-08-31 18:17:38 +00:00
Settings().setValue('servicemanager/last file', '')
2013-02-03 09:07:31 +00:00
self.plugin_manager.new_service_created()
2013-01-27 07:36:04 +00:00
def save_file(self):
"""
Save the current service file.
2013-03-29 12:53:07 +00:00
A temporary file is created so that we don't overwrite the existing one and leave a mangled service file should
there be an error when saving. Audio files are also copied into the service manager directory, and then packaged
into the zip file.
"""
2013-01-27 07:36:04 +00:00
if not self.file_name():
return self.save_file_as()
2013-08-31 18:17:38 +00:00
temp_file, temp_file_name = mkstemp('.osz', 'openlp_')
# We don't need the file handle.
os.close(temp_file)
2013-12-31 20:29:03 +00:00
self.log_debug(temp_file_name)
2013-08-31 18:17:38 +00:00
path_file_name = str(self.file_name())
2011-08-28 17:45:13 +00:00
path, file_name = os.path.split(path_file_name)
base_name = os.path.splitext(file_name)[0]
service_file_name = '%s.osj' % base_name
2013-12-31 20:29:03 +00:00
self.log_debug('ServiceManager.save_file - %s', path_file_name)
2013-08-31 18:17:38 +00:00
Settings().setValue(self.main_window.service_manager_settings_section + '/last directory', path)
service = []
write_list = []
missing_list = []
2011-08-28 17:45:13 +00:00
audio_files = []
total_size = 0
2013-02-03 19:23:12 +00:00
self.application.set_busy_cursor()
2011-04-29 07:40:19 +00:00
# Number of items + 1 to zip it
2013-03-16 11:05:52 +00:00
self.main_window.display_progress_bar(len(self.service_items) + 1)
# Get list of missing files, and list of files to write
2013-01-27 07:36:04 +00:00
for item in self.service_items:
2013-08-31 18:17:38 +00:00
if not item['service_item'].uses_file():
continue
2013-08-31 18:17:38 +00:00
for frame in item['service_item'].get_frames():
path_from = item['service_item'].get_frame_path(frame=frame)
if path_from in write_list or path_from in missing_list:
continue
2011-08-04 13:43:05 +00:00
if not os.path.exists(path_from):
missing_list.append(path_from)
else:
write_list.append(path_from)
if missing_list:
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
2012-12-20 12:09:12 +00:00
title = translate('OpenLP.ServiceManager', 'Service File(s) Missing')
2012-12-05 20:33:39 +00:00
message = translate('OpenLP.ServiceManager',
2013-12-24 15:55:01 +00:00
'The following file(s) in the service are missing:\n\t%s\n\n'
'These files will be removed if you continue to save.') % "\n\t".join(missing_list)
2012-12-10 18:04:58 +00:00
answer = QtGui.QMessageBox.critical(self, title, message,
2013-12-24 15:55:01 +00:00
QtGui.QMessageBox.StandardButtons(QtGui.QMessageBox.Ok |
QtGui.QMessageBox.Cancel))
if answer == QtGui.QMessageBox.Cancel:
2013-03-16 11:05:52 +00:00
self.main_window.finished_progress_bar()
return False
# Check if item contains a missing file.
2013-01-27 07:36:04 +00:00
for item in list(self.service_items):
2013-03-07 08:05:43 +00:00
self.main_window.increment_progress_bar()
2013-08-31 18:17:38 +00:00
item['service_item'].remove_invalid_frames(missing_list)
if item['service_item'].missing_frames():
2013-01-27 07:36:04 +00:00
self.service_items.remove(item)
else:
2013-08-31 18:17:38 +00:00
service_item = item['service_item'].get_service_repr(self._save_lite)
if service_item['header']['background_audio']:
for i, file_name in enumerate(service_item['header']['background_audio']):
new_file = os.path.join('audio', item['service_item'].unique_identifier, file_name)
2013-01-27 07:36:04 +00:00
audio_files.append((file_name, new_file))
2013-08-31 18:17:38 +00:00
service_item['header']['background_audio'][i] = new_file
# Add the service item to the service.
2013-08-31 18:17:38 +00:00
service.append({'serviceitem': service_item})
2013-01-27 20:36:18 +00:00
self.repaint_service_list(-1, -1)
2013-01-21 07:29:43 +00:00
for file_item in write_list:
file_size = os.path.getsize(file_item)
total_size += file_size
2013-12-31 20:29:03 +00:00
self.log_debug('ServiceManager.save_file - ZIP contents size is %i bytes' % total_size)
service_content = json.dumps(service)
2013-03-29 12:53:07 +00:00
# Usual Zip file cannot exceed 2GiB, file with Zip64 cannot be extracted using unzip in UNIX.
allow_zip_64 = (total_size > 2147483648 + len(service_content))
2013-12-31 20:29:03 +00:00
self.log_debug('ServiceManager.save_file - allowZip64 is %s' % allow_zip_64)
2013-01-21 07:29:43 +00:00
zip_file = None
success = True
2013-03-07 08:05:43 +00:00
self.main_window.increment_progress_bar()
try:
2013-01-21 07:29:43 +00:00
zip_file = zipfile.ZipFile(temp_file_name, 'w', zipfile.ZIP_STORED, allow_zip_64)
2013-07-12 19:32:46 +00:00
# First we add service contents..
zip_file.writestr(service_file_name, service_content)
# Finally add all the listed media files.
for write_from in write_list:
2013-07-12 19:32:46 +00:00
zip_file.write(write_from, write_from)
for audio_from, audio_to in audio_files:
2013-08-31 18:17:38 +00:00
if audio_from.startswith('audio'):
2013-03-29 12:53:07 +00:00
# When items are saved, they get new unique_identifier. Let's copy the file to the new location.
# Unused files can be ignored, OpenLP automatically cleans up the service manager dir on exit.
2013-12-26 08:56:53 +00:00
audio_from = os.path.join(self.service_path, audio_from)
save_file = os.path.join(self.service_path, audio_to)
save_path = os.path.split(save_file)[0]
2012-07-07 14:54:14 +00:00
check_directory_exists(save_path)
if not os.path.exists(save_file):
shutil.copy(audio_from, save_file)
2013-07-12 19:32:46 +00:00
zip_file.write(audio_from, audio_to)
except IOError:
2013-12-31 20:29:03 +00:00
self.log_exception('Failed to save service to disk: %s', temp_file_name)
2013-08-31 18:17:38 +00:00
self.main_window.error_message(translate('OpenLP.ServiceManager', 'Error Saving File'),
2013-12-24 21:24:52 +00:00
translate('OpenLP.ServiceManager', 'There was an error saving your file.'))
success = False
finally:
2013-01-21 07:29:43 +00:00
if zip_file:
zip_file.close()
2013-03-16 11:05:52 +00:00
self.main_window.finished_progress_bar()
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
if success:
try:
shutil.copy(temp_file_name, path_file_name)
2013-02-01 20:40:23 +00:00
except shutil.Error:
2013-01-27 07:36:04 +00:00
return self.save_file_as()
2013-03-16 11:05:52 +00:00
self.main_window.add_recent_file(path_file_name)
2013-01-27 07:36:04 +00:00
self.set_modified(False)
2012-06-09 15:46:01 +00:00
delete_file(temp_file_name)
return success
2013-01-27 07:36:04 +00:00
def save_local_file(self):
2012-09-14 16:35:07 +00:00
"""
2013-01-21 11:11:47 +00:00
Save the current service file but leave all the file references alone to point to the current machine.
This format is not transportable as it will not contain any files.
2012-09-14 16:35:07 +00:00
"""
2013-01-27 07:36:04 +00:00
if not self.file_name():
return self.save_file_as()
2013-08-31 18:17:38 +00:00
temp_file, temp_file_name = mkstemp('.oszl', 'openlp_')
2012-09-14 16:35:07 +00:00
# We don't need the file handle.
os.close(temp_file)
2013-12-31 20:29:03 +00:00
self.log_debug(temp_file_name)
2013-08-31 18:17:38 +00:00
path_file_name = str(self.file_name())
2012-09-14 16:35:07 +00:00
path, file_name = os.path.split(path_file_name)
2013-01-21 11:11:47 +00:00
base_name = os.path.splitext(file_name)[0]
service_file_name = '%s.osj' % base_name
2013-12-31 20:29:03 +00:00
self.log_debug('ServiceManager.save_file - %s', path_file_name)
2013-08-31 18:17:38 +00:00
Settings().setValue(self.main_window.service_manager_settings_section + '/last directory', path)
2012-09-14 16:35:07 +00:00
service = []
2013-02-03 19:23:12 +00:00
self.application.set_busy_cursor()
2012-09-14 16:35:07 +00:00
# Number of items + 1 to zip it
2013-03-16 11:05:52 +00:00
self.main_window.display_progress_bar(len(self.service_items) + 1)
2013-01-27 07:36:04 +00:00
for item in self.service_items:
2013-03-07 08:05:43 +00:00
self.main_window.increment_progress_bar()
2013-08-31 18:17:38 +00:00
service_item = item['service_item'].get_service_repr(self._save_lite)
2013-03-29 12:53:07 +00:00
#TODO: check for file item on save.
2013-08-31 18:17:38 +00:00
service.append({'serviceitem': service_item})
2013-03-07 08:05:43 +00:00
self.main_window.increment_progress_bar()
service_content = json.dumps(service)
2013-01-21 07:29:43 +00:00
zip_file = None
2012-09-14 16:35:07 +00:00
success = True
2013-03-07 08:05:43 +00:00
self.main_window.increment_progress_bar()
2012-09-14 16:35:07 +00:00
try:
2013-12-24 21:24:52 +00:00
zip_file = zipfile.ZipFile(temp_file_name, 'w', zipfile.ZIP_STORED, True)
2012-09-14 16:35:07 +00:00
# First we add service contents.
2013-07-12 19:26:46 +00:00
zip_file.writestr(service_file_name, service_content)
2012-09-14 16:35:07 +00:00
except IOError:
2013-12-31 20:29:03 +00:00
self.log_exception('Failed to save service to disk: %s', temp_file_name)
2013-08-31 18:17:38 +00:00
self.main_window.error_message(translate('OpenLP.ServiceManager', 'Error Saving File'),
2013-12-24 21:24:52 +00:00
translate('OpenLP.ServiceManager', 'There was an error saving your file.'))
2012-09-14 16:35:07 +00:00
success = False
finally:
2013-01-21 07:29:43 +00:00
if zip_file:
zip_file.close()
2013-03-16 11:05:52 +00:00
self.main_window.finished_progress_bar()
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
2012-09-14 16:35:07 +00:00
if success:
try:
shutil.copy(temp_file_name, path_file_name)
2013-02-01 20:40:23 +00:00
except shutil.Error:
2013-01-27 07:36:04 +00:00
return self.save_file_as()
2013-03-16 11:05:52 +00:00
self.main_window.add_recent_file(path_file_name)
2013-01-27 07:36:04 +00:00
self.set_modified(False)
2012-09-14 16:35:07 +00:00
delete_file(temp_file_name)
return success
2013-01-27 07:36:04 +00:00
def save_file_as(self):
"""
2013-03-29 12:53:07 +00:00
Get a file name and then call :func:`ServiceManager.save_file` to save the file.
"""
2013-08-31 18:17:38 +00:00
default_service_enabled = Settings().value('advanced/default service enabled')
if default_service_enabled:
2013-08-31 18:17:38 +00:00
service_day = Settings().value('advanced/default service day')
if service_day == 7:
2012-09-16 15:33:05 +00:00
local_time = datetime.now()
else:
2013-08-31 18:17:38 +00:00
service_hour = Settings().value('advanced/default service hour')
service_minute = Settings().value('advanced/default service minute')
now = datetime.now()
day_delta = service_day - now.weekday()
if day_delta < 0:
day_delta += 7
time = now + timedelta(days=day_delta)
2012-09-16 15:33:05 +00:00
local_time = time.replace(hour=service_hour, minute=service_minute)
2013-08-31 18:17:38 +00:00
default_pattern = Settings().value('advanced/default service name')
2013-01-27 07:36:04 +00:00
default_file_name = format_time(default_pattern, local_time)
else:
2013-08-31 18:17:38 +00:00
default_file_name = ''
directory = Settings().value(self.main_window.service_manager_settings_section + '/last directory')
2013-01-27 07:36:04 +00:00
path = os.path.join(directory, default_file_name)
2013-03-29 12:53:07 +00:00
# SaveAs from osz to oszl is not valid as the files will be deleted on exit which is not sensible or usable in
# the long term.
2013-08-31 18:17:38 +00:00
if self._file_name.endswith('oszl') or self.service_has_all_original_files:
file_name = QtGui.QFileDialog.getSaveFileName(self.main_window, UiStrings().SaveService, path,
2013-12-24 15:55:01 +00:00
translate('OpenLP.ServiceManager',
'OpenLP Service Files (*.osz);; OpenLP Service Files - lite '
'(*.oszl)'))
2012-11-01 16:36:45 +00:00
else:
file_name = QtGui.QFileDialog.getSaveFileName(self.main_window, UiStrings().SaveService, path,
translate('OpenLP.ServiceManager',
'OpenLP Service Files (*.osz);;'))
2013-01-27 07:36:04 +00:00
if not file_name:
return False
2013-08-31 18:17:38 +00:00
if os.path.splitext(file_name)[1] == '':
file_name += '.osz'
else:
2013-01-27 07:36:04 +00:00
ext = os.path.splitext(file_name)[1]
2013-08-31 18:17:38 +00:00
file_name.replace(ext, '.osz')
2013-01-27 07:36:04 +00:00
self.set_file_name(file_name)
self.decide_save_method()
2013-12-31 20:29:03 +00:00
def decide_save_method(self, field=None):
"""
Determine which type of save method to use.
"""
2013-01-27 07:36:04 +00:00
if not self.file_name():
return self.save_file_as()
2013-03-06 22:23:01 +00:00
if self._save_lite:
2013-01-27 07:36:04 +00:00
return self.save_local_file()
2012-09-14 16:35:07 +00:00
else:
2013-01-27 07:36:04 +00:00
return self.save_file()
2013-01-27 07:36:04 +00:00
def load_file(self, file_name):
"""
Load an existing service file
"""
2013-01-27 07:36:04 +00:00
if not file_name:
return False
2013-08-31 18:17:38 +00:00
file_name = str(file_name)
2013-01-27 07:36:04 +00:00
if not os.path.exists(file_name):
return False
zip_file = None
2013-01-27 07:36:04 +00:00
file_to = None
2013-02-03 19:23:12 +00:00
self.application.set_busy_cursor()
try:
2013-01-27 07:36:04 +00:00
zip_file = zipfile.ZipFile(file_name)
for zip_info in zip_file.infolist():
try:
2013-07-12 19:35:30 +00:00
ucs_file = zip_info.filename
except UnicodeDecodeError:
2013-12-31 20:29:03 +00:00
self.log_exception('file_name "%s" is not valid UTF-8' % zip_info.file_name)
2012-12-10 18:04:58 +00:00
critical_error_message_box(message=translate('OpenLP.ServiceManager',
2013-12-24 15:55:01 +00:00
'File is not a valid service.\n The content encoding is not UTF-8.'))
continue
2013-12-26 08:56:53 +00:00
os_file = ucs_file.replace('/', os.path.sep)
if not os_file.startswith('audio'):
os_file = os.path.split(os_file)[1]
2013-12-31 20:29:03 +00:00
self.log_debug('Extract file: %s', os_file)
2013-12-26 08:56:53 +00:00
zip_info.filename = os_file
zip_file.extract(zip_info, self.service_path)
if os_file.endswith('osj') or os_file.endswith('osd'):
p_file = os.path.join(self.service_path, os_file)
if 'p_file' in locals():
2013-08-31 18:17:38 +00:00
file_to = open(p_file, 'r')
if p_file.endswith('osj'):
items = json.load(file_to)
else:
critical_error_message_box(message=translate('OpenLP.ServiceManager',
2013-12-24 15:55:01 +00:00
'The service file you are trying to open is in an old format.\n '
'Please save it using OpenLP 2.0.2 or greater.'))
return
2013-01-27 07:36:04 +00:00
file_to.close()
self.new_file()
self.set_file_name(file_name)
2013-03-16 11:05:52 +00:00
self.main_window.display_progress_bar(len(items))
for item in items:
2013-03-07 08:05:43 +00:00
self.main_window.increment_progress_bar()
2013-01-27 07:36:04 +00:00
service_item = ServiceItem()
2013-03-06 22:23:01 +00:00
if self._save_lite:
2013-01-27 07:36:04 +00:00
service_item.set_from_service(item)
else:
2013-12-26 08:56:53 +00:00
service_item.set_from_service(item, self.service_path)
2013-01-27 07:36:04 +00:00
service_item.validate_item(self.suffixes)
if service_item.is_capable(ItemCapabilities.OnLoadUpdate):
2013-04-21 15:53:51 +00:00
new_item = Registry().get(service_item.name).service_load(service_item)
if new_item:
service_item = new_item
2013-01-27 20:36:18 +00:00
self.add_service_item(service_item, repaint=False)
2011-01-14 18:58:47 +00:00
delete_file(p_file)
2013-03-16 11:05:52 +00:00
self.main_window.add_recent_file(file_name)
2013-01-27 07:36:04 +00:00
self.set_modified(False)
Settings().setValue('servicemanager/last file', file_name)
else:
2012-12-10 18:04:58 +00:00
critical_error_message_box(message=translate('OpenLP.ServiceManager', 'File is not a valid service.'))
2013-12-31 20:29:03 +00:00
self.log_error('File contains no service data')
2011-03-24 20:37:02 +00:00
except (IOError, NameError, zipfile.BadZipfile):
2013-12-31 20:29:03 +00:00
self.log_exception('Problem loading service file %s' % file_name)
2012-12-10 18:04:58 +00:00
critical_error_message_box(message=translate('OpenLP.ServiceManager',
2013-12-24 15:55:01 +00:00
'File could not be opened because it is corrupt.'))
2011-03-15 14:38:16 +00:00
except zipfile.BadZipfile:
2013-01-27 07:36:04 +00:00
if os.path.getsize(file_name) == 0:
2013-12-31 20:29:03 +00:00
self.log_exception('Service file is zero sized: %s' % file_name)
2012-12-10 18:04:58 +00:00
QtGui.QMessageBox.information(self, translate('OpenLP.ServiceManager', 'Empty File'),
2013-12-24 15:55:01 +00:00
translate('OpenLP.ServiceManager', 'This service file does not contain '
'any data.'))
else:
2013-12-31 20:29:03 +00:00
self.log_exception('Service file is cannot be extracted as zip: %s' % file_name)
2012-12-10 18:04:58 +00:00
QtGui.QMessageBox.information(self, translate('OpenLP.ServiceManager', 'Corrupt File'),
2013-12-24 15:55:01 +00:00
translate('OpenLP.ServiceManager',
2013-12-24 21:24:52 +00:00
'This file is either corrupt or it is not an OpenLP 2 service file.'))
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
2011-03-15 14:38:16 +00:00
return
finally:
2013-01-27 07:36:04 +00:00
if file_to:
file_to.close()
if zip_file:
zip_file.close()
2013-03-16 11:05:52 +00:00
self.main_window.finished_progress_bar()
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
2013-01-27 20:36:18 +00:00
self.repaint_service_list(-1, -1)
2013-12-26 08:56:53 +00:00
def load_last_file(self):
"""
2013-03-29 12:53:07 +00:00
Load the last service item from the service manager when the service was last closed. Can be blank if there was
no service present.
"""
2013-08-31 18:17:38 +00:00
file_name = Settings().value('servicemanager/last file')
2013-01-27 07:36:04 +00:00
if file_name:
self.load_file(file_name)
2013-01-27 07:36:04 +00:00
def context_menu(self, point):
2013-01-21 11:11:47 +00:00
"""
The Right click context menu from the Serviceitem list
"""
2013-01-27 07:36:04 +00:00
item = self.service_manager_list.itemAt(point)
2010-03-12 19:39:15 +00:00
if item is None:
return
2012-05-17 20:06:23 +00:00
if item.parent():
2012-05-19 09:13:32 +00:00
pos = item.parent().data(0, QtCore.Qt.UserRole)
2010-03-05 09:24:42 +00:00
else:
2012-05-19 09:13:32 +00:00
pos = item.data(0, QtCore.Qt.UserRole)
2013-01-27 07:36:04 +00:00
service_item = self.service_items[pos - 1]
2013-01-27 10:18:06 +00:00
self.edit_action.setVisible(False)
self.create_custom_action.setVisible(False)
2013-01-27 10:18:06 +00:00
self.maintain_action.setVisible(False)
self.notes_action.setVisible(False)
self.time_action.setVisible(False)
self.auto_start_action.setVisible(False)
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_capable(ItemCapabilities.CanEdit) and service_item['service_item'].edit_id:
2013-01-27 10:18:06 +00:00
self.edit_action.setVisible(True)
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_capable(ItemCapabilities.CanMaintain):
2013-01-27 10:18:06 +00:00
self.maintain_action.setVisible(True)
2010-03-05 09:24:42 +00:00
if item.parent() is None:
2013-01-27 10:18:06 +00:00
self.notes_action.setVisible(True)
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_capable(ItemCapabilities.CanLoop) and \
len(service_item['service_item'].get_frames()) > 1:
self.auto_play_slides_menu.menuAction().setVisible(True)
2013-08-31 18:17:38 +00:00
self.auto_play_slides_once.setChecked(service_item['service_item'].auto_play_slides_once)
self.auto_play_slides_loop.setChecked(service_item['service_item'].auto_play_slides_loop)
self.timed_slide_interval.setChecked(service_item['service_item'].timed_slide_interval > 0)
if service_item['service_item'].timed_slide_interval > 0:
delay_suffix = ' %s s' % str(service_item['service_item'].timed_slide_interval)
2012-11-09 18:44:25 +00:00
else:
2013-08-31 18:17:38 +00:00
delay_suffix = ' ...'
2013-02-01 21:34:23 +00:00
self.timed_slide_interval.setText(
translate('OpenLP.ServiceManager', '&Delay between slides') + delay_suffix)
2012-12-25 20:22:41 +00:00
# TODO for future: make group explains itself more visually
2012-11-09 18:44:25 +00:00
else:
self.auto_play_slides_menu.menuAction().setVisible(False)
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_capable(ItemCapabilities.HasVariableStartTime):
2013-01-27 10:18:06 +00:00
self.time_action.setVisible(True)
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_capable(ItemCapabilities.CanAutoStartForLive):
2013-01-27 10:18:06 +00:00
self.auto_start_action.setVisible(True)
self.auto_start_action.setIcon(self.inactive)
2013-02-01 21:34:23 +00:00
self.auto_start_action.setText(translate('OpenLP.ServiceManager', '&Auto Start - inactive'))
2013-08-31 18:17:38 +00:00
if service_item['service_item'].will_auto_start:
2013-01-27 10:18:06 +00:00
self.auto_start_action.setText(translate('OpenLP.ServiceManager', '&Auto Start - active'))
self.auto_start_action.setIcon(self.active)
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_text():
2013-01-22 21:25:16 +00:00
for plugin in self.plugin_manager.plugins:
2013-08-31 18:17:38 +00:00
if plugin.name == 'custom' and plugin.status == PluginStatus.Active:
self.create_custom_action.setVisible(True)
break
2013-01-27 07:36:04 +00:00
self.theme_menu.menuAction().setVisible(False)
2011-05-27 06:29:14 +00:00
# Set up the theme menu.
2013-08-31 18:17:38 +00:00
if service_item['service_item'].is_text() and self.renderer.theme_level == ThemeLevel.Song:
2013-01-27 07:36:04 +00:00
self.theme_menu.menuAction().setVisible(True)
2011-05-27 06:29:14 +00:00
# The service item does not have a theme, check the "Default".
2013-08-31 18:17:38 +00:00
if service_item['service_item'].theme is None:
2013-01-27 10:18:06 +00:00
theme_action = self.theme_menu.defaultAction()
2011-05-26 08:15:56 +00:00
else:
2013-08-31 18:17:38 +00:00
theme_action = self.theme_menu.findChild(QtGui.QAction, service_item['service_item'].theme)
2013-01-27 10:18:06 +00:00
if theme_action is not None:
theme_action.setChecked(True)
2013-01-27 07:36:04 +00:00
self.menu.exec_(self.service_manager_list.mapToGlobal(point))
2013-12-31 07:27:07 +00:00
def on_service_item_note_form(self, field=None):
"""
Allow the service note to be edited
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
self.service_note_form.text_edit.setPlainText(self.service_items[item]['service_item'].notes)
2013-03-06 22:23:01 +00:00
if self.service_note_form.exec_():
2013-08-31 18:17:38 +00:00
self.service_items[item]['service_item'].notes = self.service_note_form.text_edit.toPlainText()
2013-01-27 20:36:18 +00:00
self.repaint_service_list(item, -1)
2013-01-27 07:36:04 +00:00
self.set_modified()
2010-03-03 17:48:37 +00:00
2013-12-31 07:27:07 +00:00
def on_start_time_form(self, field=None):
"""
Opens a dialog to type in service item notes.
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-03-06 22:23:01 +00:00
self.start_time_form.item = self.service_items[item]
if self.start_time_form.exec_():
2013-01-27 20:36:18 +00:00
self.repaint_service_list(item, -1)
2013-12-31 20:29:03 +00:00
def toggle_auto_play_slides_once(self, field=None):
2012-11-09 18:44:25 +00:00
"""
2013-03-29 12:53:07 +00:00
Toggle Auto play slide once. Inverts auto play once option for the item
2012-11-09 18:44:25 +00:00
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
service_item = self.service_items[item]['service_item']
2012-11-14 11:47:15 +00:00
service_item.auto_play_slides_once = not service_item.auto_play_slides_once
if service_item.auto_play_slides_once:
service_item.auto_play_slides_loop = False
2013-01-27 07:36:04 +00:00
self.auto_play_slides_loop.setChecked(False)
2012-11-14 11:47:15 +00:00
if service_item.auto_play_slides_once and service_item.timed_slide_interval == 0:
2013-01-24 09:18:38 +00:00
service_item.timed_slide_interval = Settings().value(
2013-08-31 18:17:38 +00:00
self.main_window.general_settings_section + '/loop delay')
2013-01-27 07:36:04 +00:00
self.set_modified()
2012-11-09 18:44:25 +00:00
2013-12-31 20:29:03 +00:00
def toggle_auto_play_slides_loop(self, field=None):
2012-11-09 18:44:25 +00:00
"""
Toggle Auto play slide loop.
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
service_item = self.service_items[item]['service_item']
2012-11-14 11:47:15 +00:00
service_item.auto_play_slides_loop = not service_item.auto_play_slides_loop
if service_item.auto_play_slides_loop:
service_item.auto_play_slides_once = False
2013-01-27 07:36:04 +00:00
self.auto_play_slides_once.setChecked(False)
2012-11-14 11:47:15 +00:00
if service_item.auto_play_slides_loop and service_item.timed_slide_interval == 0:
2013-01-24 09:18:38 +00:00
service_item.timed_slide_interval = Settings().value(
2013-08-31 18:17:38 +00:00
self.main_window.general_settings_section + '/loop delay')
2013-01-27 07:36:04 +00:00
self.set_modified()
2012-11-09 18:44:25 +00:00
2013-12-31 07:27:07 +00:00
def on_timed_slide_interval(self, field=None):
2012-11-09 18:44:25 +00:00
"""
2012-12-29 22:52:19 +00:00
Shows input dialog for enter interval in seconds for delay
2012-11-09 18:44:25 +00:00
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
service_item = self.service_items[item]['service_item']
2012-11-14 11:47:15 +00:00
if service_item.timed_slide_interval == 0:
2013-08-31 18:17:38 +00:00
timed_slide_interval = Settings().value(self.main_window.general_settings_section + '/loop delay')
2012-11-09 18:44:25 +00:00
else:
2012-11-14 11:47:15 +00:00
timed_slide_interval = service_item.timed_slide_interval
2012-12-29 22:52:19 +00:00
timed_slide_interval, ok = QtGui.QInputDialog.getInteger(self, translate('OpenLP.ServiceManager',
2013-12-24 21:24:52 +00:00
'Input delay'),
translate('OpenLP.ServiceManager',
'Delay between slides in seconds.'),
2013-12-24 15:55:01 +00:00
timed_slide_interval, 0, 180, 1)
2012-11-09 18:44:25 +00:00
if ok:
2012-11-14 11:47:15 +00:00
service_item.timed_slide_interval = timed_slide_interval
2013-01-21 11:11:47 +00:00
if service_item.timed_slide_interval != 0 and not service_item.auto_play_slides_loop \
and not service_item.auto_play_slides_once:
2012-11-14 11:47:15 +00:00
service_item.auto_play_slides_loop = True
elif service_item.timed_slide_interval == 0:
service_item.auto_play_slides_loop = False
service_item.auto_play_slides_once = False
2013-01-27 07:36:04 +00:00
self.set_modified()
2012-11-09 18:44:25 +00:00
2013-01-27 07:36:04 +00:00
def on_auto_start(self):
"""
Toggles to Auto Start Setting.
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
self.service_items[item]['service_item'].will_auto_start = \
not self.service_items[item]['service_item'].will_auto_start
2013-12-31 07:27:07 +00:00
def on_service_item_edit_form(self, field=None):
2012-08-24 20:06:56 +00:00
"""
2013-03-29 12:53:07 +00:00
Opens a dialog to edit the service item and update the service display if changes are saved.
2012-08-24 20:06:56 +00:00
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
self.service_item_edit_form.set_service_item(self.service_items[item]['service_item'])
2013-03-06 22:23:01 +00:00
if self.service_item_edit_form.exec_():
self.add_service_item(self.service_item_edit_form.get_service_item(),
2013-12-24 15:55:01 +00:00
replace=True, expand=self.service_items[item]['expanded'])
2013-02-03 09:07:31 +00:00
def preview_live(self, unique_identifier, row):
"""
2013-03-29 12:53:07 +00:00
Called by the SlideController to request a preview item be made live and allows the next preview to be updated
if relevant.
2013-02-03 09:07:31 +00:00
``unique_identifier``
Reference to the service_item
``row``
individual row number
"""
2013-01-27 07:36:04 +00:00
for sitem in self.service_items:
2013-08-31 18:17:38 +00:00
if sitem['service_item'].unique_identifier == unique_identifier:
item = self.service_manager_list.topLevelItem(sitem['order'] - 1)
2013-01-27 07:36:04 +00:00
self.service_manager_list.setCurrentItem(item)
self.make_live(int(row))
return
2013-01-27 07:36:04 +00:00
def next_item(self):
"""
Called by the SlideController to select the next service item.
"""
2013-12-26 08:56:53 +00:00
if not self.service_manager_list.selected_items():
2010-02-04 19:25:32 +00:00
return
2013-12-26 08:56:53 +00:00
selected = self.service_manager_list.selected_items()[0]
look_for = 0
service_iterator = QtGui.QTreeWidgetItemIterator(self.service_manager_list)
while service_iterator.value():
if look_for == 1 and service_iterator.value().parent() is None:
self.service_manager_list.setCurrentItem(service_iterator.value())
2013-01-27 07:36:04 +00:00
self.make_live()
return
2013-12-26 08:56:53 +00:00
if service_iterator.value() == selected:
look_for = 1
service_iterator += 1
2013-02-03 09:07:31 +00:00
def previous_item(self, last_slide=False):
2010-04-26 21:29:40 +00:00
"""
Called by the SlideController to select the previous service item.
2013-02-03 09:07:31 +00:00
``last_slide``
Is this the last slide in the service_item
2010-04-26 21:29:40 +00:00
"""
2013-12-26 08:56:53 +00:00
if not self.service_manager_list.selected_items():
2010-04-26 21:29:40 +00:00
return
2013-12-26 08:56:53 +00:00
selected = self.service_manager_list.selected_items()[0]
prev_item = None
prev_item_last_slide = None
service_iterator = QtGui.QTreeWidgetItemIterator(self.service_manager_list)
while service_iterator.value():
if service_iterator.value() == selected:
if last_slide and prev_item_last_slide:
pos = prev_item.data(0, QtCore.Qt.UserRole)
2013-08-31 18:17:38 +00:00
check_expanded = self.service_items[pos - 1]['expanded']
2013-12-26 08:56:53 +00:00
self.service_manager_list.setCurrentItem(prev_item_last_slide)
if not check_expanded:
2013-12-26 08:56:53 +00:00
self.service_manager_list.collapseItem(prev_item)
2013-01-27 07:36:04 +00:00
self.make_live()
2013-12-26 08:56:53 +00:00
self.service_manager_list.setCurrentItem(prev_item)
elif prev_item:
self.service_manager_list.setCurrentItem(prev_item)
2013-01-27 07:36:04 +00:00
self.make_live()
2010-04-26 21:29:40 +00:00
return
2013-12-26 08:56:53 +00:00
if service_iterator.value().parent() is None:
prev_item = service_iterator.value()
if service_iterator.value().parent() is prev_item:
prev_item_last_slide = service_iterator.value()
service_iterator += 1
2010-04-26 21:29:40 +00:00
2013-12-31 07:27:07 +00:00
def on_set_item(self, message, field=None):
"""
2013-03-20 20:17:00 +00:00
Called by a signal to select a specific item and make it live usually from remote.
"""
2013-01-27 07:36:04 +00:00
self.set_item(int(message))
2010-05-04 20:01:45 +00:00
2013-01-27 07:36:04 +00:00
def set_item(self, index):
"""
Makes a specific item in the service live.
"""
if 0 >= index < self.service_manager_list.topLevelItemCount():
2013-01-27 07:36:04 +00:00
item = self.service_manager_list.topLevelItem(index)
self.service_manager_list.setCurrentItem(item)
self.make_live()
2013-01-27 07:36:04 +00:00
def on_move_selection_up(self):
"""
2013-03-29 12:53:07 +00:00
Moves the cursor selection up the window. Called by the up arrow.
"""
2013-01-27 07:36:04 +00:00
item = self.service_manager_list.currentItem()
2013-12-26 08:56:53 +00:00
item_before = self.service_manager_list.item_above(item)
if item_before is None:
return
2013-12-26 08:56:53 +00:00
self.service_manager_list.setCurrentItem(item_before)
2013-01-27 07:36:04 +00:00
def on_move_selection_down(self):
"""
2013-03-29 12:53:07 +00:00
Moves the cursor selection down the window. Called by the down arrow.
"""
2013-01-27 07:36:04 +00:00
item = self.service_manager_list.currentItem()
2013-12-26 08:56:53 +00:00
item_after = self.service_manager_list.itemBelow(item)
if item_after is None:
return
2013-12-26 08:56:53 +00:00
self.service_manager_list.setCurrentItem(item_after)
2013-12-31 20:29:03 +00:00
def on_collapse_all(self, field=None):
"""
Collapse all the service items.
"""
2013-01-27 07:36:04 +00:00
for item in self.service_items:
2013-08-31 18:17:38 +00:00
item['expanded'] = False
2013-01-27 07:36:04 +00:00
self.service_manager_list.collapseAll()
def collapsed(self, item):
"""
2013-03-29 12:53:07 +00:00
Record if an item is collapsed. Used when repainting the list to get the correct state.
"""
2012-05-19 09:13:32 +00:00
pos = item.data(0, QtCore.Qt.UserRole)
2013-08-31 18:17:38 +00:00
self.service_items[pos - 1]['expanded'] = False
2013-12-31 20:29:03 +00:00
def on_expand_all(self, field=None):
"""
Collapse all the service items.
"""
2013-01-27 07:36:04 +00:00
for item in self.service_items:
2013-08-31 18:17:38 +00:00
item['expanded'] = True
2013-01-27 07:36:04 +00:00
self.service_manager_list.expandAll()
def expanded(self, item):
"""
2013-03-29 12:53:07 +00:00
Record if an item is collapsed. Used when repainting the list to get the correct state.
"""
2012-05-19 09:13:32 +00:00
pos = item.data(0, QtCore.Qt.UserRole)
2013-08-31 18:17:38 +00:00
self.service_items[pos - 1]['expanded'] = True
2013-12-31 20:29:03 +00:00
def on_service_top(self, field=None):
"""
Move the current ServiceItem to the top of the list.
"""
2013-01-27 07:36:04 +00:00
item, child = self.find_service_item()
if item < len(self.service_items) and item != -1:
temp = self.service_items[item]
self.service_items.remove(self.service_items[item])
self.service_items.insert(0, temp)
2013-01-27 20:36:18 +00:00
self.repaint_service_list(0, child)
2013-01-27 07:36:04 +00:00
self.set_modified()
2013-12-31 20:29:03 +00:00
def on_service_up(self, field=None):
"""
Move the current ServiceItem one position up in the list.
"""
2013-01-27 07:36:04 +00:00
item, child = self.find_service_item()
if item > 0:
2013-01-27 07:36:04 +00:00
temp = self.service_items[item]
self.service_items.remove(self.service_items[item])
self.service_items.insert(item - 1, temp)
2013-01-27 20:36:18 +00:00
self.repaint_service_list(item - 1, child)
2013-01-27 07:36:04 +00:00
self.set_modified()
2013-12-31 20:29:03 +00:00
def on_service_down(self, field=None):
"""
Move the current ServiceItem one position down in the list.
"""
2013-01-27 07:36:04 +00:00
item, child = self.find_service_item()
if item < len(self.service_items) and item != -1:
temp = self.service_items[item]
self.service_items.remove(self.service_items[item])
self.service_items.insert(item + 1, temp)
2013-01-27 20:36:18 +00:00
self.repaint_service_list(item + 1, child)
2013-01-27 07:36:04 +00:00
self.set_modified()
2013-12-31 20:29:03 +00:00
def on_service_end(self, field=None):
"""
Move the current ServiceItem to the bottom of the list.
"""
2013-01-27 07:36:04 +00:00
item, child = self.find_service_item()
if item < len(self.service_items) and item != -1:
temp = self.service_items[item]
self.service_items.remove(self.service_items[item])
self.service_items.insert(len(self.service_items), temp)
2013-01-27 20:36:18 +00:00
self.repaint_service_list(len(self.service_items) - 1, child)
2013-01-27 07:36:04 +00:00
self.set_modified()
2013-12-31 07:27:07 +00:00
def on_delete_from_service(self, field=None):
"""
Remove the current ServiceItem from the list.
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
if item != -1:
2013-01-27 07:36:04 +00:00
self.service_items.remove(self.service_items[item])
2013-01-27 20:36:18 +00:00
self.repaint_service_list(item - 1, -1)
2013-01-27 07:36:04 +00:00
self.set_modified()
2013-03-06 22:23:01 +00:00
def repaint_service_list(self, service_item, service_item_child):
"""
2013-03-29 12:53:07 +00:00
Clear the existing service list and prepaint all the items. This is used when moving items as the move takes
place in a supporting list, and when regenerating all the items due to theme changes.
2013-03-06 22:23:01 +00:00
``service_item``
2011-01-20 16:12:07 +00:00
The item which changed. (int)
2013-03-06 22:23:01 +00:00
``service_item_child``
The child of the ``service_item``, which will be selected. (int)
"""
# Correct order of items in array
count = 1
2012-12-08 16:20:52 +00:00
self.service_has_all_original_files = True
2013-01-27 07:36:04 +00:00
for item in self.service_items:
2013-08-31 18:17:38 +00:00
item['order'] = count
count += 1
2013-08-31 18:17:38 +00:00
if not item['service_item'].has_original_files:
2012-12-08 16:20:52 +00:00
self.service_has_all_original_files = False
# Repaint the screen
2013-01-27 07:36:04 +00:00
self.service_manager_list.clear()
2013-03-11 19:11:46 +00:00
self.service_manager_list.clearSelection()
2013-01-27 07:36:04 +00:00
for item_count, item in enumerate(self.service_items):
2013-12-26 08:56:53 +00:00
service_item_from_item = item['service_item']
tree_widget_item = QtGui.QTreeWidgetItem(self.service_manager_list)
if service_item_from_item.is_valid:
if service_item_from_item.notes:
icon = QtGui.QImage(service_item_from_item.icon)
2012-12-10 18:04:58 +00:00
icon = icon.scaled(80, 80, QtCore.Qt.KeepAspectRatio, QtCore.Qt.SmoothTransformation)
2010-05-04 20:01:45 +00:00
overlay = QtGui.QImage(':/services/service_item_notes.png')
2012-12-10 18:04:58 +00:00
overlay = overlay.scaled(80, 80, QtCore.Qt.KeepAspectRatio, QtCore.Qt.SmoothTransformation)
2010-05-04 20:01:45 +00:00
painter = QtGui.QPainter(icon)
painter.drawImage(0, 0, overlay)
painter.end()
2013-12-26 08:56:53 +00:00
tree_widget_item.setIcon(0, build_icon(icon))
elif service_item_from_item.temporary_edit:
icon = QtGui.QImage(service_item_from_item.icon)
2012-12-10 18:04:58 +00:00
icon = icon.scaled(80, 80, QtCore.Qt.KeepAspectRatio, QtCore.Qt.SmoothTransformation)
2011-12-10 08:45:17 +00:00
overlay = QtGui.QImage(':/general/general_export.png')
2012-12-10 18:04:58 +00:00
overlay = overlay.scaled(40, 40, QtCore.Qt.KeepAspectRatio, QtCore.Qt.SmoothTransformation)
2011-12-10 08:45:17 +00:00
painter = QtGui.QPainter(icon)
painter.drawImage(40, 0, overlay)
painter.end()
2013-12-26 08:56:53 +00:00
tree_widget_item.setIcon(0, build_icon(icon))
2010-05-04 20:01:45 +00:00
else:
2013-12-26 08:56:53 +00:00
tree_widget_item.setIcon(0, service_item_from_item.iconic_representation)
2010-03-06 08:00:36 +00:00
else:
2013-12-26 08:56:53 +00:00
tree_widget_item.setIcon(0, build_icon(':/general/general_delete.png'))
tree_widget_item.setText(0, service_item_from_item.get_display_title())
tips = []
2013-12-26 08:56:53 +00:00
if service_item_from_item.temporary_edit:
2013-08-31 18:17:38 +00:00
tips.append('<strong>%s:</strong> <em>%s</em>' %
2013-12-26 08:56:53 +00:00
(translate('OpenLP.ServiceManager', 'Edit'),
(translate('OpenLP.ServiceManager', 'Service copy only'))))
if service_item_from_item.theme and service_item_from_item.theme != -1:
2013-08-31 18:17:38 +00:00
tips.append('<strong>%s:</strong> <em>%s</em>' %
2013-12-26 08:56:53 +00:00
(translate('OpenLP.ServiceManager', 'Slide theme'), service_item_from_item.theme))
if service_item_from_item.notes:
2013-08-31 18:17:38 +00:00
tips.append('<strong>%s: </strong> %s' %
2013-12-26 08:56:53 +00:00
(translate('OpenLP.ServiceManager', 'Notes'), html.escape(service_item_from_item.notes)))
2013-08-31 18:17:38 +00:00
if item['service_item'].is_capable(ItemCapabilities.HasVariableStartTime):
tips.append(item['service_item'].get_media_time())
2013-12-26 08:56:53 +00:00
tree_widget_item.setToolTip(0, '<br>'.join(tips))
tree_widget_item.setData(0, QtCore.Qt.UserRole, item['order'])
tree_widget_item.setSelected(item['selected'])
# Add the children to their parent tree_widget_item.
for count, frame in enumerate(service_item_from_item.get_frames()):
child = QtGui.QTreeWidgetItem(tree_widget_item)
2013-08-31 18:17:38 +00:00
text = frame['title'].replace('\n', ' ')
2011-01-22 07:31:32 +00:00
child.setText(0, text[:40])
2012-05-17 15:13:09 +00:00
child.setData(0, QtCore.Qt.UserRole, count)
2013-03-06 22:23:01 +00:00
if service_item == item_count:
2013-08-31 18:17:38 +00:00
if item['expanded'] and service_item_child == count:
2013-01-27 07:36:04 +00:00
self.service_manager_list.setCurrentItem(child)
2013-03-06 22:23:01 +00:00
elif service_item_child == -1:
2013-12-26 08:56:53 +00:00
self.service_manager_list.setCurrentItem(tree_widget_item)
tree_widget_item.setExpanded(item['expanded'])
2013-02-01 20:40:23 +00:00
def clean_up(self):
2009-09-19 21:45:50 +00:00
"""
2013-12-26 08:56:53 +00:00
Empties the service_path of temporary files on system exit.
2009-09-19 21:45:50 +00:00
"""
2013-12-26 08:56:53 +00:00
for file_name in os.listdir(self.service_path):
file_path = os.path.join(self.service_path, file_name)
2011-01-14 18:58:47 +00:00
delete_file(file_path)
2013-12-26 08:56:53 +00:00
if os.path.exists(os.path.join(self.service_path, 'audio')):
shutil.rmtree(os.path.join(self.service_path, 'audio'), True)
def on_theme_combo_box_selected(self, current_index):
"""
Set the theme for the current service.
"""
2013-01-27 07:36:04 +00:00
self.service_theme = self.theme_combo_box.currentText()
2013-01-22 21:09:43 +00:00
self.renderer.set_service_theme(self.service_theme)
2013-08-31 18:17:38 +00:00
Settings().setValue(self.main_window.service_manager_settings_section + '/service theme', self.service_theme)
2013-12-26 08:56:53 +00:00
self.regenerate_service_items(True)
2009-06-26 16:39:16 +00:00
2013-03-14 20:21:04 +00:00
def theme_change(self):
"""
2013-03-29 12:53:07 +00:00
The theme may have changed in the settings dialog so make sure the theme combo box is in the correct state.
"""
2013-01-22 21:09:43 +00:00
visible = self.renderer.theme_level == ThemeLevel.Global
2013-01-27 07:36:04 +00:00
self.theme_label.setVisible(visible)
self.theme_combo_box.setVisible(visible)
2013-12-26 08:56:53 +00:00
def regenerate_service_items(self, changed=False):
"""
2013-03-29 12:53:07 +00:00
Rebuild the service list as things have changed and a repaint is the easiest way to do this.
"""
2013-02-03 19:23:12 +00:00
self.application.set_busy_cursor()
2010-08-26 05:01:29 +00:00
# force reset of renderer as theme data has changed
2012-12-08 16:20:52 +00:00
self.service_has_all_original_files = True
2013-01-27 07:36:04 +00:00
if self.service_items:
for item in self.service_items:
2013-08-31 18:17:38 +00:00
item['selected'] = False
2013-12-26 08:56:53 +00:00
service_iterator = QtGui.QTreeWidgetItemIterator(self.service_manager_list)
selected_item = None
while service_iterator.value():
if service_iterator.value().isSelected():
selected_item = service_iterator.value()
service_iterator += 1
if selected_item is not None:
if selected_item.parent() is None:
pos = selected_item.data(0, QtCore.Qt.UserRole)
2011-06-04 18:08:48 +00:00
else:
2013-12-26 08:56:53 +00:00
pos = selected_item.parent().data(0, QtCore.Qt.UserRole)
2013-08-31 18:17:38 +00:00
self.service_items[pos - 1]['selected'] = True
2013-12-26 08:56:53 +00:00
temp_service_items = self.service_items
2013-01-27 07:36:04 +00:00
self.service_manager_list.clear()
self.service_items = []
2013-12-26 08:56:53 +00:00
self.is_new = True
for item in temp_service_items:
2013-08-31 18:17:38 +00:00
self.add_service_item(item['service_item'], False, expand=item['expanded'], repaint=False,
2013-12-26 08:56:53 +00:00
selected=item['selected'])
2013-03-29 12:53:07 +00:00
# Set to False as items may have changed rendering does not impact the saved song so True may also be valid
if changed:
2013-01-27 07:36:04 +00:00
self.set_modified()
2011-06-04 18:08:48 +00:00
# Repaint it once only at the end
2013-01-27 20:36:18 +00:00
self.repaint_service_list(-1, -1)
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
def replace_service_item(self, new_item):
"""
2013-03-29 12:53:07 +00:00
Using the service item passed replace the one with the same edit id if found.
"""
2013-01-27 07:36:04 +00:00
for item_count, item in enumerate(self.service_items):
if item['service_item'].edit_id == new_item.edit_id and item['service_item'].name == new_item.name:
new_item.render()
new_item.merge(item['service_item'])
item['service_item'] = new_item
2013-01-27 20:36:18 +00:00
self.repaint_service_list(item_count + 1, 0)
self.live_controller.replace_service_manager_item(new_item)
2013-01-27 07:36:04 +00:00
self.set_modified()
2013-01-27 20:36:18 +00:00
def add_service_item(self, item, rebuild=False, expand=None, replace=False, repaint=True, selected=False):
"""
Add a Service item to the list
``item``
Service Item to be added
``expand``
Override the default expand settings. (Tristate)
"""
2010-11-19 18:05:49 +00:00
# if not passed set to config value
if expand is None:
2013-08-31 18:17:38 +00:00
expand = Settings().value('advanced/expand service item')
item.from_service = True
2010-04-20 22:00:55 +00:00
if replace:
2013-01-27 07:36:04 +00:00
sitem, child = self.find_service_item()
2013-08-31 18:17:38 +00:00
item.merge(self.service_items[sitem]['service_item'])
self.service_items[sitem]['service_item'] = item
2013-01-27 20:36:18 +00:00
self.repaint_service_list(sitem, child)
2013-06-16 07:54:16 +00:00
self.live_controller.replace_service_manager_item(item)
2009-10-29 09:18:26 +00:00
else:
2011-10-02 19:57:42 +00:00
item.render()
2010-09-30 05:12:06 +00:00
# nothing selected for dnd
2013-01-27 07:36:04 +00:00
if self.drop_position == 0:
if isinstance(item, list):
for inditem in item:
2013-08-31 18:17:38 +00:00
self.service_items.append({'service_item': inditem,
2013-12-26 08:56:53 +00:00
'order': len(self.service_items) + 1,
'expanded': expand, 'selected': selected})
else:
2013-08-31 18:17:38 +00:00
self.service_items.append({'service_item': item,
2013-12-26 08:56:53 +00:00
'order': len(self.service_items) + 1,
'expanded': expand, 'selected': selected})
2011-04-29 08:45:36 +00:00
if repaint:
2013-01-27 20:36:18 +00:00
self.repaint_service_list(len(self.service_items) - 1, -1)
2009-10-29 09:18:26 +00:00
else:
2013-01-27 07:36:04 +00:00
self.service_items.insert(self.drop_position,
{'service_item': item, 'order': self.drop_position,
'expanded': expand, 'selected': selected})
2013-01-27 20:36:18 +00:00
self.repaint_service_list(self.drop_position, -1)
2010-09-30 05:12:06 +00:00
# if rebuilding list make sure live is fixed.
if rebuild:
2013-06-16 07:54:16 +00:00
self.live_controller.replace_service_manager_item(item)
2013-01-27 07:36:04 +00:00
self.drop_position = 0
self.set_modified()
2013-01-27 20:36:18 +00:00
def make_preview(self):
"""
Send the current item to the Preview slide controller
"""
2013-02-03 19:23:12 +00:00
self.application.set_busy_cursor()
2013-01-27 07:36:04 +00:00
item, child = self.find_service_item()
2013-08-31 18:17:38 +00:00
if self.service_items[item]['service_item'].is_valid:
self.preview_controller.add_service_manager_item(self.service_items[item]['service_item'], child)
2010-05-06 16:49:12 +00:00
else:
2012-12-10 18:04:58 +00:00
critical_error_message_box(translate('OpenLP.ServiceManager', 'Missing Display Handler'),
2013-12-24 21:24:52 +00:00
translate('OpenLP.ServiceManager',
'Your item cannot be displayed as there is no handler to display it'))
2013-02-03 19:23:12 +00:00
self.application.set_normal_cursor()
2013-01-27 20:36:18 +00:00
def get_service_item(self):
2010-03-16 20:22:28 +00:00
"""
Send the current item to the Preview slide controller
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2010-03-16 20:22:28 +00:00
if item == -1:
return False
else:
2013-08-31 18:17:38 +00:00
return self.service_items[item]['service_item']
def on_make_live(self, field=None):
2011-03-25 06:13:42 +00:00
"""
2013-03-29 12:53:07 +00:00
Send the current item to the Live slide controller but triggered by a tablewidget click event.
2011-03-25 06:13:42 +00:00
"""
2013-01-27 07:36:04 +00:00
self.make_live()
2011-03-25 06:13:42 +00:00
2013-01-27 07:36:04 +00:00
def make_live(self, row=-1):
"""
Send the current item to the Live slide controller
``row``
2013-03-29 12:53:07 +00:00
Row number to be displayed if from preview. -1 is passed if the value is not set
"""
2013-01-27 07:36:04 +00:00
item, child = self.find_service_item()
# No items in service
if item == -1:
return
if row != -1:
child = row
2013-02-03 19:23:12 +00:00
self.application.set_busy_cursor()
2013-08-31 18:17:38 +00:00
if self.service_items[item]['service_item'].is_valid:
self.live_controller.add_service_manager_item(self.service_items[item]['service_item'], child)
if Settings().value(self.main_window.general_settings_section + '/auto preview'):
2010-05-04 20:01:45 +00:00
item += 1
2013-01-27 07:36:04 +00:00
if self.service_items and item < len(self.service_items) and \
2013-08-31 18:17:38 +00:00
self.service_items[item]['service_item'].is_capable(ItemCapabilities.CanPreview):
self.preview_controller.add_service_manager_item(self.service_items[item]['service_item'], 0)
2013-01-27 07:36:04 +00:00
next_item = self.service_manager_list.topLevelItem(item)
self.service_manager_list.setCurrentItem(next_item)
self.live_controller.preview_widget.setFocus()
2010-05-04 20:01:45 +00:00
else:
2012-12-10 18:04:58 +00:00
critical_error_message_box(translate('OpenLP.ServiceManager', 'Missing Display Handler'),
2013-12-24 15:55:01 +00:00
translate('OpenLP.ServiceManager',
'Your item cannot be displayed as the plugin required to display it '
'is missing or inactive'))
2013-02-21 17:30:18 +00:00
self.application.set_normal_cursor()
2013-12-31 20:29:03 +00:00
def remote_edit(self, field=None):
2009-10-29 09:18:26 +00:00
"""
2013-01-27 09:57:03 +00:00
Triggers a remote edit to a plugin to allow item to be edited.
2009-10-29 09:18:26 +00:00
"""
2013-02-01 20:40:23 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
if self.service_items[item]['service_item'].is_capable(ItemCapabilities.CanEdit):
new_item = Registry().get(self.service_items[item]['service_item'].name). \
on_remote_edit(self.service_items[item]['service_item'].edit_id)
2013-01-27 09:57:03 +00:00
if new_item:
2013-01-27 20:36:18 +00:00
self.add_service_item(new_item, replace=True)
2009-10-29 09:18:26 +00:00
2013-12-31 20:29:03 +00:00
def create_custom(self, field=None):
"""
Saves the current text item as a custom slide
"""
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
Registry().execute('custom_create_from_service', self.service_items[item]['service_item'])
2013-01-27 07:36:04 +00:00
def find_service_item(self):
"""
2013-12-26 08:56:53 +00:00
Finds the first selected ServiceItem in the list and returns the position of the service_item_from_item and its selected
2013-03-29 12:53:07 +00:00
child item. For example, if the third child item (in the Slidecontroller known as slide) in the second service
item is selected this will return::
2011-01-20 16:12:07 +00:00
(1, 2)
"""
2013-01-27 07:36:04 +00:00
items = self.service_manager_list.selectedItems()
2013-03-06 22:23:01 +00:00
service_item = -1
service_item_child = -1
for item in items:
2013-01-21 07:29:43 +00:00
parent_item = item.parent()
if parent_item is None:
2013-03-06 22:23:01 +00:00
service_item = item.data(0, QtCore.Qt.UserRole)
else:
2013-03-06 22:23:01 +00:00
service_item = parent_item.data(0, QtCore.Qt.UserRole)
service_item_child = item.data(0, QtCore.Qt.UserRole)
2012-05-07 08:21:21 +00:00
# Adjust for zero based arrays.
2013-03-06 22:23:01 +00:00
service_item -= 1
2012-05-07 08:21:21 +00:00
# Only process the first item on the list for this method.
break
2013-03-06 22:23:01 +00:00
return service_item, service_item_child
2013-01-27 07:36:04 +00:00
def drop_event(self, event):
"""
2013-03-29 12:53:07 +00:00
Receive drop event and trigger an internal event to get the plugins to build and push the correct service item.
The drag event payload carries the plugin name
``event``
Handle of the event pint passed
"""
link = event.mimeData()
2012-03-12 22:12:16 +00:00
if link.hasUrls():
2011-07-27 18:28:35 +00:00
event.setDropAction(QtCore.Qt.CopyAction)
event.accept()
2012-03-12 22:12:16 +00:00
for url in link.urls():
2013-01-27 07:36:04 +00:00
file_name = url.toLocalFile()
2013-08-31 18:17:38 +00:00
if file_name.endswith('.osz'):
2013-01-27 07:36:04 +00:00
self.on_load_service_clicked(file_name)
2013-08-31 18:17:38 +00:00
elif file_name.endswith('.oszl'):
2012-09-14 16:35:07 +00:00
# todo correct
2013-01-27 07:36:04 +00:00
self.on_load_service_clicked(file_name)
2012-03-12 22:12:16 +00:00
elif link.hasText():
2012-05-17 18:57:01 +00:00
plugin = link.text()
2013-01-27 07:36:04 +00:00
item = self.service_manager_list.itemAt(event.pos())
2010-09-30 05:12:06 +00:00
# ServiceManager started the drag and drop
2013-08-31 18:17:38 +00:00
if plugin == 'ServiceManager':
2013-12-26 08:56:53 +00:00
start_pos, child = self.find_service_item()
2010-11-28 13:41:52 +00:00
# If no items selected
2013-12-26 08:56:53 +00:00
if start_pos == -1:
2010-11-28 13:41:52 +00:00
return
2010-02-27 00:11:26 +00:00
if item is None:
2013-12-26 08:56:53 +00:00
end_pos = len(self.service_items)
2009-10-11 09:31:27 +00:00
else:
2013-12-26 08:56:53 +00:00
end_pos = self._get_parent_item_data(item) - 1
service_item = self.service_items[start_pos]
self.service_items.remove(service_item)
self.service_items.insert(end_pos, service_item)
self.repaint_service_list(end_pos, child)
2013-01-27 07:36:04 +00:00
self.set_modified()
2009-10-11 09:31:27 +00:00
else:
2010-09-30 05:12:06 +00:00
# we are not over anything so drop
2010-05-01 13:05:17 +00:00
replace = False
2010-06-09 17:09:32 +00:00
if item is None:
2013-01-27 07:36:04 +00:00
self.drop_position = len(self.service_items)
2010-03-14 07:56:39 +00:00
else:
# we are over something so lets investigate
2013-01-21 11:11:47 +00:00
pos = self._get_parent_item_data(item) - 1
2013-12-26 08:56:53 +00:00
service_item = self.service_items[pos]
if (plugin == service_item['service_item'].name and
service_item['service_item'].is_capable(ItemCapabilities.CanAppend)):
action = self.dnd_menu.exec_(QtGui.QCursor.pos())
2010-09-30 05:12:06 +00:00
# New action required
2013-12-26 08:56:53 +00:00
if action == self.new_action:
2013-01-27 07:36:04 +00:00
self.drop_position = self._get_parent_item_data(item)
2010-09-30 05:12:06 +00:00
# Append to existing action
2013-12-26 08:56:53 +00:00
if action == self.add_to_action:
2013-01-27 07:36:04 +00:00
self.drop_position = self._get_parent_item_data(item)
2010-05-24 22:37:20 +00:00
item.setSelected(True)
replace = True
2010-05-01 13:05:17 +00:00
else:
2013-01-27 07:36:04 +00:00
self.drop_position = self._get_parent_item_data(item)
2013-08-31 18:17:38 +00:00
Registry().execute('%s_add_service_item' % plugin, replace)
2013-01-27 07:36:04 +00:00
def update_theme_list(self, theme_list):
"""
Called from ThemeManager when the Themes have changed
``theme_list``
A list of current themes to be displayed
"""
2013-01-27 07:36:04 +00:00
self.theme_combo_box.clear()
self.theme_menu.clear()
2013-08-31 18:17:38 +00:00
self.theme_combo_box.addItem('')
2013-01-27 07:36:04 +00:00
theme_group = QtGui.QActionGroup(self.theme_menu)
theme_group.setExclusive(True)
2013-08-31 18:17:38 +00:00
theme_group.setObjectName('theme_group')
2013-03-29 12:53:07 +00:00
# Create a "Default" theme, which allows the user to reset the item's theme to the service theme or global
# theme.
2013-12-26 08:56:53 +00:00
default_theme = create_widget_action(self.theme_menu, text=UiStrings().Default, checked=False,
triggers=self.on_theme_change_action)
self.theme_menu.setDefaultAction(default_theme)
theme_group.addAction(default_theme)
2013-01-27 07:36:04 +00:00
self.theme_menu.addSeparator()
for theme in theme_list:
2013-01-27 07:36:04 +00:00
self.theme_combo_box.addItem(theme)
theme_group.addAction(create_widget_action(self.theme_menu, theme, text=theme, checked=False,
2013-12-24 15:55:01 +00:00
triggers=self.on_theme_change_action))
2013-01-27 07:36:04 +00:00
find_and_set_in_combo_box(self.theme_combo_box, self.service_theme)
2013-01-22 21:09:43 +00:00
self.renderer.set_service_theme(self.service_theme)
2013-12-26 08:56:53 +00:00
self.regenerate_service_items()
2013-12-31 20:29:03 +00:00
def on_theme_change_action(self, field=None):
2013-01-21 17:22:10 +00:00
"""
Handles theme change events
"""
2012-05-17 18:57:01 +00:00
theme = self.sender().objectName()
2011-05-27 06:29:14 +00:00
# No object name means that the "Default" theme is supposed to be used.
if not theme:
theme = None
2013-01-27 07:36:04 +00:00
item = self.find_service_item()[0]
2013-08-31 18:17:38 +00:00
self.service_items[item]['service_item'].update_theme(theme)
2013-12-26 08:56:53 +00:00
self.regenerate_service_items(True)
2013-01-21 11:11:47 +00:00
def _get_parent_item_data(self, item):
2013-01-21 17:22:10 +00:00
"""
Finds and returns the parent item for any item
"""
2013-01-21 07:29:43 +00:00
parent_item = item.parent()
if parent_item is None:
2012-05-19 09:13:32 +00:00
return item.data(0, QtCore.Qt.UserRole)
else:
2013-01-21 07:29:43 +00:00
return parent_item.data(0, QtCore.Qt.UserRole)
2010-05-04 20:01:45 +00:00
2013-01-27 07:36:04 +00:00
def print_service_order(self):
"""
Print a Service Order Sheet.
"""
2013-12-26 08:56:53 +00:00
setting_dialog = PrintServiceForm()
setting_dialog.exec_()
2013-01-22 21:09:43 +00:00
def _get_renderer(self):
"""
Adds the Renderer to the class dynamically
"""
2013-08-31 18:17:38 +00:00
if not hasattr(self, '_renderer'):
self._renderer = Registry().get('renderer')
2013-01-22 21:09:43 +00:00
return self._renderer
2013-01-22 21:25:16 +00:00
renderer = property(_get_renderer)
def _get_live_controller(self):
"""
Adds the live controller to the class dynamically
"""
2013-08-31 18:17:38 +00:00
if not hasattr(self, '_live_controller'):
self._live_controller = Registry().get('live_controller')
2013-01-22 21:25:16 +00:00
return self._live_controller
live_controller = property(_get_live_controller)
def _get_preview_controller(self):
"""
Adds the preview controller to the class dynamically
"""
2013-08-31 18:17:38 +00:00
if not hasattr(self, '_preview_controller'):
self._preview_controller = Registry().get('preview_controller')
2013-01-22 21:25:16 +00:00
return self._preview_controller
preview_controller = property(_get_preview_controller)
def _get_plugin_manager(self):
"""
Adds the plugin manager to the class dynamically
"""
2013-08-31 18:17:38 +00:00
if not hasattr(self, '_plugin_manager'):
self._plugin_manager = Registry().get('plugin_manager')
2013-01-22 21:25:16 +00:00
return self._plugin_manager
plugin_manager = property(_get_plugin_manager)
def _get_main_window(self):
"""
Adds the main window to the class dynamically
"""
2013-08-31 18:17:38 +00:00
if not hasattr(self, '_main_window'):
self._main_window = Registry().get('main_window')
return self._main_window
2013-02-03 09:07:31 +00:00
main_window = property(_get_main_window)
2013-02-03 19:23:12 +00:00
def _get_application(self):
2013-02-03 09:07:31 +00:00
"""
2013-06-21 05:16:35 +00:00
Adds the openlp to the class dynamically.
Windows needs to access the application in a dynamic manner.
2013-02-03 09:07:31 +00:00
"""
2013-08-31 18:17:38 +00:00
if os.name == 'nt':
return Registry().get('application')
2013-06-21 05:16:35 +00:00
else:
2013-08-31 18:17:38 +00:00
if not hasattr(self, '_application'):
self._application = Registry().get('application')
2013-06-21 05:16:35 +00:00
return self._application
2013-02-03 09:07:31 +00:00
2013-02-03 19:23:12 +00:00
application = property(_get_application)