2008-11-27 20:09:11 +00:00
|
|
|
#!/usr/bin/env python
|
2008-11-22 09:28:03 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
2008-11-29 05:36:16 +00:00
|
|
|
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
|
2009-09-02 20:42:57 +00:00
|
|
|
|
2009-09-08 19:58:05 +00:00
|
|
|
###############################################################################
|
|
|
|
# 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-07-24 22:10:47 +00:00
|
|
|
# Gorven, Scott Guerrieri, Meinert Jordan, Andreas Preikschat, Christian #
|
|
|
|
# Richter, Philip Ridout, Maikel Stuivenberg, Martin Thompson, Jon Tibble, #
|
|
|
|
# Carsten Tinggaard, Frode Woldsund #
|
2009-09-08 19:58:05 +00:00
|
|
|
# --------------------------------------------------------------------------- #
|
|
|
|
# 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 #
|
|
|
|
###############################################################################
|
2008-03-04 01:08:01 +00:00
|
|
|
|
2009-10-17 22:13:32 +00:00
|
|
|
import os
|
2008-10-29 18:51:43 +00:00
|
|
|
import sys
|
2009-09-25 23:06:54 +00:00
|
|
|
import logging
|
|
|
|
from optparse import OptionParser
|
2010-09-07 20:42:33 +00:00
|
|
|
from traceback import format_exception
|
2010-10-10 19:44:41 +00:00
|
|
|
from subprocess import Popen, PIPE
|
2010-07-14 15:58:28 +00:00
|
|
|
|
2009-09-29 12:51:38 +00:00
|
|
|
from PyQt4 import QtCore, QtGui
|
2009-07-08 06:55:08 +00:00
|
|
|
|
2010-04-27 16:27:57 +00:00
|
|
|
from openlp.core.lib import Receiver
|
2009-09-20 15:45:33 +00:00
|
|
|
from openlp.core.resources import qInitResources
|
2010-08-01 08:41:09 +00:00
|
|
|
from openlp.core.ui.mainwindow import MainWindow
|
2010-09-07 20:42:33 +00:00
|
|
|
from openlp.core.ui.exceptionform import ExceptionForm
|
2010-08-01 08:41:09 +00:00
|
|
|
from openlp.core.ui import SplashScreen, ScreenList
|
2010-07-30 22:48:09 +00:00
|
|
|
from openlp.core.utils import AppLocation, LanguageManager, VersionThread
|
|
|
|
|
|
|
|
log = logging.getLogger()
|
2008-10-29 18:51:43 +00:00
|
|
|
|
2009-10-17 22:06:38 +00:00
|
|
|
application_stylesheet = u"""
|
|
|
|
QMainWindow::separator
|
|
|
|
{
|
|
|
|
border: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDockWidget::title
|
|
|
|
{
|
2010-02-21 10:28:35 +00:00
|
|
|
border: 1px solid palette(dark);
|
2009-10-18 15:56:59 +00:00
|
|
|
padding-left: 5px;
|
2010-02-21 10:28:35 +00:00
|
|
|
padding-top: 2px;
|
|
|
|
margin: 1px 0;
|
2009-10-17 22:06:38 +00:00
|
|
|
}
|
2009-10-18 17:06:34 +00:00
|
|
|
|
|
|
|
QToolBar
|
|
|
|
{
|
|
|
|
border: none;
|
2009-10-18 18:15:20 +00:00
|
|
|
margin: 0;
|
2009-10-18 17:06:34 +00:00
|
|
|
padding: 0;
|
|
|
|
}
|
2009-10-17 22:06:38 +00:00
|
|
|
"""
|
|
|
|
|
2009-09-29 12:51:38 +00:00
|
|
|
class OpenLP(QtGui.QApplication):
|
2009-07-08 06:55:08 +00:00
|
|
|
"""
|
|
|
|
The core application class. This class inherits from Qt's QApplication
|
|
|
|
class in order to provide the core of the application.
|
|
|
|
"""
|
2009-08-31 08:39:19 +00:00
|
|
|
log.info(u'OpenLP Application Loaded')
|
2008-03-04 01:08:01 +00:00
|
|
|
|
2010-10-10 19:44:41 +00:00
|
|
|
def _get_version(self):
|
|
|
|
"""
|
|
|
|
Load and store current Application Version
|
|
|
|
"""
|
|
|
|
if u'--dev-version' in sys.argv:
|
|
|
|
# If we're running the dev version, let's use bzr to get the version
|
|
|
|
try:
|
|
|
|
# If bzrlib is availble, use it
|
|
|
|
from bzrlib.branch import Branch
|
|
|
|
b = Branch.open_containing('.')[0]
|
|
|
|
b.lock_read()
|
|
|
|
try:
|
|
|
|
# Get the branch's latest revision number.
|
|
|
|
revno = b.revno()
|
|
|
|
# Convert said revision number into a bzr revision id.
|
|
|
|
revision_id = b.dotted_revno_to_revision_id((revno,))
|
|
|
|
# Get a dict of tags, with the revision id as the key.
|
|
|
|
tags = b.tags.get_reverse_tag_dict()
|
|
|
|
# Check if the latest
|
|
|
|
if revision_id in tags:
|
|
|
|
full_version = u'%s' % tags[revision_id][0]
|
|
|
|
else:
|
|
|
|
full_version = '%s-bzr%s' % \
|
|
|
|
(sorted(b.tags.get_tag_dict().keys())[-1], revno)
|
|
|
|
finally:
|
|
|
|
b.unlock()
|
|
|
|
except:
|
|
|
|
# Otherwise run the command line bzr client
|
|
|
|
bzr = Popen((u'bzr', u'tags', u'--sort', u'time'), stdout=PIPE)
|
|
|
|
output, error = bzr.communicate()
|
|
|
|
code = bzr.wait()
|
|
|
|
if code != 0:
|
|
|
|
raise Exception(u'Error running bzr tags')
|
|
|
|
lines = output.splitlines()
|
|
|
|
if len(lines) == 0:
|
|
|
|
tag = u'0.0.0'
|
|
|
|
revision = u'0'
|
|
|
|
else:
|
|
|
|
tag, revision = lines[-1].split()
|
|
|
|
bzr = Popen((u'bzr', u'log', u'--line', u'-r', u'-1'),
|
|
|
|
stdout=PIPE)
|
|
|
|
output, error = bzr.communicate()
|
|
|
|
code = bzr.wait()
|
|
|
|
if code != 0:
|
|
|
|
raise Exception(u'Error running bzr log')
|
|
|
|
latest = output.split(u':')[0]
|
|
|
|
full_version = latest == revision and tag or \
|
|
|
|
u'%s-bzr%s' % (tag, latest)
|
|
|
|
else:
|
|
|
|
# We're not running the development version, let's use the file
|
|
|
|
filepath = AppLocation.get_directory(AppLocation.VersionDir)
|
|
|
|
filepath = os.path.join(filepath, u'.version')
|
|
|
|
fversion = None
|
|
|
|
try:
|
|
|
|
fversion = open(filepath, u'r')
|
|
|
|
full_version = unicode(fversion.read()).rstrip()
|
|
|
|
except IOError:
|
|
|
|
log.exception('Error in version file.')
|
|
|
|
full_version = u'0.0.0-bzr000'
|
|
|
|
finally:
|
|
|
|
if fversion:
|
|
|
|
fversion.close()
|
|
|
|
bits = full_version.split(u'-')
|
|
|
|
app_version = {
|
|
|
|
u'full': full_version,
|
|
|
|
u'version': bits[0],
|
|
|
|
u'build': bits[1] if len(bits) > 1 else None
|
|
|
|
}
|
|
|
|
if app_version[u'build']:
|
|
|
|
log.info(
|
|
|
|
u'Openlp version %s build %s',
|
|
|
|
app_version[u'version'],
|
|
|
|
app_version[u'build']
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
log.info(u'Openlp version %s' % app_version[u'version'])
|
|
|
|
return app_version
|
|
|
|
|
2010-01-28 17:36:13 +00:00
|
|
|
def notify(self, obj, evt):
|
|
|
|
#TODO needed for presentation exceptions
|
|
|
|
return QtGui.QApplication.notify(self, obj, evt)
|
|
|
|
|
2008-10-29 18:51:43 +00:00
|
|
|
def run(self):
|
2009-07-08 06:55:08 +00:00
|
|
|
"""
|
|
|
|
Run the OpenLP application.
|
|
|
|
"""
|
2010-10-10 19:44:41 +00:00
|
|
|
app_version = self._get_version()
|
2009-02-11 20:52:06 +00:00
|
|
|
#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)
|
2010-04-23 18:30:53 +00:00
|
|
|
self.setOrganizationName(u'OpenLP')
|
|
|
|
self.setOrganizationDomain(u'openlp.org')
|
2009-09-12 21:31:35 +00:00
|
|
|
self.setApplicationName(u'OpenLP')
|
2009-11-29 14:07:25 +00:00
|
|
|
self.setApplicationVersion(app_version[u'version'])
|
2009-10-17 22:13:32 +00:00
|
|
|
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()
|
2009-09-12 21:31:35 +00:00
|
|
|
if show_splash:
|
2010-07-30 16:13:17 +00:00
|
|
|
self.splash = SplashScreen()
|
2009-09-12 21:31:35 +00:00
|
|
|
self.splash.show()
|
2008-11-03 19:32:54 +00:00
|
|
|
# make sure Qt really display the splash screen
|
|
|
|
self.processEvents()
|
2010-01-22 18:59:36 +00:00
|
|
|
screens = ScreenList()
|
2009-04-09 18:50:20 +00:00
|
|
|
# Decide how many screens we have and their size
|
2009-05-01 11:50:09 +00:00
|
|
|
for screen in xrange(0, self.desktop().numScreens()):
|
2010-09-04 20:08:36 +00:00
|
|
|
size = self.desktop().screenGeometry(screen);
|
2010-01-16 07:22:50 +00:00
|
|
|
screens.add_screen({u'number': screen,
|
2010-09-04 17:55:10 +00:00
|
|
|
u'size': size,
|
2010-04-28 14:17:42 +00:00
|
|
|
u'primary': (self.desktop().primaryScreen() == screen)})
|
2010-09-04 17:55:10 +00:00
|
|
|
log.info(u'Screen %d found with resolution %s', screen, size)
|
2009-04-09 18:50:20 +00:00
|
|
|
# start the main app window
|
2009-11-29 14:07:25 +00:00
|
|
|
self.mainWindow = MainWindow(screens, app_version)
|
2009-05-20 20:17:20 +00:00
|
|
|
self.mainWindow.show()
|
2009-09-12 21:31:35 +00:00
|
|
|
if show_splash:
|
|
|
|
# now kill the splashscreen
|
|
|
|
self.splash.finish(self.mainWindow)
|
2010-01-28 11:46:25 +00:00
|
|
|
self.mainWindow.repaint()
|
2010-07-30 22:48:09 +00:00
|
|
|
VersionThread(self.mainWindow, app_version).start()
|
2009-09-12 21:31:35 +00:00
|
|
|
return self.exec_()
|
2009-09-02 20:42:57 +00:00
|
|
|
|
2010-09-07 20:42:33 +00:00
|
|
|
def hookException(self, exctype, value, traceback):
|
2010-09-15 12:23:42 +00:00
|
|
|
if not hasattr(self, u'mainWindow'):
|
2010-09-14 20:06:01 +00:00
|
|
|
log.exception(''.join(format_exception(exctype, value, traceback)))
|
|
|
|
return
|
2010-09-07 20:42:33 +00:00
|
|
|
if not hasattr(self, u'exceptionForm'):
|
|
|
|
self.exceptionForm = ExceptionForm(self.mainWindow)
|
|
|
|
self.exceptionForm.exceptionTextEdit.setPlainText(
|
|
|
|
''.join(format_exception(exctype, value, traceback)))
|
|
|
|
self.exceptionForm.exec_()
|
|
|
|
|
2009-08-30 21:14:15 +00:00
|
|
|
def main():
|
2009-09-02 20:42:57 +00:00
|
|
|
"""
|
|
|
|
The main function which parses command line options and then runs
|
|
|
|
the PyQt4 Application.
|
|
|
|
"""
|
|
|
|
# Set up command line options.
|
2010-09-16 21:01:44 +00:00
|
|
|
usage = 'Usage: %prog [options] [qt-options]'
|
2009-08-30 21:14:15 +00:00
|
|
|
parser = OptionParser(usage=usage)
|
2010-09-16 21:01:44 +00:00
|
|
|
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).')
|
2010-10-10 19:44:41 +00:00
|
|
|
parser.add_option('-d', '--dev-version', dest='dev_version',
|
|
|
|
action='store_true', help='Ignore the version file and pull the '
|
|
|
|
'version directly from Bazaar')
|
2010-09-16 21:01:44 +00:00
|
|
|
parser.add_option('-s', '--style', dest='style',
|
|
|
|
help='Set the Qt4 style (passed directly to Qt4).')
|
2009-09-02 20:42:57 +00:00
|
|
|
# Set up logging
|
2010-06-22 12:32:15 +00:00
|
|
|
log_path = AppLocation.get_directory(AppLocation.CacheDir)
|
2010-03-02 20:40:01 +00:00
|
|
|
if not os.path.exists(log_path):
|
|
|
|
os.makedirs(log_path)
|
|
|
|
filename = os.path.join(log_path, u'openlp.log')
|
2010-07-14 15:58:28 +00:00
|
|
|
logfile = logging.FileHandler(filename, u'w')
|
2009-09-29 02:54:32 +00:00
|
|
|
logfile.setFormatter(logging.Formatter(
|
2010-05-07 20:29:21 +00:00
|
|
|
u'%(asctime)s %(name)-55s %(levelname)-8s %(message)s'))
|
2009-09-02 20:42:57 +00:00
|
|
|
log.addHandler(logfile)
|
2009-11-11 19:10:38 +00:00
|
|
|
logging.addLevelName(15, u'Timer')
|
2009-09-02 20:42:57 +00:00
|
|
|
# Parse command line options and deal with them.
|
2009-08-30 21:14:15 +00:00
|
|
|
(options, args) = parser.parse_args()
|
2009-11-09 19:10:28 +00:00
|
|
|
qt_args = []
|
|
|
|
if options.loglevel.lower() in ['d', 'debug']:
|
2009-08-31 08:39:19 +00:00
|
|
|
log.setLevel(logging.DEBUG)
|
2010-02-21 10:28:35 +00:00
|
|
|
print 'Logging to:', filename
|
2009-11-09 19:10:28 +00:00
|
|
|
elif options.loglevel.lower() in ['w', 'warning']:
|
|
|
|
log.setLevel(logging.WARNING)
|
2009-09-02 20:42:57 +00:00
|
|
|
else:
|
|
|
|
log.setLevel(logging.INFO)
|
2009-11-09 19:10:28 +00:00
|
|
|
if options.style:
|
|
|
|
qt_args.extend(['-style', options.style])
|
|
|
|
# Throw the rest of the arguments at Qt, just in case.
|
|
|
|
qt_args.extend(args)
|
2009-09-20 15:45:33 +00:00
|
|
|
# Initialise the resources
|
|
|
|
qInitResources()
|
2009-09-02 20:42:57 +00:00
|
|
|
# Now create and actually run the application.
|
2009-11-09 19:10:28 +00:00
|
|
|
app = OpenLP(qt_args)
|
2010-04-16 22:06:28 +00:00
|
|
|
#i18n Set Language
|
2010-04-19 19:12:34 +00:00
|
|
|
language = LanguageManager.get_language()
|
|
|
|
appTranslator = LanguageManager.get_translator(language)
|
2010-04-16 22:06:28 +00:00
|
|
|
app.installTranslator(appTranslator)
|
2010-09-14 20:06:01 +00:00
|
|
|
if not options.no_error_form:
|
|
|
|
sys.excepthook = app.hookException
|
2009-09-12 21:31:35 +00:00
|
|
|
sys.exit(app.run())
|
2009-09-02 20:42:57 +00:00
|
|
|
|
2009-05-20 20:17:20 +00:00
|
|
|
if __name__ == u'__main__':
|
2009-07-08 06:55:08 +00:00
|
|
|
"""
|
|
|
|
Instantiate and run the application.
|
|
|
|
"""
|
2010-07-27 09:32:52 +00:00
|
|
|
main()
|