openlp/openlp/plugins/songs/lib/mediaitem.py

384 lines
17 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2009-12-31 12:52:01 +00:00
# Copyright (c) 2008-2010 Raoul Snyman #
# Portions copyright (c) 2008-2010 Tim Bentley, Jonathan Corwin, Michael #
2010-03-21 23:58:01 +00:00
# Gorven, Scott Guerrieri, Christian Richter, Maikel Stuivenberg, Martin #
# Thompson, Jon Tibble, Carsten Tinggaard #
# --------------------------------------------------------------------------- #
# This program is free software; you can redistribute it and/or modify it #
# under the terms of the GNU General Public License as published by the Free #
# Software Foundation; version 2 of the License. #
# #
# This program is distributed in the hope that it will be useful, but WITHOUT #
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or #
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for #
# more details. #
# #
# You should have received a copy of the GNU General Public License along #
# with this program; if not, write to the Free Software Foundation, Inc., 59 #
# Temple Place, Suite 330, Boston, MA 02111-1307 USA #
###############################################################################
import logging
from PyQt4 import QtCore, QtGui
2009-10-24 16:40:36 +00:00
from openlp.core.lib import MediaManagerItem, SongXMLParser, \
2010-06-24 15:50:40 +00:00
BaseListWithDnD, Receiver, ItemCapabilities, translate, check_item_selected
2010-04-02 12:23:40 +00:00
from openlp.plugins.songs.forms import EditSongForm, SongMaintenanceForm, \
ImportWizardForm
2010-06-12 23:00:14 +00:00
from openlp.plugins.songs.lib.db import Song
2010-02-27 15:31:23 +00:00
log = logging.getLogger(__name__)
class SongListView(BaseListWithDnD):
def __init__(self, parent=None):
2009-08-24 20:05:46 +00:00
self.PluginName = u'Songs'
BaseListWithDnD.__init__(self, parent)
class SongMediaItem(MediaManagerItem):
"""
This is the custom media manager item for Songs.
"""
log.info(u'Song Media Item loaded')
def __init__(self, parent, icon, title):
2009-10-28 01:36:24 +00:00
self.PluginNameShort = u'Song'
2009-09-26 07:13:59 +00:00
self.IconPath = u'songs/song'
self.ListViewWithDnD_class = SongListView
MediaManagerItem.__init__(self, parent, icon, title)
2010-04-03 23:00:05 +00:00
self.edit_song_form = EditSongForm(self, self.parent.manager)
2010-04-06 19:16:14 +00:00
self.singleServiceItem = False
#self.edit_song_form = EditSongForm(self.parent.manager, self)
2009-09-15 18:56:56 +00:00
self.song_maintenance_form = SongMaintenanceForm(
2010-04-02 12:23:40 +00:00
self.parent.manager, self)
2009-11-03 19:21:58 +00:00
# Holds information about whether the edit is remotly triggered and
# which Song is required.
2009-11-03 06:15:35 +00:00
self.remoteSong = -1
def initPluginNameVisible(self):
self.PluginNameVisible = translate('SongsPlugin.MediaItem', 'Song')
2009-09-26 09:11:39 +00:00
def requiredIcons(self):
MediaManagerItem.requiredIcons(self)
2009-09-26 07:13:59 +00:00
def addEndHeaderBar(self):
self.addToolbarSeparator()
## Song Maintenance Button ##
self.addToolbarButton(
translate('SongsPlugin.MediaItem', 'Song Maintenance'),
2010-06-21 18:28:36 +00:00
translate('SongsPlugin.MediaItem',
'Maintain the lists of authors, topics and books'),
':/songs/song_maintenance.png', self.onSongMaintenanceClick)
self.PageLayout.setSpacing(4)
self.SearchLayout = QtGui.QFormLayout()
self.SearchLayout.setMargin(0)
2009-09-15 18:56:56 +00:00
self.SearchLayout.setSpacing(4)
self.SearchLayout.setObjectName(u'SearchLayout')
self.SearchTextLabel = QtGui.QLabel(self)
self.SearchTextLabel.setAlignment(
QtCore.Qt.AlignBottom|QtCore.Qt.AlignLeading|QtCore.Qt.AlignLeft)
self.SearchTextLabel.setObjectName(u'SearchTextLabel')
self.SearchLayout.setWidget(
0, QtGui.QFormLayout.LabelRole, self.SearchTextLabel)
self.SearchTextEdit = QtGui.QLineEdit(self)
self.SearchTextEdit.setObjectName(u'SearchTextEdit')
self.SearchLayout.setWidget(
0, QtGui.QFormLayout.FieldRole, self.SearchTextEdit)
self.SearchTypeLabel = QtGui.QLabel(self)
self.SearchTypeLabel.setAlignment(
QtCore.Qt.AlignBottom|QtCore.Qt.AlignLeading|QtCore.Qt.AlignLeft)
self.SearchTypeLabel.setObjectName(u'SearchTypeLabel')
self.SearchLayout.setWidget(
1, QtGui.QFormLayout.LabelRole, self.SearchTypeLabel)
self.SearchTypeComboBox = QtGui.QComboBox(self)
self.SearchTypeComboBox.setObjectName(u'SearchTypeComboBox')
self.SearchLayout.setWidget(
1, QtGui.QFormLayout.FieldRole, self.SearchTypeComboBox)
self.PageLayout.addLayout(self.SearchLayout)
self.SearchButtonLayout = QtGui.QHBoxLayout()
self.SearchButtonLayout.setMargin(0)
self.SearchButtonLayout.setSpacing(4)
self.SearchButtonLayout.setObjectName(u'SearchButtonLayout')
self.SearchButtonSpacer = QtGui.QSpacerItem(40, 20,
QtGui.QSizePolicy.Expanding, QtGui.QSizePolicy.Minimum)
self.SearchButtonLayout.addItem(self.SearchButtonSpacer)
self.SearchTextButton = QtGui.QPushButton(self)
self.SearchTextButton.setObjectName(u'SearchTextButton')
self.SearchButtonLayout.addWidget(self.SearchTextButton)
self.ClearTextButton = QtGui.QPushButton(self)
self.ClearTextButton.setObjectName(u'ClearTextButton')
self.SearchButtonLayout.addWidget(self.ClearTextButton)
self.PageLayout.addLayout(self.SearchButtonLayout)
# Signals and slots
QtCore.QObject.connect(self.SearchTextEdit,
QtCore.SIGNAL(u'returnPressed()'), self.onSearchTextButtonClick)
QtCore.QObject.connect(self.SearchTextButton,
QtCore.SIGNAL(u'pressed()'), self.onSearchTextButtonClick)
QtCore.QObject.connect(self.ClearTextButton,
QtCore.SIGNAL(u'pressed()'), self.onClearTextButtonClick)
QtCore.QObject.connect(self.SearchTextEdit,
2009-09-15 18:56:56 +00:00
QtCore.SIGNAL(u'textChanged(const QString&)'),
self.onSearchTextEditChanged)
2009-08-26 05:00:19 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2010-04-16 07:31:01 +00:00
QtCore.SIGNAL(u'songs_load_list'), self.onSearchTextButtonClick)
2009-10-17 05:47:17 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
QtCore.SIGNAL(u'config_updated'), self.configUpdated)
2009-10-24 13:07:41 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2010-04-16 07:31:01 +00:00
QtCore.SIGNAL(u'songs_preview'), self.onPreviewClick)
2009-10-29 09:18:26 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2010-04-16 07:31:01 +00:00
QtCore.SIGNAL(u'songs_edit'), self.onRemoteEdit)
2009-10-29 09:18:26 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2010-04-20 22:00:55 +00:00
QtCore.SIGNAL(u'songs_edit_clear'), self.onRemoteEditClear)
2009-10-17 05:47:17 +00:00
def configUpdated(self):
2010-04-27 16:27:57 +00:00
self.searchAsYouType = QtCore.QSettings().value(
self.settingsSection + u'/search as type',
2010-04-28 14:17:42 +00:00
QtCore.QVariant(u'False')).toBool()
2009-08-26 05:00:19 +00:00
def retranslateUi(self):
self.SearchTextLabel.setText(
translate('SongsPlugin.MediaItem', 'Search:'))
self.SearchTypeLabel.setText(
translate('SongsPlugin.MediaItem', 'Type:'))
self.ClearTextButton.setText(
translate('SongsPlugin.MediaItem', 'Clear'))
self.SearchTextButton.setText(
translate('SongsPlugin.MediaItem', 'Search'))
def initialise(self):
self.SearchTypeComboBox.addItem(
translate('SongsPlugin.MediaItem', 'Titles'))
self.SearchTypeComboBox.addItem(
translate('SongsPlugin.MediaItem', 'Lyrics'))
self.SearchTypeComboBox.addItem(
translate('SongsPlugin.MediaItem', 'Authors'))
2009-10-17 05:47:17 +00:00
self.configUpdated()
2009-06-21 17:45:59 +00:00
def onSearchTextButtonClick(self):
search_keywords = unicode(self.SearchTextEdit.displayText())
2009-09-21 23:11:50 +00:00
search_results = []
2009-06-21 17:45:59 +00:00
search_type = self.SearchTypeComboBox.currentIndex()
if search_type == 0:
log.debug(u'Titles Search')
2010-04-02 12:23:40 +00:00
search_results = self.parent.manager.search_song_title(
2009-09-15 18:56:56 +00:00
search_keywords)
2009-06-21 17:45:59 +00:00
self.displayResultsSong(search_results)
elif search_type == 1:
log.debug(u'Lyrics Search')
2010-04-02 12:23:40 +00:00
search_results = self.parent.manager.search_song_lyrics(
2009-09-15 18:56:56 +00:00
search_keywords)
2009-06-21 17:45:59 +00:00
self.displayResultsSong(search_results)
elif search_type == 2:
log.debug(u'Authors Search')
2010-04-02 12:23:40 +00:00
search_results = self.parent.manager.get_song_from_author(
2009-09-15 18:56:56 +00:00
search_keywords)
2009-06-21 17:45:59 +00:00
self.displayResultsAuthor(search_results)
2009-11-03 06:15:35 +00:00
#Called to redisplay the song list screen edith from a search
#or from the exit of the Song edit dialog. If remote editing is active
#Trigger it and clean up so it will not update again.
if self.remoteTriggered == u'L':
self.onAddClick()
if self.remoteTriggered == u'P':
self.onPreviewClick()
self.onRemoteEditClear()
2009-06-21 17:45:59 +00:00
def displayResultsSong(self, searchresults):
log.debug(u'display results Song')
2009-06-27 19:55:55 +00:00
self.ListView.clear()
for song in searchresults:
author_list = u''
for author in song.authors:
if author_list != u'':
author_list = author_list + u', '
author_list = author_list + author.display_name
2010-06-05 23:05:53 +00:00
song_title = unicode(song.title)
song_detail = u'%s (%s)' % (song_title, author_list)
song_name = QtGui.QListWidgetItem(song_detail)
song_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(song.id))
2009-06-27 19:55:55 +00:00
self.ListView.addItem(song_name)
2009-06-21 17:45:59 +00:00
def displayResultsAuthor(self, searchresults):
log.debug(u'display results Author')
2009-06-27 19:55:55 +00:00
self.ListView.clear()
2009-06-21 17:45:59 +00:00
for author in searchresults:
for song in author.songs:
song_detail = unicode(
translate('SongsPlugin.MediaItem', '%s (%s)')) % \
2010-06-06 23:23:29 +00:00
(author.display_name, song.title)
2009-06-21 17:45:59 +00:00
song_name = QtGui.QListWidgetItem(song_detail)
song_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(song.id))
2009-06-27 19:55:55 +00:00
self.ListView.addItem(song_name)
2009-06-21 17:45:59 +00:00
def onClearTextButtonClick(self):
"""
Clear the search text.
"""
self.SearchTextEdit.clear()
self.onSearchTextButtonClick()
def onSearchTextEditChanged(self, text):
2009-11-03 06:15:35 +00:00
"""
If search as type enabled invoke the search on each key press.
If the Lyrics are being searched do not start till 7 characters
have been entered.
"""
2009-10-17 05:47:17 +00:00
if self.searchAsYouType:
search_length = 1
if self.SearchTypeComboBox.currentIndex() == 1:
search_length = 7
if len(text) > search_length:
self.onSearchTextButtonClick()
2010-04-02 12:23:40 +00:00
def onImportClick(self):
2010-05-17 20:48:16 +00:00
songimportform = ImportWizardForm(self, self.parent.manager,
self.parent)
2010-04-02 12:23:40 +00:00
songimportform.exec_()
2009-09-26 07:13:59 +00:00
def onNewClick(self):
2009-06-12 17:20:40 +00:00
self.edit_song_form.newSong()
self.edit_song_form.exec_()
def onEditAuthorClick(self):
self.authors_form.load_form()
self.authors_form.exec_()
def onEditTopicClick(self):
self.topics_form.load_form()
self.topics_form.exec_()
def onEditBookClick(self):
self.song_book_form.load_form()
self.song_book_form.exec_()
def onSongMaintenanceClick(self):
self.song_maintenance_form.exec_()
2009-10-29 09:18:26 +00:00
def onRemoteEditClear(self):
2009-11-03 06:15:35 +00:00
self.remoteTriggered = None
self.remoteSong = -1
2009-10-29 09:18:26 +00:00
def onRemoteEdit(self, songid):
2009-11-03 06:15:35 +00:00
"""
Called by ServiceManager or SlideController by event passing
the Song Id in the payload along with an indicator to say which
type of display is required.
"""
fields = songid.split(u':')
2010-06-12 23:00:14 +00:00
valid = self.parent.manager.get_object(Song, fields[1])
2009-11-03 19:01:53 +00:00
if valid:
2009-11-03 06:15:35 +00:00
self.remoteSong = fields[1]
self.remoteTriggered = fields[0]
self.edit_song_form.loadSong(fields[1], (fields[0] == u'P'))
2009-10-29 09:18:26 +00:00
self.edit_song_form.exec_()
def onEditClick(self):
"""
Edit a song
"""
2010-06-24 15:50:40 +00:00
if check_item_selected(self.ListView, translate('SongsPlugin.MediaItem',
2010-06-21 18:28:36 +00:00
'You must select an item to edit.')):
item = self.ListView.currentItem()
item_id = (item.data(QtCore.Qt.UserRole)).toInt()[0]
2009-11-03 06:15:35 +00:00
self.edit_song_form.loadSong(item_id, False)
self.edit_song_form.exec_()
2009-09-26 07:13:59 +00:00
def onDeleteClick(self):
"""
Remove a song from the list and database
"""
2010-06-24 15:50:40 +00:00
if check_item_selected(self.ListView, translate('SongsPlugin.MediaItem',
2010-06-21 18:28:36 +00:00
'You must select an item to delete.')):
items = self.ListView.selectedIndexes()
2010-04-03 19:17:37 +00:00
if len(items) == 1:
2010-06-21 18:28:36 +00:00
del_message = translate('SongsPlugin.MediaItem',
'Delete song?')
2010-04-03 19:17:37 +00:00
else:
del_message = unicode(
2010-06-21 18:28:36 +00:00
translate('SongsPlugin.MediaItem',
'Delete %d songs?')) % len(items)
2010-04-04 08:26:53 +00:00
ans = QtGui.QMessageBox.question(self,
translate('SongsPlugin.MediaItem', 'Delete Confirmation'),
del_message,
2010-04-03 19:17:37 +00:00
QtGui.QMessageBox.StandardButtons(QtGui.QMessageBox.Ok|
QtGui.QMessageBox.Cancel),
QtGui.QMessageBox.Ok)
if ans == QtGui.QMessageBox.Cancel:
2010-04-04 13:53:39 +00:00
return
2010-04-03 19:17:37 +00:00
for item in items:
2010-04-02 23:24:51 +00:00
item_id = (item.data(QtCore.Qt.UserRole)).toInt()[0]
2010-06-12 23:00:14 +00:00
self.parent.manager.delete_object(Song, item_id)
2010-04-03 19:17:37 +00:00
self.onSearchTextButtonClick()
def generateSlideData(self, service_item, item=None):
raw_footer = []
2009-09-18 16:06:39 +00:00
author_list = u''
author_audit = []
2010-03-21 15:55:45 +00:00
ccli = u''
if item is None:
2010-04-04 13:53:39 +00:00
if self.remoteTriggered is None:
item = self.ListView.currentItem()
if item is None:
return False
item_id = (item.data(QtCore.Qt.UserRole)).toInt()[0]
else:
item_id = self.remoteSong
2009-10-29 09:18:26 +00:00
else:
2010-04-04 13:53:39 +00:00
item_id = (item.data(QtCore.Qt.UserRole)).toInt()[0]
2010-04-03 07:10:31 +00:00
service_item.add_capability(ItemCapabilities.AllowsEdit)
service_item.add_capability(ItemCapabilities.AllowsPreview)
2010-04-08 16:00:04 +00:00
service_item.add_capability(ItemCapabilities.AllowsLoop)
2010-06-12 23:00:14 +00:00
song = self.parent.manager.get_object(Song, item_id)
service_item.theme = song.theme_name
2009-10-29 09:18:26 +00:00
service_item.editId = item_id
if song.lyrics.startswith(u'<?xml version='):
2009-11-12 00:44:26 +00:00
songXML = SongXMLParser(song.lyrics)
verseList = songXML.get_verses()
#no verse list or only 1 space (in error)
2010-04-03 23:00:05 +00:00
if not song.verse_order or not song.verse_order.strip():
for verse in verseList:
verseTag = u'%s:%s' % (
verse[0][u'type'], verse[0][u'label'])
2010-06-12 20:22:58 +00:00
service_item.add_from_text(
verse[1][:30], unicode(verse[1]), verseTag)
else:
#Loop through the verse list and expand the song accordingly.
for order in song.verse_order.upper().split(u' '):
if len(order) == 0:
break
for verse in verseList:
2010-06-19 21:54:53 +00:00
if verse[0][u'type'][0] == order[0] and \
(verse[0][u'label'] == order[1:] or not order[1:]):
2010-06-05 16:06:39 +00:00
verseTag = u'%s:%s' % \
(verse[0][u'type'], verse[0][u'label'])
2010-06-12 20:22:58 +00:00
service_item.add_from_text(
verse[1][:30], verse[1], verseTag)
else:
verses = song.lyrics.split(u'\n\n')
for slide in verses:
service_item.add_from_text(slide[:30], unicode(slide))
service_item.title = song.title
for author in song.authors:
2009-09-18 16:06:39 +00:00
if len(author_list) > 1:
author_list = author_list + u', '
author_list = author_list + unicode(author.display_name)
author_audit.append(unicode(author.display_name))
2009-09-25 23:06:54 +00:00
if song.ccli_number is None or len(song.ccli_number) == 0:
2010-04-27 16:27:57 +00:00
ccli = self.parent.settings_form.GeneralTab.CCLINumber
else:
2009-11-30 18:29:22 +00:00
ccli = unicode(song.ccli_number)
raw_footer.append(song.title)
raw_footer.append(author_list)
raw_footer.append(song.copyright )
2009-09-15 18:56:56 +00:00
raw_footer.append(unicode(
translate('SongsPlugin.MediaItem', 'CCLI Licence: ') + ccli))
service_item.raw_footer = raw_footer
2009-10-30 17:44:16 +00:00
service_item.audit = [
song.title, author_audit, song.copyright, song.ccli_number
]
2010-04-02 12:23:40 +00:00
return True