openlp/openlp.pyw

264 lines
10 KiB
Python
Raw Normal View History

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2010-12-26 11:04:47 +00:00
# Copyright (c) 2008-2011 Raoul Snyman #
# Portions copyright (c) 2008-2011 Tim Bentley, Jonathan Corwin, Michael #
2011-03-24 19:04:02 +00:00
# Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan, Armin Köhler, #
# Andreas Preikschat, Mattias Põldaru, Christian Richter, Philip Ridout, #
2011-05-24 19:29:02 +00:00
# Jeffrey Smith, Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode #
# Woldsund #
# --------------------------------------------------------------------------- #
# 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 os
import sys
2009-09-25 23:06:54 +00:00
import logging
2011-03-24 20:46:34 +00:00
# Import uuid now, to avoid the rare bug described in the support system:
# http://support.openlp.org/issues/102
# If https://bugs.gentoo.org/show_bug.cgi?id=317557 is fixed, the import can be
# removed.
import uuid
2009-09-25 23:06:54 +00:00
from optparse import OptionParser
from traceback import format_exception
2009-09-29 12:51:38 +00:00
from PyQt4 import QtCore, QtGui
2011-01-09 08:17:17 +00:00
from openlp.core.lib import Receiver, check_directory_exists
from openlp.core.lib.ui import UiStrings
from openlp.core.resources import qInitResources
2010-08-01 08:41:09 +00:00
from openlp.core.ui.mainwindow import MainWindow
2011-03-02 17:44:33 +00:00
from openlp.core.ui.firsttimelanguageform import FirstTimeLanguageForm
2011-02-26 11:16:21 +00:00
from openlp.core.ui.firsttimeform import FirstTimeForm
from openlp.core.ui.exceptionform import ExceptionForm
2010-08-01 08:41:09 +00:00
from openlp.core.ui import SplashScreen, ScreenList
2011-03-24 19:12:27 +00:00
from openlp.core.utils import AppLocation, LanguageManager, VersionThread, \
2011-04-30 07:42:20 +00:00
get_application_version, DelayStartThread
2010-07-30 22:48:09 +00:00
log = logging.getLogger()
application_stylesheet = u"""
QMainWindow::separator
{
border: none;
}
QDockWidget::title
{
border: 1px solid palette(dark);
padding-left: 5px;
padding-top: 2px;
margin: 1px 0;
}
QToolBar
{
border: none;
margin: 0;
padding: 0;
}
"""
2009-09-29 12:51:38 +00:00
class OpenLP(QtGui.QApplication):
"""
The core application class. This class inherits from Qt's QApplication
class in order to provide the core of the application.
"""
2011-05-21 22:21:34 +00:00
def exec_(self):
"""
Override exec method to allow the shared memory to be released on exit
"""
QtGui.QApplication.exec_()
self.sharedMemory.detach()
2011-05-21 22:16:28 +00:00
def run(self, args):
"""
Run the OpenLP application.
"""
2011-05-21 22:16:28 +00:00
# On Windows, the args passed into the constructor are
# ignored. Not very handy, so set the ones we want to use.
self.args = args
# provide a listener for widgets to reqest a screen update.
2009-09-29 12:51:38 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2010-04-16 07:31:01 +00:00
QtCore.SIGNAL(u'openlp_process_events'), self.processEvents)
2011-01-01 11:44:16 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
QtCore.SIGNAL(u'cursor_busy'), self.setBusyCursor)
QtCore.QObject.connect(Receiver.get_receiver(),
QtCore.SIGNAL(u'cursor_normal'), self.setNormalCursor)
2011-03-05 09:23:47 +00:00
# Decide how many screens we have and their size
2011-05-09 08:15:59 +00:00
screens = ScreenList(self.desktop())
2011-03-05 09:23:47 +00:00
# First time checks in settings
2011-03-11 15:18:51 +00:00
has_run_wizard = QtCore.QSettings().value(
2011-03-11 10:20:09 +00:00
u'general/has run wizard', QtCore.QVariant(False)).toBool()
2011-03-11 15:18:51 +00:00
if not has_run_wizard:
if FirstTimeForm(screens).exec_() == QtGui.QDialog.Accepted:
QtCore.QSettings().setValue(u'general/has run wizard',
QtCore.QVariant(True))
if os.name == u'nt':
self.setStyleSheet(application_stylesheet)
2010-04-27 16:27:57 +00:00
show_splash = QtCore.QSettings().value(
2010-04-28 14:17:42 +00:00
u'general/show splash', QtCore.QVariant(True)).toBool()
if show_splash:
self.splash = SplashScreen()
self.splash.show()
# make sure Qt really display the splash screen
self.processEvents()
# start the main app window
2011-05-21 22:16:28 +00:00
self.mainWindow = MainWindow(self.clipboard(), self.args)
2009-05-20 20:17:20 +00:00
self.mainWindow.show()
if show_splash:
# now kill the splashscreen
self.splash.finish(self.mainWindow)
2011-04-03 15:42:18 +00:00
log.debug(u'Splashscreen closed')
self.mainWindow.repaint()
self.processEvents()
if not has_run_wizard:
self.mainWindow.firstTime()
update_check = QtCore.QSettings().value(
u'general/update check', QtCore.QVariant(True)).toBool()
if update_check:
2011-03-25 16:29:39 +00:00
VersionThread(self.mainWindow).start()
2011-04-30 07:42:20 +00:00
DelayStartThread(self.mainWindow).start()
return self.exec_()
def isAlreadyRunning(self):
2011-03-26 19:50:15 +00:00
"""
Look to see if OpenLP is already running and ask if a 2nd copy
is to be started.
"""
self.sharedMemory = QtCore.QSharedMemory('OpenLP')
if self.sharedMemory.attach():
2011-03-26 19:34:01 +00:00
status = QtGui.QMessageBox.critical(None,
UiStrings().Error, UiStrings().OpenLPStart,
2011-03-26 19:34:01 +00:00
QtGui.QMessageBox.StandardButtons(
QtGui.QMessageBox.Yes | QtGui.QMessageBox.No))
if status == QtGui.QMessageBox.No:
return True
return False
else:
self.sharedMemory.create(1)
return False
def hookException(self, exctype, value, traceback):
if not hasattr(self, u'mainWindow'):
log.exception(''.join(format_exception(exctype, value, traceback)))
return
if not hasattr(self, u'exceptionForm'):
self.exceptionForm = ExceptionForm(self.mainWindow)
self.exceptionForm.exceptionTextEdit.setPlainText(
''.join(format_exception(exctype, value, traceback)))
2011-01-01 12:49:38 +00:00
self.setNormalCursor()
self.exceptionForm.exec_()
2011-01-01 11:44:16 +00:00
def setBusyCursor(self):
"""
2011-01-01 12:49:38 +00:00
Sets the Busy Cursor for the Application
2011-01-01 11:44:16 +00:00
"""
self.setOverrideCursor(QtCore.Qt.BusyCursor)
2011-02-03 17:23:24 +00:00
self.processEvents()
2011-01-01 11:44:16 +00:00
def setNormalCursor(self):
"""
2011-01-01 17:23:24 +00:00
Sets the Normal Cursor for the Application
2011-01-01 11:44:16 +00:00
"""
self.restoreOverrideCursor()
def main():
"""
The main function which parses command line options and then runs
the PyQt4 Application.
"""
# Set up command line options.
usage = 'Usage: %prog [options] [qt-options]'
parser = OptionParser(usage=usage)
parser.add_option('-e', '--no-error-form', dest='no_error_form',
action='store_true', help='Disable the error notification form.')
parser.add_option('-l', '--log-level', dest='loglevel',
default='warning', metavar='LEVEL', help='Set logging to LEVEL '
'level. Valid values are "debug", "info", "warning".')
parser.add_option('-p', '--portable', dest='portable',
action='store_true', help='Specify if this should be run as a '
'portable app, off a USB flash drive (not implemented).')
parser.add_option('-d', '--dev-version', dest='dev_version',
action='store_true', help='Ignore the version file and pull the '
'version directly from Bazaar')
parser.add_option('-s', '--style', dest='style',
help='Set the Qt4 style (passed directly to Qt4).')
# Set up logging
2010-06-22 12:32:15 +00:00
log_path = AppLocation.get_directory(AppLocation.CacheDir)
2011-01-09 08:17:17 +00:00
check_directory_exists(log_path)
filename = os.path.join(log_path, u'openlp.log')
logfile = logging.FileHandler(filename, u'w')
logfile.setFormatter(logging.Formatter(
u'%(asctime)s %(name)-55s %(levelname)-8s %(message)s'))
log.addHandler(logfile)
2009-11-11 19:10:38 +00:00
logging.addLevelName(15, u'Timer')
# Parse command line options and deal with them.
(options, args) = parser.parse_args()
qt_args = []
if options.loglevel.lower() in ['d', 'debug']:
2009-08-31 08:39:19 +00:00
log.setLevel(logging.DEBUG)
print 'Logging to:', filename
elif options.loglevel.lower() in ['w', 'warning']:
log.setLevel(logging.WARNING)
else:
log.setLevel(logging.INFO)
if options.style:
qt_args.extend(['-style', options.style])
# Throw the rest of the arguments at Qt, just in case.
qt_args.extend(args)
# Initialise the resources
qInitResources()
# Now create and actually run the application.
app = OpenLP(qt_args)
# Instance check
if app.isAlreadyRunning():
sys.exit()
app.setOrganizationName(u'OpenLP')
app.setOrganizationDomain(u'openlp.org')
app.setApplicationName(u'OpenLP')
2011-03-25 16:29:39 +00:00
app.setApplicationVersion(get_application_version()[u'version'])
2011-02-26 20:52:26 +00:00
# First time checks in settings
2011-03-24 17:18:28 +00:00
if not QtCore.QSettings().value(u'general/has run wizard',
QtCore.QVariant(False)).toBool():
2011-03-06 13:44:35 +00:00
if not FirstTimeLanguageForm().exec_():
# if cancel then stop processing
sys.exit()
2011-03-04 19:02:00 +00:00
if sys.platform == u'darwin':
OpenLP.addLibraryPath(QtGui.QApplication.applicationDirPath()
+ "/qt4_plugins")
2011-02-26 09:26:17 +00:00
# i18n Set Language
2010-04-19 19:12:34 +00:00
language = LanguageManager.get_language()
2011-04-23 15:15:01 +00:00
app_translator, default_translator = \
2011-04-23 14:40:37 +00:00
LanguageManager.get_translator(language)
2011-04-23 15:15:01 +00:00
if not app_translator.isEmpty():
app.installTranslator(app_translator)
if not default_translator.isEmpty():
app.installTranslator(default_translator)
else:
log.debug(u'Could not find default_translator.')
if not options.no_error_form:
sys.excepthook = app.hookException
2011-05-21 22:16:28 +00:00
sys.exit(app.run(qt_args))
2009-05-20 20:17:20 +00:00
if __name__ == u'__main__':
"""
Instantiate and run the application.
"""
2011-04-23 14:40:37 +00:00
main()