openlp/openlp/plugins/songs/forms/editsongform.py

970 lines
43 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 #
# --------------------------------------------------------------------------- #
2012-12-30 19:41:24 +00:00
# Copyright (c) 2008-2013 Raoul Snyman #
# Portions copyright (c) 2008-2013 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, #
2012-11-07 21:37:01 +00:00
# Frode Woldsund, Martin Zibricky #
# --------------------------------------------------------------------------- #
# 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
2009-11-26 18:41:39 +00:00
import re
2011-08-24 11:53:22 +00:00
import os
import shutil
2009-09-25 00:43:42 +00:00
from PyQt4 import QtCore, QtGui
2012-02-16 20:36:35 +00:00
from openlp.core.lib import PluginStatus, Receiver, MediaType, translate, \
2012-07-07 14:54:14 +00:00
create_separated_list, check_directory_exists
from openlp.core.lib.ui import UiStrings, set_case_insensitive_completer, \
2011-04-10 18:54:26 +00:00
critical_error_message_box, find_and_set_in_combo_box
2011-08-24 11:53:22 +00:00
from openlp.core.utils import AppLocation
from openlp.plugins.songs.forms import EditVerseForm, MediaFilesForm
2011-03-14 20:36:30 +00:00
from openlp.plugins.songs.lib import SongXML, VerseType, clean_song
from openlp.plugins.songs.lib.db import Book, Song, Author, Topic, MediaFile
from openlp.plugins.songs.lib.ui import SongStrings
from editsongdialog import Ui_EditSongDialog
2010-02-27 09:55:44 +00:00
log = logging.getLogger(__name__)
class EditSongForm(QtGui.QDialog, Ui_EditSongDialog):
"""
Class to manage the editing of a song
"""
2010-02-27 09:55:44 +00:00
log.info(u'%s EditSongForm loaded', __name__)
def __init__(self, mediaitem, parent, manager):
"""
Constructor
"""
QtGui.QDialog.__init__(self, parent)
self.mediaitem = mediaitem
self.song = None
# can this be automated?
2010-05-18 19:29:40 +00:00
self.width = 400
self.setupUi(self)
# Connecting signals and slots
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.authorAddButton,
QtCore.SIGNAL(u'clicked()'), self.onAuthorAddButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.authorRemoveButton,
QtCore.SIGNAL(u'clicked()'), self.onAuthorRemoveButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.authorsListView,
2009-09-21 00:12:47 +00:00
QtCore.SIGNAL(u'itemClicked(QListWidgetItem*)'),
self.onAuthorsListViewClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.topicAddButton,
QtCore.SIGNAL(u'clicked()'), self.onTopicAddButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.topicRemoveButton,
QtCore.SIGNAL(u'clicked()'), self.onTopicRemoveButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.topicsListView,
2009-09-21 00:12:47 +00:00
QtCore.SIGNAL(u'itemClicked(QListWidgetItem*)'),
self.onTopicListViewClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.copyrightInsertButton,
QtCore.SIGNAL(u'clicked()'), self.onCopyrightInsertButtonTriggered)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.verseAddButton,
QtCore.SIGNAL(u'clicked()'), self.onVerseAddButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.verseListWidget,
2009-11-04 01:16:15 +00:00
QtCore.SIGNAL(u'doubleClicked(QModelIndex)'),
self.onVerseEditButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.verseEditButton,
QtCore.SIGNAL(u'clicked()'), self.onVerseEditButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.verseEditAllButton,
2009-09-04 17:19:54 +00:00
QtCore.SIGNAL(u'clicked()'), self.onVerseEditAllButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.verseDeleteButton,
QtCore.SIGNAL(u'clicked()'), self.onVerseDeleteButtonClicked)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.verseListWidget,
2010-05-17 18:39:28 +00:00
QtCore.SIGNAL(u'itemClicked(QTableWidgetItem*)'),
self.onVerseListViewClicked)
QtCore.QObject.connect(self.verseOrderEdit,
QtCore.SIGNAL(u'textChanged(QString)'),
self.onVerseOrderTextChanged)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.themeAddButton,
2009-09-21 02:42:59 +00:00
QtCore.SIGNAL(u'clicked()'),
2012-06-04 10:51:50 +00:00
self.mediaitem.plugin.renderer.theme_manager.onAddTheme)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.maintenanceButton,
QtCore.SIGNAL(u'clicked()'), self.onMaintenanceButtonClicked)
QtCore.QObject.connect(self.audioAddFromFileButton,
QtCore.SIGNAL(u'clicked()'), self.onAudioAddFromFileButtonClicked)
QtCore.QObject.connect(self.audioAddFromMediaButton,
QtCore.SIGNAL(u'clicked()'), self.onAudioAddFromMediaButtonClicked)
QtCore.QObject.connect(self.audioRemoveButton,
QtCore.SIGNAL(u'clicked()'), self.onAudioRemoveButtonClicked)
QtCore.QObject.connect(self.audioRemoveAllButton,
QtCore.SIGNAL(u'clicked()'), self.onAudioRemoveAllButtonClicked)
2009-08-25 05:18:09 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2010-04-16 07:31:01 +00:00
QtCore.SIGNAL(u'theme_update_list'), self.loadThemes)
self.previewButton = QtGui.QPushButton()
self.previewButton.setObjectName(u'previewButton')
self.previewButton.setText(UiStrings().SaveAndPreview)
2011-01-04 21:06:50 +00:00
self.buttonBox.addButton(
self.previewButton, QtGui.QDialogButtonBox.ActionRole)
2011-01-04 21:06:50 +00:00
QtCore.QObject.connect(self.buttonBox,
QtCore.SIGNAL(u'clicked(QAbstractButton*)'), self.onPreview)
# Create other objects and forms
self.manager = manager
self.verseForm = EditVerseForm(self)
self.mediaForm = MediaFilesForm(self)
self.initialise()
2011-01-04 21:06:50 +00:00
self.authorsListView.setSortingEnabled(False)
self.authorsListView.setAlternatingRowColors(True)
self.topicsListView.setSortingEnabled(False)
self.topicsListView.setAlternatingRowColors(True)
self.audioListWidget.setAlternatingRowColors(True)
2009-11-26 18:41:39 +00:00
self.findVerseSplit = re.compile(u'---\[\]---\n', re.UNICODE)
self.whitespace = re.compile(r'\W+', re.UNICODE)
def keyPressEvent(self, event):
"""
Reimplement the keyPressEvent to react on Return/Enter keys. When some
combo boxes have focus we do not want dialog's default action be
triggered but instead our own.
``event``
A QtGui.QKeyEvent event.
"""
if event.key() in (QtCore.Qt.Key_Enter, QtCore.Qt.Key_Return):
if self.authorsComboBox.hasFocus() and \
self.authorsComboBox.currentText():
self.onAuthorAddButtonClicked()
return
if self.topicsComboBox.hasFocus() and \
self.topicsComboBox.currentText():
self.onTopicAddButtonClicked()
return
QtGui.QDialog.keyPressEvent(self, event)
def initialise(self):
2011-01-04 21:06:50 +00:00
self.verseEditButton.setEnabled(False)
self.verseDeleteButton.setEnabled(False)
self.authorRemoveButton.setEnabled(False)
self.topicRemoveButton.setEnabled(False)
def loadAuthors(self):
authors = self.manager.get_all_objects(Author,
2010-07-18 23:37:24 +00:00
order_by_ref=Author.display_name)
2011-01-04 21:06:50 +00:00
self.authorsComboBox.clear()
self.authorsComboBox.addItem(u'')
self.authors = []
for author in authors:
2011-01-04 21:06:50 +00:00
row = self.authorsComboBox.count()
self.authorsComboBox.addItem(author.display_name)
self.authorsComboBox.setItemData(
2009-09-21 00:12:47 +00:00
row, QtCore.QVariant(author.id))
self.authors.append(author.display_name)
set_case_insensitive_completer(self.authors, self.authorsComboBox)
def loadTopics(self):
2010-12-12 08:35:02 +00:00
self.topics = []
2011-02-04 18:17:28 +00:00
self.__loadObjects(Topic, self.topicsComboBox, self.topics)
def loadBooks(self):
2010-12-12 08:35:02 +00:00
self.books = []
2011-02-04 18:17:28 +00:00
self.__loadObjects(Book, self.songBookComboBox, self.books)
def __loadObjects(self, cls, combo, cache):
objects = self.manager.get_all_objects(cls, order_by_ref=cls.name)
2011-02-04 18:17:28 +00:00
combo.clear()
combo.addItem(u'')
for object in objects:
row = combo.count()
combo.addItem(object.name)
cache.append(object.name)
combo.setItemData(row, QtCore.QVariant(object.id))
set_case_insensitive_completer(cache, combo)
2009-06-14 07:02:58 +00:00
def loadThemes(self, theme_list):
2011-01-04 21:06:50 +00:00
self.themeComboBox.clear()
self.themeComboBox.addItem(u'')
2012-06-09 15:46:01 +00:00
self.themes = theme_list
self.themeComboBox.addItems(theme_list)
set_case_insensitive_completer(self.themes, self.themeComboBox)
2009-06-14 07:02:58 +00:00
def loadMediaFiles(self):
self.audioAddFromMediaButton.setVisible(False)
for plugin in self.parent().pluginManager.plugins:
2012-06-09 15:46:01 +00:00
if plugin.name == u'media' and plugin.status == PluginStatus.Active:
self.audioAddFromMediaButton.setVisible(True)
self.mediaForm.populateFiles(
plugin.mediaItem.getList(MediaType.Audio))
break
2009-06-12 17:20:40 +00:00
def newSong(self):
log.debug(u'New Song')
2011-02-15 01:58:18 +00:00
self.song = None
self.initialise()
2011-01-04 21:06:50 +00:00
self.songTabWidget.setCurrentIndex(0)
self.titleEdit.clear()
self.alternativeEdit.clear()
self.copyrightEdit.clear()
self.verseOrderEdit.clear()
self.commentsEdit.clear()
self.CCLNumberEdit.clear()
2011-01-04 21:06:50 +00:00
self.verseListWidget.clear()
self.verseListWidget.setRowCount(0)
self.authorsListView.clear()
self.topicsListView.clear()
self.audioListWidget.clear()
2011-01-04 21:06:50 +00:00
self.titleEdit.setFocus(QtCore.Qt.OtherFocusReason)
self.songBookNumberEdit.clear()
self.loadAuthors()
self.loadTopics()
self.loadBooks()
self.loadMediaFiles()
2011-01-04 21:06:50 +00:00
self.themeComboBox.setCurrentIndex(0)
# it's a new song to preview is not possible
self.previewButton.setVisible(False)
2009-06-12 17:20:40 +00:00
def loadSong(self, id, preview=False):
"""
Loads a song.
``id``
The song id (int).
``preview``
Should be ``True`` if the song is also previewed (boolean).
"""
log.debug(u'Load Song')
self.initialise()
2011-01-04 21:06:50 +00:00
self.songTabWidget.setCurrentIndex(0)
self.loadAuthors()
self.loadTopics()
self.loadBooks()
self.loadMediaFiles()
self.song = self.manager.get_object(Song, id)
2011-01-04 21:06:50 +00:00
self.titleEdit.setText(self.song.title)
2012-04-01 14:22:32 +00:00
self.alternativeEdit.setText(
self.song.alternate_title if self.song.alternate_title else u'')
if self.song.song_book_id != 0:
book_name = self.manager.get_object(Book, self.song.song_book_id)
find_and_set_in_combo_box(
self.songBookComboBox, unicode(book_name.name))
2009-11-03 19:01:53 +00:00
if self.song.theme_name:
find_and_set_in_combo_box(
self.themeComboBox, unicode(self.song.theme_name))
2012-04-01 14:22:32 +00:00
self.copyrightEdit.setText(
self.song.copyright if self.song.copyright else u'')
self.commentsEdit.setPlainText(
self.song.comments if self.song.comments else u'')
self.CCLNumberEdit.setText(
self.song.ccli_number if self.song.ccli_number else u'')
self.songBookNumberEdit.setText(
self.song.song_number if self.song.song_number else u'')
# lazy xml migration for now
2011-01-04 21:06:50 +00:00
self.verseListWidget.clear()
self.verseListWidget.setRowCount(0)
# This is just because occasionally the lyrics come back as a "buffer"
if isinstance(self.song.lyrics, buffer):
self.song.lyrics = unicode(self.song.lyrics)
verse_tags_translated = False
2009-06-12 17:20:40 +00:00
if self.song.lyrics.startswith(u'<?xml version='):
2011-01-09 16:52:31 +00:00
songXML = SongXML()
verse_list = songXML.get_verses(self.song.lyrics)
for count, verse in enumerate(verse_list):
2011-01-04 21:06:50 +00:00
self.verseListWidget.setRowCount(
self.verseListWidget.rowCount() + 1)
2011-02-17 19:46:01 +00:00
# This silently migrates from localized verse type markup.
# If we trusted the database, this would be unnecessary.
verse_tag = verse[0][u'type']
index = None
if len(verse_tag) > 1:
index = VerseType.from_translated_string(verse_tag)
if index is None:
index = VerseType.from_string(verse_tag, None)
else:
verse_tags_translated = True
if index is None:
index = VerseType.from_tag(verse_tag)
verse[0][u'type'] = VerseType.Tags[index]
if verse[0][u'label'] == u'':
verse[0][u'label'] = u'1'
verse_def = u'%s%s' % (verse[0][u'type'], verse[0][u'label'])
2010-05-17 18:39:28 +00:00
item = QtGui.QTableWidgetItem(verse[1])
item.setData(QtCore.Qt.UserRole, QtCore.QVariant(verse_def))
2011-01-04 21:06:50 +00:00
self.verseListWidget.setItem(count, 0, item)
2009-06-12 17:20:40 +00:00
else:
verses = self.song.lyrics.split(u'\n\n')
2009-11-22 20:33:39 +00:00
for count, verse in enumerate(verses):
2011-01-04 21:06:50 +00:00
self.verseListWidget.setRowCount(
self.verseListWidget.rowCount() + 1)
2010-05-17 18:39:28 +00:00
item = QtGui.QTableWidgetItem(verse)
verse_def = u'%s%s' % \
(VerseType.Tags[VerseType.Verse], unicode(count + 1))
item.setData(QtCore.Qt.UserRole, QtCore.QVariant(verse_def))
2011-01-04 21:06:50 +00:00
self.verseListWidget.setItem(count, 0, item)
if self.song.verse_order:
# we translate verse order
translated = []
for verse_def in self.song.verse_order.split():
verse_index = None
if verse_tags_translated:
verse_index = VerseType.from_translated_tag(verse_def[0],
None)
if verse_index is None:
verse_index = VerseType.from_tag(verse_def[0])
verse_tag = VerseType.TranslatedTags[verse_index].upper()
translated.append(u'%s%s' % (verse_tag, verse_def[1:]))
self.verseOrderEdit.setText(u' '.join(translated))
else:
self.verseOrderEdit.setText(u'')
2010-05-18 19:29:40 +00:00
self.tagRows()
# clear the results
2011-01-04 21:06:50 +00:00
self.authorsListView.clear()
for author in self.song.authors:
author_name = QtGui.QListWidgetItem(unicode(author.display_name))
author_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(author.id))
2011-01-04 21:06:50 +00:00
self.authorsListView.addItem(author_name)
2009-06-12 17:20:40 +00:00
# clear the results
2011-01-04 21:06:50 +00:00
self.topicsListView.clear()
2009-06-14 06:30:09 +00:00
for topic in self.song.topics:
topic_name = QtGui.QListWidgetItem(unicode(topic.name))
topic_name.setData(QtCore.Qt.UserRole, QtCore.QVariant(topic.id))
2011-01-04 21:06:50 +00:00
self.topicsListView.addItem(topic_name)
self.audioListWidget.clear()
for media in self.song.media_files:
2011-11-26 21:26:50 +00:00
media_file = QtGui.QListWidgetItem(
os.path.split(media.file_name)[1])
media_file.setData(QtCore.Qt.UserRole,
QtCore.QVariant(media.file_name))
self.audioListWidget.addItem(media_file)
2011-01-04 21:06:50 +00:00
self.titleEdit.setFocus(QtCore.Qt.OtherFocusReason)
2011-03-17 12:32:45 +00:00
# Hide or show the preview button.
self.previewButton.setVisible(preview)
2013-06-16 17:19:32 +00:00
# Check if all verse tags are used.
self.onVerseOrderTextChanged(self.verseOrderEdit.text())
2010-05-18 19:29:40 +00:00
def tagRows(self):
"""
Tag the Song List rows based on the verse list
"""
row_label = []
for row in range(self.verseListWidget.rowCount()):
2011-01-04 21:06:50 +00:00
item = self.verseListWidget.item(row, 0)
verse_def = unicode(item.data(QtCore.Qt.UserRole).toString())
verse_tag = VerseType.translated_tag(verse_def[0])
row_def = u'%s%s' % (verse_tag, verse_def[1:])
row_label.append(row_def)
self.verseListWidget.setVerticalHeaderLabels(row_label)
2011-03-17 12:32:45 +00:00
self.verseListWidget.resizeRowsToContents()
self.verseListWidget.repaint()
2010-05-18 19:29:40 +00:00
def onAuthorAddButtonClicked(self):
2011-01-04 21:06:50 +00:00
item = int(self.authorsComboBox.currentIndex())
2012-03-22 20:03:11 +00:00
text = unicode(self.authorsComboBox.currentText()).strip(u' \r\n\t')
# This if statement is for OS X, which doesn't seem to work well with
# the QCompleter autocompletion class. See bug #812628.
2012-03-22 20:03:11 +00:00
if text in self.authors:
# Index 0 is a blank string, so add 1
item = self.authors.index(text) + 1
if item == 0 and text:
if QtGui.QMessageBox.question(self,
translate('SongsPlugin.EditSongForm', 'Add Author'),
translate('SongsPlugin.EditSongForm', 'This author does not '
'exist, do you want to add them?'),
QtGui.QMessageBox.Yes | QtGui.QMessageBox.No,
QtGui.QMessageBox.Yes) == QtGui.QMessageBox.Yes:
2010-07-21 13:13:03 +00:00
if text.find(u' ') == -1:
author = Author.populate(first_name=u'', last_name=u'',
display_name=text)
else:
author = Author.populate(first_name=text.rsplit(u' ', 1)[0],
last_name=text.rsplit(u' ', 1)[1], display_name=text)
2011-01-17 00:52:00 +00:00
self.manager.save_object(author)
2011-02-01 00:33:50 +00:00
self.__addAuthorToList(author)
self.loadAuthors()
2011-01-04 21:06:50 +00:00
self.authorsComboBox.setCurrentIndex(0)
else:
return
elif item > 0:
2011-01-04 21:06:50 +00:00
item_id = (self.authorsComboBox.itemData(item)).toInt()[0]
author = self.manager.get_object(Author, item_id)
2011-01-04 21:06:50 +00:00
if self.authorsListView.findItems(unicode(author.display_name),
QtCore.Qt.MatchExactly):
critical_error_message_box(
2011-01-15 19:24:50 +00:00
message=translate('SongsPlugin.EditSongForm',
'This author is already in the list.'))
else:
2011-02-01 00:33:50 +00:00
self.__addAuthorToList(author)
2011-01-04 21:06:50 +00:00
self.authorsComboBox.setCurrentIndex(0)
else:
QtGui.QMessageBox.warning(self, UiStrings().NISs,
translate('SongsPlugin.EditSongForm', 'You have not selected '
'a valid author. Either select an author from the list, '
'or type in a new author and click the "Add Author to '
2010-07-31 02:06:44 +00:00
'Song" button to add the new author.'))
2011-02-01 00:33:50 +00:00
def __addAuthorToList(self, author):
"""
Add an author to the author list.
"""
author_item = QtGui.QListWidgetItem(unicode(author.display_name))
author_item.setData(QtCore.Qt.UserRole, QtCore.QVariant(author.id))
self.authorsListView.addItem(author_item)
def onAuthorsListViewClicked(self):
2011-01-04 21:06:50 +00:00
if self.authorsListView.count() > 1:
self.authorRemoveButton.setEnabled(True)
def onAuthorRemoveButtonClicked(self):
2011-01-04 21:06:50 +00:00
self.authorRemoveButton.setEnabled(False)
item = self.authorsListView.currentItem()
row = self.authorsListView.row(item)
self.authorsListView.takeItem(row)
2009-06-14 06:30:09 +00:00
def onTopicAddButtonClicked(self):
2011-01-04 21:06:50 +00:00
item = int(self.topicsComboBox.currentIndex())
text = unicode(self.topicsComboBox.currentText())
if item == 0 and text:
if QtGui.QMessageBox.question(self,
translate('SongsPlugin.EditSongForm', 'Add Topic'),
translate('SongsPlugin.EditSongForm', 'This topic does not '
'exist, do you want to add it?'),
QtGui.QMessageBox.Yes | QtGui.QMessageBox.No,
QtGui.QMessageBox.Yes) == QtGui.QMessageBox.Yes:
topic = Topic.populate(name=text)
2011-01-17 00:52:00 +00:00
self.manager.save_object(topic)
topic_item = QtGui.QListWidgetItem(unicode(topic.name))
2010-06-18 22:13:10 +00:00
topic_item.setData(QtCore.Qt.UserRole,
QtCore.QVariant(topic.id))
2011-01-04 21:06:50 +00:00
self.topicsListView.addItem(topic_item)
self.loadTopics()
2011-01-04 21:06:50 +00:00
self.topicsComboBox.setCurrentIndex(0)
else:
return
elif item > 0:
2011-01-04 21:06:50 +00:00
item_id = (self.topicsComboBox.itemData(item)).toInt()[0]
topic = self.manager.get_object(Topic, item_id)
2011-01-04 21:06:50 +00:00
if self.topicsListView.findItems(unicode(topic.name),
QtCore.Qt.MatchExactly):
critical_error_message_box(
2011-01-15 19:24:50 +00:00
message=translate('SongsPlugin.EditSongForm',
'This topic is already in the list.'))
else:
topic_item = QtGui.QListWidgetItem(unicode(topic.name))
topic_item.setData(QtCore.Qt.UserRole,
QtCore.QVariant(topic.id))
2011-01-04 21:06:50 +00:00
self.topicsListView.addItem(topic_item)
self.topicsComboBox.setCurrentIndex(0)
else:
QtGui.QMessageBox.warning(self, UiStrings().NISs,
translate('SongsPlugin.EditSongForm', 'You have not selected '
'a valid topic. Either select a topic from the list, or '
'type in a new topic and click the "Add Topic to Song" '
2010-07-31 02:06:44 +00:00
'button to add the new topic.'))
2009-06-14 06:30:09 +00:00
def onTopicListViewClicked(self):
2011-01-04 21:06:50 +00:00
self.topicRemoveButton.setEnabled(True)
2009-06-14 06:30:09 +00:00
def onTopicRemoveButtonClicked(self):
2011-01-04 21:06:50 +00:00
self.topicRemoveButton.setEnabled(False)
item = self.topicsListView.currentItem()
row = self.topicsListView.row(item)
self.topicsListView.takeItem(row)
def onVerseListViewClicked(self):
2011-01-04 21:06:50 +00:00
self.verseEditButton.setEnabled(True)
self.verseDeleteButton.setEnabled(True)
def onVerseAddButtonClicked(self):
self.verseForm.setVerse(u'', True)
if self.verseForm.exec_():
after_text, verse_tag, verse_num = self.verseForm.getVerse()
verse_def = u'%s%s' % (verse_tag, verse_num)
item = QtGui.QTableWidgetItem(after_text)
item.setData(QtCore.Qt.UserRole, QtCore.QVariant(verse_def))
item.setText(after_text)
2011-01-04 21:06:50 +00:00
self.verseListWidget.setRowCount(
self.verseListWidget.rowCount() + 1)
self.verseListWidget.setItem(
2011-03-17 12:32:45 +00:00
self.verseListWidget.rowCount() - 1, 0, item)
2010-05-18 19:29:40 +00:00
self.tagRows()
# Check if all verse tags are used.
self.onVerseOrderTextChanged(self.verseOrderEdit.text())
def onVerseEditButtonClicked(self):
2011-01-04 21:06:50 +00:00
item = self.verseListWidget.currentItem()
2009-11-03 19:01:53 +00:00
if item:
temp_text = item.text()
2012-06-16 16:12:54 +00:00
verse_id = unicode(item.data(QtCore.Qt.UserRole).toString())
self.verseForm.setVerse(temp_text, True, verse_id)
if self.verseForm.exec_():
after_text, verse_tag, verse_num = self.verseForm.getVerse()
verse_def = u'%s%s' % (verse_tag, verse_num)
item.setData(QtCore.Qt.UserRole, QtCore.QVariant(verse_def))
item.setText(after_text)
# number of lines has changed, repaint the list moving the data
if len(temp_text.split(u'\n')) != len(after_text.split(u'\n')):
temp_list = []
temp_ids = []
for row in range(self.verseListWidget.rowCount()):
item = self.verseListWidget.item(row, 0)
temp_list.append(item.text())
temp_ids.append(item.data(QtCore.Qt.UserRole))
2011-01-04 21:06:50 +00:00
self.verseListWidget.clear()
for row, entry in enumerate(temp_list):
item = QtGui.QTableWidgetItem(entry, 0)
item.setData(QtCore.Qt.UserRole, temp_ids[row])
2011-01-04 21:06:50 +00:00
self.verseListWidget.setItem(row, 0, item)
2010-05-18 19:29:40 +00:00
self.tagRows()
# Check if all verse tags are used.
self.onVerseOrderTextChanged(self.verseOrderEdit.text())
2009-09-04 17:19:54 +00:00
def onVerseEditAllButtonClicked(self):
verse_list = u''
2011-01-04 21:06:50 +00:00
if self.verseListWidget.rowCount() > 0:
for row in range(self.verseListWidget.rowCount()):
2011-01-04 21:06:50 +00:00
item = self.verseListWidget.item(row, 0)
field = unicode(item.data(QtCore.Qt.UserRole).toString())
verse_tag = VerseType.translated_name(field[0])
verse_num = field[1:]
verse_list += u'---[%s:%s]---\n' % (verse_tag, verse_num)
2009-09-04 17:19:54 +00:00
verse_list += item.text()
2009-11-26 18:41:39 +00:00
verse_list += u'\n'
self.verseForm.setVerse(verse_list)
2009-09-04 17:19:54 +00:00
else:
self.verseForm.setVerse(u'')
if not self.verseForm.exec_():
2011-03-17 12:32:45 +00:00
return
verse_list = self.verseForm.getVerseAll()
2011-03-17 12:32:45 +00:00
verse_list = unicode(verse_list.replace(u'\r\n', u'\n'))
self.verseListWidget.clear()
self.verseListWidget.setRowCount(0)
for row in self.findVerseSplit.split(verse_list):
for match in row.split(u'---['):
for count, parts in enumerate(match.split(u']---\n')):
if count == 0:
2012-07-22 20:20:53 +00:00
if len(parts) == 0:
continue
2011-03-17 12:32:45 +00:00
# handling carefully user inputted versetags
separator = parts.find(u':')
if separator >= 0:
verse_name = parts[0:separator].strip()
verse_num = parts[separator+1:].strip()
else:
verse_name = parts
verse_num = u'1'
verse_index = VerseType.from_loose_input(verse_name)
verse_tag = VerseType.Tags[verse_index]
# Later we need to handle v1a as well.
#regex = re.compile(r'(\d+\w.)')
regex = re.compile(r'\D*(\d+)\D*')
match = regex.match(verse_num)
if match:
verse_num = match.group(1)
else:
verse_num = u'1'
verse_def = u'%s%s' % (verse_tag, verse_num)
else:
if parts.endswith(u'\n'):
parts = parts.rstrip(u'\n')
item = QtGui.QTableWidgetItem(parts)
item.setData(QtCore.Qt.UserRole,
QtCore.QVariant(verse_def))
self.verseListWidget.setRowCount(
self.verseListWidget.rowCount() + 1)
self.verseListWidget.setItem(
self.verseListWidget.rowCount() - 1, 0, item)
self.tagRows()
self.verseEditButton.setEnabled(False)
self.verseDeleteButton.setEnabled(False)
# Check if all verse tags are used.
self.onVerseOrderTextChanged(self.verseOrderEdit.text())
2009-09-04 17:19:54 +00:00
def onVerseDeleteButtonClicked(self):
2011-01-04 21:06:50 +00:00
self.verseListWidget.removeRow(self.verseListWidget.currentRow())
2011-03-18 19:15:45 +00:00
if not self.verseListWidget.selectedItems():
self.verseEditButton.setEnabled(False)
self.verseDeleteButton.setEnabled(False)
def onVerseOrderTextChanged(self, text):
verses = []
verse_names = []
order = self.__extractVerseOrder(text)
for index in range(self.verseListWidget.rowCount()):
verse = self.verseListWidget.item(index, 0)
verse = unicode(verse.data(QtCore.Qt.UserRole).toString())
if verse not in verse_names:
verses.append(verse)
verse_names.append(u'%s%s' % (
VerseType.translated_tag(verse[0]), verse[1:]))
verses_not_used = []
2012-03-12 22:12:16 +00:00
for verse in verses:
if not verse in order:
verses_not_used.append(verse)
2013-06-16 17:19:32 +00:00
label_text = u''
if not self.verseOrderEdit.text():
label_text = self.noVerseOrder
elif verses_not_used:
label_text = self.notAlllVersesUsed
self.warningLabel.setText(label_text)
def __extractVerseOrder(self, verse_order):
order = []
order_names = unicode(verse_order).split()
for item in order_names:
if len(item) == 1:
verse_index = VerseType.from_translated_tag(item, None)
if verse_index is not None:
order.append(VerseType.Tags[verse_index] + u'1')
else:
# it matches no verses anyway
order.append(u'')
else:
verse_index = VerseType.from_translated_tag(item[0], None)
if verse_index is None:
# it matches no verses anyway
order.append(u'')
else:
verse_tag = VerseType.Tags[verse_index]
verse_num = item[1:].lower()
order.append(verse_tag + verse_num)
return order
def __validateVerseList(self, verse_order, verse_count):
verses = []
invalid_verses = []
verse_names = []
order_names = unicode(verse_order).split()
order = self.__extractVerseOrder(verse_order)
for index in range(verse_count):
verse = self.verseListWidget.item(index, 0)
verse = unicode(verse.data(QtCore.Qt.UserRole).toString())
if verse not in verse_names:
verses.append(verse)
verse_names.append(u'%s%s' % (
VerseType.translated_tag(verse[0]), verse[1:]))
for count, item in enumerate(order):
if item not in verses:
invalid_verses.append(order_names[count])
if invalid_verses:
valid = create_separated_list(verse_names)
if len(invalid_verses) > 1:
critical_error_message_box(message=unicode(translate(
'SongsPlugin.EditSongForm', 'The verse order is invalid. '
'There are no verses corresponding to %s. Valid entries '
'are %s.')) % (u', '.join(invalid_verses), valid))
else:
critical_error_message_box(message=unicode(translate(
'SongsPlugin.EditSongForm', 'The verse order is invalid. '
'There is no verse corresponding to %s. Valid entries '
'are %s.')) % (invalid_verses[0], valid))
return len(invalid_verses) == 0
def __validateSong(self):
"""
2011-01-17 00:52:00 +00:00
Check the validity of the song.
"""
2011-02-25 17:05:01 +00:00
# This checks data in the form *not* self.song. self.song is still
2011-01-17 00:52:00 +00:00
# None at this point.
log.debug(u'Validate Song')
# Lets be nice and assume the data is correct.
2011-01-17 00:52:00 +00:00
if not self.titleEdit.text():
2011-01-04 21:06:50 +00:00
self.songTabWidget.setCurrentIndex(0)
self.titleEdit.setFocus()
critical_error_message_box(
2011-01-15 19:24:50 +00:00
message=translate('SongsPlugin.EditSongForm',
'You need to type in a song title.'))
2010-06-20 06:48:32 +00:00
return False
2011-01-04 21:06:50 +00:00
if self.verseListWidget.rowCount() == 0:
self.songTabWidget.setCurrentIndex(0)
self.verseListWidget.setFocus()
critical_error_message_box(
2011-01-15 19:24:50 +00:00
message=translate('SongsPlugin.EditSongForm',
'You need to type in at least one verse.'))
2010-06-20 06:48:32 +00:00
return False
2011-01-04 21:06:50 +00:00
if self.authorsListView.count() == 0:
self.songTabWidget.setCurrentIndex(1)
self.authorsListView.setFocus()
critical_error_message_box(
2011-01-15 19:24:50 +00:00
message=translate('SongsPlugin.EditSongForm',
2010-12-18 17:11:21 +00:00
'You need to have an author for this song.'))
return False
2011-01-17 00:52:00 +00:00
if self.verseOrderEdit.text():
result = self.__validateVerseList(self.verseOrderEdit.text(),
self.verseListWidget.rowCount())
if not result:
return False
2011-01-17 00:52:00 +00:00
text = unicode(self.songBookComboBox.currentText())
if self.songBookComboBox.findText(text, QtCore.Qt.MatchExactly) < 0:
if QtGui.QMessageBox.question(self,
translate('SongsPlugin.EditSongForm', 'Add Book'),
translate('SongsPlugin.EditSongForm', 'This song book does '
'not exist, do you want to add it?'),
QtGui.QMessageBox.Yes | QtGui.QMessageBox.No,
QtGui.QMessageBox.Yes) == QtGui.QMessageBox.Yes:
book = Book.populate(name=text, publisher=u'')
self.manager.save_object(book)
else:
return False
2010-06-20 06:48:32 +00:00
return True
def onCopyrightInsertButtonTriggered(self):
2011-01-04 21:06:50 +00:00
text = self.copyrightEdit.text()
pos = self.copyrightEdit.cursorPosition()
sign = SongStrings.CopyrightSymbol
2011-01-05 10:48:47 +00:00
text = text[:pos] + sign + text[pos:]
2011-01-04 21:06:50 +00:00
self.copyrightEdit.setText(text)
self.copyrightEdit.setFocus()
2011-01-05 10:48:47 +00:00
self.copyrightEdit.setCursorPosition(pos + len(sign))
def onMaintenanceButtonClicked(self):
temp_song_book = None
2011-01-04 21:06:50 +00:00
item = int(self.songBookComboBox.currentIndex())
text = unicode(self.songBookComboBox.currentText())
if item == 0 and text:
temp_song_book = text
2012-05-26 19:56:12 +00:00
self.mediaitem.songMaintenanceForm.exec_(True)
self.loadAuthors()
self.loadBooks()
self.loadTopics()
if temp_song_book:
2011-01-04 21:06:50 +00:00
self.songBookComboBox.setEditText(temp_song_book)
def onPreview(self, button):
2010-05-18 19:29:40 +00:00
"""
Save and Preview button clicked.
2010-05-18 19:29:40 +00:00
The Song is valid so as the plugin to add it to preview to see.
``button``
A button (QPushButton).
2010-05-18 19:29:40 +00:00
"""
log.debug(u'onPreview')
2011-02-18 19:13:37 +00:00
if unicode(button.objectName()) == u'previewButton':
self.saveSong(True)
2010-04-16 07:31:01 +00:00
Receiver.send_message(u'songs_preview')
def onAudioAddFromFileButtonClicked(self):
"""
Loads file(s) from the filesystem.
"""
filters = u'%s (*)' % UiStrings().AllFiles
filenames = QtGui.QFileDialog.getOpenFileNames(self,
translate('SongsPlugin.EditSongForm', 'Open File(s)'),
QtCore.QString(), filters)
for filename in filenames:
2011-08-24 11:53:22 +00:00
item = QtGui.QListWidgetItem(os.path.split(unicode(filename))[1])
item.setData(QtCore.Qt.UserRole, filename)
self.audioListWidget.addItem(item)
def onAudioAddFromMediaButtonClicked(self):
"""
Loads file(s) from the media plugin.
"""
2011-08-24 11:53:22 +00:00
if self.mediaForm.exec_():
for filename in self.mediaForm.getSelectedFiles():
2011-11-26 21:26:50 +00:00
item = QtGui.QListWidgetItem(
os.path.split(unicode(filename))[1])
2011-08-24 11:53:22 +00:00
item.setData(QtCore.Qt.UserRole, filename)
self.audioListWidget.addItem(item)
def onAudioRemoveButtonClicked(self):
"""
Removes a file from the list.
"""
row = self.audioListWidget.currentRow()
if row == -1:
return
self.audioListWidget.takeItem(row)
def onAudioRemoveAllButtonClicked(self):
"""
Removes all files from the list.
"""
self.audioListWidget.clear()
def onUpButtonClicked(self):
"""
Moves a file up when the user clicks the up button on the audio tab.
"""
row = self.audioListWidget.currentRow()
if row <= 0:
return
item = self.audioListWidget.takeItem(row)
self.audioListWidget.insertItem(row - 1, item)
self.audioListWidget.setCurrentRow(row - 1)
def onDownButtonClicked(self):
"""
Moves a file down when the user clicks the up button on the audio tab.
"""
row = self.audioListWidget.currentRow()
if row == -1 or row > self.audioListWidget.count() - 1:
return
item = self.audioListWidget.takeItem(row)
self.audioListWidget.insertItem(row + 1, item)
self.audioListWidget.setCurrentRow(row + 1)
2010-12-12 08:35:02 +00:00
def clearCaches(self):
"""
Free up autocompletion memory on dialog exit
"""
2011-02-01 18:05:59 +00:00
log.debug (u'SongEditForm.clearCaches')
2010-12-12 08:35:02 +00:00
self.authors = []
self.themes = []
self.books = []
self.topics = []
2011-02-01 00:33:50 +00:00
def reject(self):
2010-12-12 08:35:02 +00:00
"""
Exit Dialog and do not save
"""
2011-02-01 18:05:59 +00:00
log.debug (u'SongEditForm.reject')
2010-04-20 22:00:55 +00:00
Receiver.send_message(u'songs_edit_clear')
2010-12-12 08:35:02 +00:00
self.clearCaches()
2011-02-01 18:05:59 +00:00
QtGui.QDialog.reject(self)
2009-10-29 09:18:26 +00:00
2009-08-08 06:19:09 +00:00
def accept(self):
2010-12-12 08:35:02 +00:00
"""
Exit Dialog and save song if valid
2010-12-12 08:35:02 +00:00
"""
2011-02-01 18:05:59 +00:00
log.debug(u'SongEditForm.accept')
2010-12-12 08:35:02 +00:00
self.clearCaches()
if self.__validateSong():
2011-01-17 00:52:00 +00:00
self.saveSong()
self.song = None
2011-02-01 18:05:59 +00:00
QtGui.QDialog.accept(self)
def saveSong(self, preview=False):
2010-09-17 19:02:25 +00:00
"""
Get all the data from the widgets on the form, and then save it to the
2011-02-25 17:05:01 +00:00
database. The form has been validated and all reference items
2011-01-17 00:52:00 +00:00
(Authors, Books and Topics) have been saved before this function is
called.
``preview``
Should be ``True`` if the song is also previewed (boolean).
2010-09-17 19:02:25 +00:00
"""
2011-02-25 17:05:01 +00:00
# The Song() assignment. No database calls should be made while a
2011-01-17 00:52:00 +00:00
# Song() is in a partially complete state.
if not self.song:
self.song = Song()
2011-01-04 21:06:50 +00:00
self.song.title = unicode(self.titleEdit.text())
self.song.alternate_title = unicode(self.alternativeEdit.text())
self.song.copyright = unicode(self.copyrightEdit.text())
2011-03-17 12:32:45 +00:00
# Values will be set when cleaning the song.
2011-03-14 20:36:30 +00:00
self.song.search_title = u''
2011-03-15 17:52:42 +00:00
self.song.search_lyrics = u''
2011-03-15 20:12:28 +00:00
self.song.verse_order = u''
2011-01-04 21:06:50 +00:00
self.song.comments = unicode(self.commentsEdit.toPlainText())
ordertext = unicode(self.verseOrderEdit.text())
order = []
for item in ordertext.split():
2011-03-14 20:36:30 +00:00
verse_tag = VerseType.Tags[VerseType.from_translated_tag(item[0])]
verse_num = item[1:].lower()
order.append(u'%s%s' % (verse_tag, verse_num))
self.song.verse_order = u' '.join(order)
2010-06-05 23:05:53 +00:00
self.song.ccli_number = unicode(self.CCLNumberEdit.text())
self.song.song_number = unicode(self.songBookNumberEdit.text())
2011-01-04 21:06:50 +00:00
book_name = unicode(self.songBookComboBox.currentText())
2010-09-25 11:55:01 +00:00
if book_name:
self.song.book = self.manager.get_object_filtered(Book,
2010-09-25 11:55:01 +00:00
Book.name == book_name)
else:
self.song.book = None
2011-01-04 21:06:50 +00:00
theme_name = unicode(self.themeComboBox.currentText())
if theme_name:
self.song.theme_name = theme_name
else:
self.song.theme_name = None
2011-03-17 12:32:45 +00:00
self._processLyrics()
2011-01-17 00:52:00 +00:00
self.song.authors = []
2011-08-24 11:53:22 +00:00
for row in xrange(self.authorsListView.count()):
2011-01-17 00:52:00 +00:00
item = self.authorsListView.item(row)
authorId = (item.data(QtCore.Qt.UserRole)).toInt()[0]
author = self.manager.get_object(Author, authorId)
if author is not None:
self.song.authors.append(author)
2011-01-17 00:52:00 +00:00
self.song.topics = []
2011-08-24 11:53:22 +00:00
for row in xrange(self.topicsListView.count()):
2011-01-17 00:52:00 +00:00
item = self.topicsListView.item(row)
topicId = (item.data(QtCore.Qt.UserRole)).toInt()[0]
topic = self.manager.get_object(Topic, topicId)
if topic is not None:
self.song.topics.append(topic)
# Save the song here because we need a valid id for the audio files.
2011-08-24 11:53:22 +00:00
clean_song(self.manager, self.song)
self.manager.save_object(self.song)
audio_files = map(lambda a: a.file_name, self.song.media_files)
log.debug(audio_files)
2011-08-24 11:53:22 +00:00
save_path = os.path.join(
AppLocation.get_section_data_path(self.mediaitem.plugin.name),
'audio', str(self.song.id))
2012-07-07 14:54:14 +00:00
check_directory_exists(save_path)
2011-08-24 11:53:22 +00:00
self.song.media_files = []
files = []
2011-08-24 11:53:22 +00:00
for row in xrange(self.audioListWidget.count()):
item = self.audioListWidget.item(row)
filename = unicode(item.data(QtCore.Qt.UserRole).toString())
if not filename.startswith(save_path):
oldfile, filename = filename, os.path.join(save_path,
os.path.split(filename)[1])
shutil.copyfile(oldfile, filename)
files.append(filename)
media_file = MediaFile()
media_file.file_name = filename
media_file.type = u'audio'
media_file.weight = row
self.song.media_files.append(media_file)
for audio in audio_files:
if audio not in files:
try:
os.remove(audio)
except:
log.exception('Could not remove file: %s', audio)
2011-09-04 12:04:44 +00:00
if not files:
try:
os.rmdir(save_path)
except OSError:
log.exception(u'Could not remove directory: %s', save_path)
2011-03-15 18:24:13 +00:00
clean_song(self.manager, self.song)
2011-01-17 00:52:00 +00:00
self.manager.save_object(self.song)
self.mediaitem.autoSelectId = self.song.id
2011-03-17 12:32:45 +00:00
def _processLyrics(self):
2011-01-17 00:52:00 +00:00
"""
Process the lyric data entered by the user into the OpenLP XML format.
"""
# This method must only be run after the self.song = Song() assignment.
2011-03-17 12:32:45 +00:00
log.debug(u'_processLyrics')
try:
2011-01-09 16:52:31 +00:00
sxml = SongXML()
multiple = []
for i in range(self.verseListWidget.rowCount()):
2011-01-04 21:06:50 +00:00
item = self.verseListWidget.item(i, 0)
2012-06-16 16:12:54 +00:00
verse_id = unicode(item.data(QtCore.Qt.UserRole).toString())
verse_tag = verse_id[0]
verse_num = verse_id[1:]
2011-02-18 07:53:40 +00:00
sxml.add_verse_to_lyrics(verse_tag, verse_num,
2011-02-17 19:46:01 +00:00
unicode(item.text()))
2011-03-15 17:52:42 +00:00
if verse_num > u'1' and verse_tag not in multiple:
multiple.append(verse_tag)
2009-11-04 19:13:49 +00:00
self.song.lyrics = unicode(sxml.extract_xml(), u'utf-8')
for verse in multiple:
2010-06-20 06:48:32 +00:00
self.song.verse_order = re.sub(u'([' + verse.upper() +
2010-06-20 13:03:06 +00:00
verse.lower() + u'])(\W|$)', r'\g<1>1\2',
self.song.verse_order)
except:
2009-11-04 01:16:15 +00:00
log.exception(u'Problem processing song Lyrics \n%s',
sxml.dump_xml())
raise
2011-08-24 11:53:22 +00:00