openlp/openlp/core/__init__.py

330 lines
14 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
2012-12-05 05:53:14 +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 #
2010-03-21 23:58:01 +00:00
###############################################################################
"""
The :mod:`core` module provides all core application functions
All the core functions of the OpenLP application including the GUI, settings,
logging and a plugin framework are contained within the openlp.core module.
"""
import os
import sys
import platform
import logging
from optparse import OptionParser
from traceback import format_exception
from PyQt4 import QtCore, QtGui
2013-12-13 17:44:05 +00:00
from openlp.core.common import Registry, AppLocation, Settings, UiStrings, check_directory_exists
from openlp.core.lib import ScreenList
from openlp.core.resources import qInitResources
from openlp.core.ui.mainwindow import MainWindow
from openlp.core.ui.firsttimelanguageform import FirstTimeLanguageForm
from openlp.core.ui.firsttimeform import FirstTimeForm
from openlp.core.ui.exceptionform import ExceptionForm
from openlp.core.ui import SplashScreen
2013-10-13 13:51:13 +00:00
from openlp.core.utils import LanguageManager, VersionThread, get_application_version
2013-08-31 18:17:38 +00:00
__all__ = ['OpenLP', 'main']
2010-06-10 01:57:59 +00:00
log = logging.getLogger()
2013-08-31 18:17:38 +00:00
NT_REPAIR_STYLESHEET = """
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;
}
"""
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.
"""
args = []
def exec_(self):
"""
Override exec method to allow the shared memory to be released on exit
"""
self.is_event_loop_active = True
result = QtGui.QApplication.exec_()
self.shared_memory.detach()
return result
2013-02-03 15:06:17 +00:00
def run(self, args):
"""
Run the OpenLP application.
"""
self.is_event_loop_active = False
2013-03-07 12:59:35 +00:00
# On Windows, the args passed into the constructor are ignored. Not very handy, so set the ones we want to use.
# On Linux and FreeBSD, in order to set the WM_CLASS property for X11, we pass "OpenLP" in as a command line
# argument. This interferes with files being passed in as command line arguments, so we remove it from the list.
if 'OpenLP' in args:
args.remove('OpenLP')
self.args.extend(args)
# Decide how many screens we have and their size
screens = ScreenList.create(self.desktop())
# First time checks in settings
2013-08-31 18:17:38 +00:00
has_run_wizard = Settings().value('core/has run wizard')
if not has_run_wizard:
if FirstTimeForm(screens).exec_() == QtGui.QDialog.Accepted:
2013-08-31 18:17:38 +00:00
Settings().setValue('core/has run wizard', True)
# Correct stylesheet bugs
2013-08-31 18:17:38 +00:00
application_stylesheet = ''
if not Settings().value('advanced/alternate rows'):
2012-12-11 20:02:41 +00:00
base_color = self.palette().color(QtGui.QPalette.Active, QtGui.QPalette.Base)
2012-12-19 22:22:11 +00:00
alternate_rows_repair_stylesheet = \
2013-08-31 18:17:38 +00:00
'QTableWidget, QListWidget, QTreeWidget {alternate-background-color: ' + base_color.name() + ';}\n'
2012-12-19 22:22:11 +00:00
application_stylesheet += alternate_rows_repair_stylesheet
2013-08-31 18:17:38 +00:00
if os.name == 'nt':
2013-03-07 16:35:08 +00:00
application_stylesheet += NT_REPAIR_STYLESHEET
2012-12-10 23:35:53 +00:00
if application_stylesheet:
self.setStyleSheet(application_stylesheet)
2013-08-31 18:17:38 +00:00
show_splash = Settings().value('core/show splash')
if show_splash:
self.splash = SplashScreen()
self.splash.show()
# make sure Qt really display the splash screen
self.processEvents()
# start the main app window
2013-02-03 19:23:12 +00:00
self.main_window = MainWindow()
2013-08-31 18:17:38 +00:00
Registry().execute('bootstrap_initialise')
Registry().execute('bootstrap_post_set_up')
self.main_window.show()
if show_splash:
# now kill the splashscreen
self.splash.finish(self.main_window)
2013-08-31 18:17:38 +00:00
log.debug('Splashscreen closed')
# make sure Qt really display the splash screen
self.processEvents()
self.main_window.repaint()
self.processEvents()
if not has_run_wizard:
self.main_window.first_time()
2013-08-31 18:17:38 +00:00
update_check = Settings().value('core/update check')
if update_check:
VersionThread(self.main_window).start()
2013-12-19 20:17:06 +00:00
#self.main_window.is_display_blank()
self.main_window.app_startup()
2013-02-03 15:06:17 +00:00
return self.exec_()
def is_already_running(self):
"""
2013-03-23 07:07:06 +00:00
Look to see if OpenLP is already running and ask if a 2nd instance is to be started.
"""
self.shared_memory = QtCore.QSharedMemory('OpenLP')
if self.shared_memory.attach():
2012-12-05 05:53:14 +00:00
status = QtGui.QMessageBox.critical(None, UiStrings().Error, UiStrings().OpenLPStart,
2013-12-15 16:50:09 +00:00
QtGui.QMessageBox.StandardButtons(QtGui.QMessageBox.Yes |
QtGui.QMessageBox.No))
if status == QtGui.QMessageBox.No:
return True
return False
else:
self.shared_memory.create(1)
return False
def hook_exception(self, exctype, value, traceback):
2013-02-01 19:58:18 +00:00
"""
Add an exception hook so that any uncaught exceptions are displayed in this window rather than somewhere where
users cannot see it and cannot report when we encounter these problems.
``exctype``
The class of exception.
``value``
The actual exception object.
``traceback``
A traceback object with the details of where the exception occurred.
"""
# We can't log.exception here because the last exception no longer exists, we're actually busy handling it.
log.critical(''.join(format_exception(exctype, value, traceback)))
2013-08-31 18:17:38 +00:00
if not hasattr(self, 'exception_form'):
2013-07-19 20:56:31 +00:00
self.exception_form = ExceptionForm()
2013-03-05 14:02:56 +00:00
self.exception_form.exception_text_edit.setPlainText(''.join(format_exception(exctype, value, traceback)))
2013-02-03 09:07:31 +00:00
self.set_normal_cursor()
self.exception_form.exec_()
2013-02-03 09:07:31 +00:00
def process_events(self):
"""
Wrapper to make ProcessEvents visible and named correctly
"""
2013-08-31 18:17:38 +00:00
log.debug('processing event flush')
2013-02-03 09:07:31 +00:00
self.processEvents()
def set_busy_cursor(self):
"""
Sets the Busy Cursor for the Application
"""
self.setOverrideCursor(QtCore.Qt.BusyCursor)
self.processEvents()
2013-02-03 09:07:31 +00:00
def set_normal_cursor(self):
"""
Sets the Normal Cursor for the Application
"""
self.restoreOverrideCursor()
2013-02-03 09:07:31 +00:00
self.processEvents()
def event(self, event):
"""
Enables direct file opening on OS X
"""
if event.type() == QtCore.QEvent.FileOpen:
file_name = event.file()
2013-08-31 18:17:38 +00:00
log.debug('Got open file event for %s!', file_name)
2012-05-17 18:57:01 +00:00
self.args.insert(0, file_name)
return True
else:
return QtGui.QApplication.event(self, event)
2012-07-08 19:32:56 +00:00
def set_up_logging(log_path):
"""
Setup our logging using log_path
"""
check_directory_exists(log_path, True)
2013-08-31 18:17:38 +00:00
filename = os.path.join(log_path, 'openlp.log')
logfile = logging.FileHandler(filename, 'w')
logfile.setFormatter(logging.Formatter('%(asctime)s %(name)-55s %(levelname)-8s %(message)s'))
log.addHandler(logfile)
if log.isEnabledFor(logging.DEBUG):
2013-02-04 22:05:19 +00:00
print('Logging to: %s' % filename)
def main(args=None):
"""
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)
2012-12-05 05:53:14 +00:00
parser.add_option('-e', '--no-error-form', dest='no_error_form', action='store_true',
2013-12-15 16:50:09 +00:00
help='Disable the error notification form.')
2012-12-05 05:53:14 +00:00
parser.add_option('-l', '--log-level', dest='loglevel', default='warning', metavar='LEVEL',
2013-12-15 16:50:09 +00:00
help='Set logging to LEVEL level. Valid values are "debug", "info", "warning".')
2012-12-05 05:53:14 +00:00
parser.add_option('-p', '--portable', dest='portable', action='store_true',
2013-12-15 16:50:09 +00:00
help='Specify if this should be run as a portable app, off a USB flash drive (not implemented).')
2012-12-05 05:53:14 +00:00
parser.add_option('-d', '--dev-version', dest='dev_version', action='store_true',
2013-12-15 16:50:09 +00:00
help='Ignore the version file and pull the version directly from Bazaar')
2012-12-05 05:53:14 +00:00
parser.add_option('-s', '--style', dest='style', help='Set the Qt4 style (passed directly to Qt4).')
# Parse command line options and deal with them.
2013-12-15 16:50:09 +00:00
# Use args supplied pragmatically if possible.
2011-08-26 11:04:56 +00:00
(options, args) = parser.parse_args(args) if args else parser.parse_args()
qt_args = []
if options.loglevel.lower() in ['d', 'debug']:
log.setLevel(logging.DEBUG)
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)
# Bug #1018855: Set the WM_CLASS property in X11
if platform.system() not in ['Windows', 'Darwin']:
2012-08-31 08:39:38 +00:00
qt_args.append('OpenLP')
# Initialise the resources
qInitResources()
# Now create and actually run the application.
2013-02-03 17:42:31 +00:00
application = OpenLP(qt_args)
2013-08-31 18:17:38 +00:00
application.setOrganizationName('OpenLP')
application.setOrganizationDomain('openlp.org')
if options.portable:
2013-08-31 18:17:38 +00:00
application.setApplicationName('OpenLPPortable')
2012-05-23 17:13:22 +00:00
Settings.setDefaultFormat(Settings.IniFormat)
# Get location OpenLPPortable.ini
2013-02-03 17:42:31 +00:00
application_path = AppLocation.get_directory(AppLocation.AppDir)
2013-08-31 18:17:38 +00:00
set_up_logging(os.path.abspath(os.path.join(application_path, '..', '..', 'Other')))
log.info('Running portable')
portable_settings_file = os.path.abspath(os.path.join(application_path, '..', '..', 'Data', 'OpenLP.ini'))
# Make this our settings file
2013-08-31 18:17:38 +00:00
log.info('INI file: %s', portable_settings_file)
Settings.set_filename(portable_settings_file)
portable_settings = Settings()
# Set our data path
2013-08-31 18:17:38 +00:00
data_path = os.path.abspath(os.path.join(application_path, '..', '..', 'Data',))
log.info('Data path: %s', data_path)
# Point to our data path
2013-08-31 18:17:38 +00:00
portable_settings.setValue('advanced/data path', data_path)
portable_settings.setValue('advanced/is portable', True)
portable_settings.sync()
else:
2013-08-31 18:17:38 +00:00
application.setApplicationName('OpenLP')
set_up_logging(AppLocation.get_directory(AppLocation.CacheDir))
2013-02-01 19:58:18 +00:00
Registry.create()
2013-08-31 18:17:38 +00:00
Registry().register('application', application)
application.setApplicationVersion(get_application_version()['version'])
# Instance check
2013-02-03 17:42:31 +00:00
if application.is_already_running():
2013-02-03 15:06:17 +00:00
sys.exit()
2013-03-26 11:45:18 +00:00
# Remove/convert obsolete settings.
Settings().remove_obsolete_settings()
# First time checks in settings
2013-08-31 18:17:38 +00:00
if not Settings().value('core/has run wizard'):
if not FirstTimeLanguageForm().exec_():
# if cancel then stop processing
sys.exit()
# i18n Set Language
language = LanguageManager.get_language()
2013-02-03 17:42:31 +00:00
application_translator, default_translator = LanguageManager.get_translator(language)
if not application_translator.isEmpty():
application.installTranslator(application_translator)
if not default_translator.isEmpty():
2013-02-03 17:42:31 +00:00
application.installTranslator(default_translator)
else:
2013-08-31 18:17:38 +00:00
log.debug('Could not find default_translator.')
if not options.no_error_form:
2013-02-03 17:42:31 +00:00
sys.excepthook = application.hook_exception
sys.exit(application.run(qt_args))