openlp/openlp/core/ui/formattingtagform.py

322 lines
17 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2012-12-29 15:25:29 +00:00
# vim: autoindent shiftwidth=4 expandtab textwidth=120 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2012-12-29 20:56:56 +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, #
# 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 #
###############################################################################
2011-02-07 15:55:02 +00:00
"""
2013-02-01 20:36:27 +00:00
The :mod:`formattingtagform` provides an Tag Edit facility. The Base set are protected and included each time loaded.
Custom tags can be defined and saved. The Custom Tag arrays are saved in a pickle so QSettings works on them. Base Tags
cannot be changed.
2011-02-07 15:55:02 +00:00
"""
2013-08-25 13:29:00 +00:00
import re
import cgi
from PyQt4 import QtGui, QtCore
2013-02-05 08:05:28 +00:00
from openlp.core.lib import FormattingTags, translate
2011-02-21 22:10:08 +00:00
from openlp.core.lib.ui import critical_error_message_box
from openlp.core.ui.formattingtagdialog import Ui_FormattingTagDialog
class FormattingTagForm(QtGui.QDialog, Ui_FormattingTagDialog):
2011-02-07 15:55:02 +00:00
"""
The :class:`FormattingTagForm` manages the settings tab .
2011-02-07 15:55:02 +00:00
"""
2011-02-20 17:06:58 +00:00
def __init__(self, parent):
2011-02-07 15:55:02 +00:00
"""
2011-02-20 17:06:58 +00:00
Constructor
2011-02-07 15:55:02 +00:00
"""
2011-02-20 17:06:58 +00:00
QtGui.QDialog.__init__(self, parent)
self.setupUi(self)
2013-08-25 13:29:00 +00:00
self.html_tag_regex = re.compile(r'<(?:(?P<close>/(?=[^\s/>]+>))?'
r'(?P<tag>[^\s/!\?>]+)(?:\s+[^\s=]+="[^"]*")*\s*(?P<empty>/)?'
r'|(?P<cdata>!\[CDATA\[(?:(?!\]\]>).)*\]\])'
r'|(?P<procinst>\?(?:(?!\?>).)*\?)'
r'|(?P<comment>!--(?:(?!-->).)*--))>', re.UNICODE)
self.html_regex = re.compile(r'^(?:[^<>]*%s)*[^<>]*$' % self.html_tag_regex.pattern)
#self.tag_table_widget.itemSelectionChanged.connect(self.on_row_selected)
self.new_button.clicked.connect(self.on_new_clicked)
#self.save_push_button.clicked.connect(self.on_saved_clicked)
self.delete_button.clicked.connect(self.on_delete_clicked)
self.tag_table_widget.currentCellChanged.connect(self.on_current_cell_changed)
self.button_box.rejected.connect(self.close)
# Forces reloading of tags from openlp configuration.
FormattingTags.load_tags()
2013-08-25 13:29:00 +00:00
self.pause_validation = False
2011-02-20 17:06:58 +00:00
def exec_(self):
"""
2011-02-20 17:06:58 +00:00
Load Display and set field state.
"""
# Create initial copy from master
self._reloadTable()
2011-02-21 20:06:55 +00:00
return QtGui.QDialog.exec_(self)
2013-03-06 22:32:59 +00:00
def on_row_selected(self):
"""
Table Row selected so display items and set field state.
"""
2013-08-25 13:29:00 +00:00
#self.save_push_button.setEnabled(False)
2013-03-06 22:27:22 +00:00
self.selected = self.tag_table_widget.currentRow()
html = FormattingTags.get_html_tags()[self.selected]
2013-03-06 22:27:22 +00:00
self.description_line_edit.setText(html[u'desc'])
2013-03-06 22:32:59 +00:00
self.tag_line_edit.setText(self._strip(html[u'start tag']))
self.start_tag_line_edit.setText(html[u'start html'])
2013-03-06 22:27:22 +00:00
self.end_tag_line_edit.setText(html[u'end html'])
if html[u'protected']:
2013-03-06 22:27:22 +00:00
self.description_line_edit.setEnabled(False)
2013-03-06 22:32:59 +00:00
self.tag_line_edit.setEnabled(False)
self.start_tag_line_edit.setEnabled(False)
2013-03-06 22:27:22 +00:00
self.end_tag_line_edit.setEnabled(False)
self.delete_push_button.setEnabled(False)
else:
2013-03-06 22:27:22 +00:00
self.description_line_edit.setEnabled(True)
2013-03-06 22:32:59 +00:00
self.tag_line_edit.setEnabled(True)
self.start_tag_line_edit.setEnabled(True)
2013-03-06 22:27:22 +00:00
self.end_tag_line_edit.setEnabled(True)
self.delete_push_button.setEnabled(True)
2013-03-06 22:32:59 +00:00
def on_text_edited(self, text):
"""
2013-03-06 22:27:22 +00:00
Enable the ``save_push_button`` when any of the selected tag's properties
has been changed.
"""
2013-03-06 22:27:22 +00:00
self.save_push_button.setEnabled(True)
2013-03-06 22:32:59 +00:00
def on_new_clicked(self):
"""
Add a new tag to list only if it is not a duplicate.
"""
2013-08-25 13:29:00 +00:00
last_row = self.tag_table_widget.rowCount() - 1
self.tag_table_widget.selectRow(last_row)
self.tag_table_widget.setCurrentCell(last_row, 0)
for html in FormattingTags.get_html_tags():
if self._strip(html[u'start tag']) == u'n':
critical_error_message_box(
2011-07-30 07:43:19 +00:00
translate('OpenLP.FormattingTagForm', 'Update Error'),
2012-12-29 15:25:29 +00:00
translate('OpenLP.FormattingTagForm', 'Tag "n" already defined.'))
return
# Add new tag to list
tag = {
2011-07-30 07:43:19 +00:00
u'desc': translate('OpenLP.FormattingTagForm', 'New Tag'),
2013-08-25 13:29:00 +00:00
u'start tag': u'{n%s}' % unicode(last_row + 1),
2011-07-30 07:43:19 +00:00
u'start html': translate('OpenLP.FormattingTagForm', '<HTML here>'),
2013-08-25 13:29:00 +00:00
u'end tag': u'{/n%s}' % unicode(last_row + 1),
2011-07-30 07:43:19 +00:00
u'end html': translate('OpenLP.FormattingTagForm', '</and here>'),
u'protected': False,
u'temporary': False
}
2013-08-25 13:29:00 +00:00
#FormattingTags.add_html_tags([tag])
#FormattingTags.save_html_tags()
#self._reloadTable()
2011-01-08 09:44:44 +00:00
# Highlight new row
2013-08-25 13:29:00 +00:00
#self.tag_table_widget.selectRow(self.tag_table_widget.rowCount() - 1)
#self.on_row_selected()
2013-03-06 22:27:22 +00:00
self.tag_table_widget.scrollToBottom()
2013-03-06 22:32:59 +00:00
def on_delete_clicked(self):
"""
Delete selected custom tag.
"""
if self.selected != -1:
FormattingTags.remove_html_tag(self.selected)
# As the first items are protected we should not have to take care
# of negative indexes causing tracebacks.
2013-03-06 22:27:22 +00:00
self.tag_table_widget.selectRow(self.selected - 1)
self.selected = -1
FormattingTags.save_html_tags()
self._reloadTable()
2013-03-06 22:32:59 +00:00
def on_saved_clicked(self):
"""
2011-05-08 05:42:48 +00:00
Update Custom Tag details if not duplicate and save the data.
"""
html_expands = FormattingTags.get_html_tags()
if self.selected != -1:
html = html_expands[self.selected]
2013-03-06 22:32:59 +00:00
tag = self.tag_line_edit.text()
for linenumber, html1 in enumerate(html_expands):
2012-12-29 15:25:29 +00:00
if self._strip(html1[u'start tag']) == tag and linenumber != self.selected:
critical_error_message_box(
2011-07-30 07:43:19 +00:00
translate('OpenLP.FormattingTagForm', 'Update Error'),
2012-12-29 15:25:29 +00:00
translate('OpenLP.FormattingTagForm', 'Tag %s already defined.') % tag)
return
2013-03-06 22:27:22 +00:00
html[u'desc'] = self.description_line_edit.text()
2013-03-06 22:32:59 +00:00
html[u'start html'] = self.start_tag_line_edit.text()
2013-03-06 22:27:22 +00:00
html[u'end html'] = self.end_tag_line_edit.text()
html[u'start tag'] = u'{%s}' % tag
html[u'end tag'] = u'{/%s}' % tag
# Keep temporary tags when the user changes one.
html[u'temporary'] = False
self.selected = -1
2011-08-27 14:00:24 +00:00
FormattingTags.save_html_tags()
self._reloadTable()
def _reloadTable(self):
"""
Reset List for loading.
"""
2013-08-25 13:29:00 +00:00
self.tag_table_widget_read.clearContents()
self.tag_table_widget_read.setRowCount(0)
2013-03-06 22:27:22 +00:00
self.tag_table_widget.clearContents()
self.tag_table_widget.setRowCount(0)
2013-08-25 13:29:00 +00:00
self.new_button.setEnabled(True)
self.delete_button.setEnabled(False)
2012-05-02 18:25:37 +00:00
for linenumber, html in enumerate(FormattingTags.get_html_tags()):
2013-08-25 13:29:00 +00:00
if html[u'protected']:
line = self.tag_table_widget_read.rowCount()
self.tag_table_widget_read.setRowCount(line + 1)
print linenumber, self.tag_table_widget_read.rowCount()
self.tag_table_widget_read.setItem(line, 0, QtGui.QTableWidgetItem(html[u'desc']))
self.tag_table_widget_read.setItem(line, 1, QtGui.QTableWidgetItem(self._strip(html[u'start tag'])))
self.tag_table_widget_read.setItem(line, 2, QtGui.QTableWidgetItem(html[u'start html']))
self.tag_table_widget_read.setItem(line, 3, QtGui.QTableWidgetItem(html[u'end html']))
self.tag_table_widget_read.resizeRowsToContents()
else:
print self.tag_table_widget.rowCount(), html
line = self.tag_table_widget.rowCount()
self.tag_table_widget.setRowCount(line + 1)
self.tag_table_widget.setRowCount(self.tag_table_widget.rowCount() + 1)
self.tag_table_widget.setItem(line, 0, QtGui.QTableWidgetItem(html[u'desc']))
self.tag_table_widget.setItem(line, 1, QtGui.QTableWidgetItem(self._strip(html[u'start tag'])))
self.tag_table_widget.setItem(line, 2, QtGui.QTableWidgetItem(html[u'start html']))
self.tag_table_widget.setItem(line, 3, QtGui.QTableWidgetItem(html[u'end html']))
self.tag_table_widget.resizeRowsToContents()
# Permanent (persistent) tags do not have this key
html[u'temporary'] = False
2013-08-25 13:29:00 +00:00
def on_current_cell_changed(self, cur_row, cur_col, pre_row, pre_col):
"""
This function processes all user edits in the table. It is called on each cell change.
"""
print cur_row, cur_col, pre_col, pre_col
# only process for editable rows
pre_row_item = self.tag_table_widget.item(pre_row, 0)
edit_item = None
if pre_row_item and (pre_row_item.flags() & QtCore.Qt.ItemIsEditable) and not self.pause_validation:
data = self.item_to_data_dict(pre_row_item)
item = self.tag_table_widget.item(pre_row, pre_col)
text = unicode(item.text())
if pre_col is 0:
# Tag name edited
if text:
for row in range(self.tag_table_widget.rowCount()):
counting_item = self.tag_table_widget.item(row, 0)
if row != pre_row and counting_item and counting_item.text() == text:
answer = QtGui.QMessageBox.warning(self,
translate('OpenLP.FormattingTagForm', 'Validation Error'),
translate('OpenLP.FormattingTagForm',
'Tag %s is already defined. Please pick a different one.' % text),
QtGui.QMessageBox.Discard|QtGui.QMessageBox.Ok)
if answer == QtGui.QMessageBox.Discard:
item.setText(data.get(u'tag'))
else:
edit_item = item
break
if not edit_item:
data[u'tag'] = text
data.setdefault(u'description', u'')
data.setdefault(u'html', u'')
pre_row_item.setData(QtCore.Qt.UserRole, data)
flags = self.tag_table_widget.item(pre_row, 1).flags()
if not (flags & QtCore.Qt.ItemIsEditable):
# if description cell is read only, the user is adding a new tag.
# So we add another empty row and enable editing for description and html.
new_row = self.tag_table_widget.rowCount()
self.tag_table_widget.insertRow(new_row)
for column in range(4):
new_item = QtGui.QTableWidgetItem(u'')
if column != 0:
new_item.setFlags(flags)
self.tag_table_widget.setItem(new_row, column, new_item)
for column in [1, 2]:
self.tag_table_widget.item(pre_row, column).setFlags(item.flags())
# trigger edit as editing might have been enabled after selecting
if cur_row == pre_row and cur_col in [1, 2]:
cur_item = self.tag_table_widget.item(cur_row, cur_col)
self.tag_table_widget.editItem(cur_item)
self.tag_table_widget.resizeRowsToContents()
else:
answer = None
if self.tag_table_widget.item(pre_row, 1).text() or self.tag_table_widget.item(pre_row, 2).text():
answer = QtGui.QMessageBox.warning(self,
translate('OpenLP.FormattingTagForm', 'Validation Error'),
translate('OpenLP.FormattingTagForm',
'No tag name defined. Do you want to delete the whole tag?'),
QtGui.QMessageBox.Yes|QtGui.QMessageBox.Discard|QtGui.QMessageBox.Cancel)
if answer == QtGui.QMessageBox.Discard:
item.setText(data.get(u'tag'))
if answer == QtGui.QMessageBox.Cancel:
edit_item = item
elif pre_row < self.tag_table_widget.rowCount() - 1:
self.tag_table_widget.removeRow(pre_row)
elif pre_col is 1:
# Description edited
data[u'description'] = text
pre_row_item.setData(QtCore.Qt.UserRole, data)
elif pre_col is 2:
# HTML edited
end_html = self.start_html_to_end_html(text)
if end_html is not None:
item.setToolTip(cgi.escape(text))
if self.tag_table_widget.item(pre_row, 3) is None:
self.tag_table_widget.setItem(pre_row, 3, QtGui.QTableWidgetItem(end_html))
else:
self.tag_table_widget.item(pre_row, 3).setText(end_html)
self.tag_table_widget.item(pre_row, 3).setToolTip(cgi.escape(end_html))
data[u'html'] = text
pre_row_item.setData(QtCore.Qt.UserRole, data)
self.tag_table_widget.resizeRowsToContents()
elif QtGui.QMessageBox.question(self,
translate('OpenLP.FormattingTagForm', 'Validation Error'),
translate('OpenLP.FormattingTagForm', 'The entered HTML is not valid. Please enter valid HTML.'),
QtGui.QMessageBox.Ok | QtGui.QMessageBox.Cancel) == QtGui.QMessageBox.Cancel:
item.setText(data.get(u'html'))
else:
edit_item = item
if not edit_item:
# select the tag cell in a empty row
cur_row_item = self.tag_table_widget.item(cur_row, 0)
if cur_row_item and (cur_row_item.flags() & QtCore.Qt.ItemIsEditable) and cur_row_item.text().isEmpty():
edit_item = cur_row_item
if edit_item:
self.tag_table_widget.setCurrentItem(edit_item)
# enable delete_button for editable rows
cur_row = self.tag_table_widget.currentRow()
cur_row_item = self.tag_table_widget.item(cur_row, 0)
delete_enabled = bool(cur_row_item) and bool(cur_row_item.flags() & QtCore.Qt.ItemIsEditable)
delete_enabled &= cur_row < self.tag_table_widget.rowCount() - 1
self.delete_button.setEnabled(delete_enabled)
def _strip(self, tag):
"""
Remove tag wrappers for editing.
"""
tag = tag.replace(u'{', u'')
tag = tag.replace(u'}', u'')
2010-12-31 08:34:53 +00:00
return tag