713 lines
26 KiB
Python
713 lines
26 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2000-2006 Donald N. Allingham
|
|
# Copyright (C) 2009 Benny Malengier
|
|
# Copyright (C) 2018 Paul Culley
|
|
#
|
|
# 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; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
#
|
|
|
|
""" This contains the main class corresponding to a running gramps process """
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Python modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import sys
|
|
import os
|
|
import logging
|
|
LOG = logging.getLogger(".grampsgui")
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Gramps Modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
from gramps.gen.config import config
|
|
from gramps.gen.const import DATA_DIR, IMAGE_DIR, GTK_GETTEXT_DOMAIN
|
|
from gramps.gen.constfunc import has_display, lin
|
|
from gramps.gen.const import GRAMPS_LOCALE as glocale
|
|
_ = glocale.translation.gettext
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Miscellaneous initialization
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
|
|
MIN_PYGOBJECT_VERSION = (3, 12, 0)
|
|
PYGOBJ_ERR = False
|
|
MIN_GTK_VERSION = (3, 24)
|
|
UIDEFAULT = (
|
|
'''<?xml version="1.0" encoding="UTF-8"?>
|
|
<interface>
|
|
<menu id="menubar"><section id="menubar-update">
|
|
<submenu id='m1'>
|
|
<attribute name="label" translatable="yes">_Family Trees</attribute>
|
|
<section id="ftree">
|
|
<item>
|
|
<attribute name="action">win.Open</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''_Manage Family Trees...</attribute>
|
|
</item>
|
|
<submenu>
|
|
<attribute name="action">win.OpenRecent</attribute>
|
|
<attribute name="label" translatable="yes">Open _Recent</attribute>
|
|
<placeholder id="OpenRecentMenu">
|
|
</placeholder>
|
|
</submenu>
|
|
<item groups='RO'>
|
|
<attribute name="action">win.Close</attribute>
|
|
<attribute name="label" translatable="yes">_Close</attribute>
|
|
</item>
|
|
</section>
|
|
<section groups='RO RW'>
|
|
<item groups='RW'>
|
|
<attribute name="action">win.Import</attribute>
|
|
<attribute name="label" translatable="yes">_Import...</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.Export</attribute>
|
|
<attribute name="label" translatable="yes">_Export...</attribute>
|
|
</item>
|
|
<placeholder id="LocalExport">
|
|
</placeholder>
|
|
<item>
|
|
<attribute name="action">win.Backup</attribute>
|
|
<attribute name="label" translatable="yes">Make Backup...</attribute>
|
|
</item>
|
|
</section>
|
|
<section>
|
|
<item groups='RO'>
|
|
<attribute name="action">win.Abandon</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''_Abandon Changes and Quit</attribute>
|
|
</item>
|
|
<placeholder groups='OSX' id='osxquit'>
|
|
<item>
|
|
<attribute name="action">app.quit</attribute>
|
|
<attribute name="label" translatable="yes">_Quit</attribute>
|
|
</item>
|
|
</placeholder>
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m2' groups='RW'>
|
|
<attribute name="label" translatable="yes">_Add</attribute>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.PersonAdd</attribute>
|
|
<attribute name="label" translatable="yes">Person</attribute>
|
|
</item>
|
|
</section>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.FamilyAdd</attribute>
|
|
<attribute name="label" translatable="yes">Family</attribute>
|
|
</item>
|
|
</section>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.EventAdd</attribute>
|
|
<attribute name="label" translatable="yes">Event</attribute>
|
|
</item>
|
|
</section>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.PlaceAdd</attribute>
|
|
<attribute name="label" translatable="yes">Place</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.SourceAdd</attribute>
|
|
<attribute name="label" translatable="yes">Source</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.CitationAdd</attribute>
|
|
<attribute name="label" translatable="yes">Citation</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.RepositoryAdd</attribute>
|
|
<attribute name="label" translatable="yes">Repository</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.MediaAdd</attribute>
|
|
<attribute name="label" translatable="yes">Media</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.NoteAdd</attribute>
|
|
<attribute name="label" translatable="yes">Note</attribute>
|
|
</item>
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m3'>
|
|
<attribute name="label" translatable="yes">_Edit</attribute>
|
|
<section groups='RW'>
|
|
<placeholder id="undo">
|
|
</placeholder>
|
|
<placeholder id="redo">
|
|
</placeholder>
|
|
<item>
|
|
<attribute name="action">win.UndoHistory</attribute>
|
|
<attribute name="label" translatable="yes">Undo History</attribute>
|
|
</item>
|
|
</section>
|
|
<section id='CommonEdit' groups='RW'>
|
|
</section>
|
|
<section id='TagMenu' groups='RW'>
|
|
</section>
|
|
<section groups='RW'>
|
|
<item>
|
|
<attribute name="action">win.Clipboard</attribute>
|
|
<attribute name="label" translatable="yes">Clip_board</attribute>
|
|
</item>
|
|
</section>
|
|
<section>
|
|
<placeholder groups='OSX' id='osxpref'>
|
|
<item>
|
|
<attribute name="action">app.preferences</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''_Preferences...</attribute>
|
|
</item>
|
|
</placeholder>
|
|
<placeholder id='otheredit'>
|
|
</placeholder>
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m4'>
|
|
<attribute name="label" translatable="yes">_View</attribute>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.ConfigView</attribute>
|
|
<attribute name="label" translatable="yes">_Configure...</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.Navigator</attribute>
|
|
<attribute name="label" translatable="yes">_Navigator</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.Toolbar</attribute>
|
|
<attribute name="label" translatable="yes">_Toolbar</attribute>
|
|
</item>
|
|
<placeholder id='Bars'>
|
|
</placeholder>
|
|
<item>
|
|
<attribute name="action">win.Fullscreen</attribute>
|
|
<attribute name="label" translatable="yes">F_ull Screen</attribute>
|
|
</item>
|
|
</section>
|
|
<section id="ViewsInCatagory">
|
|
</section>
|
|
</submenu>
|
|
<submenu id="m5" groups='RO'>
|
|
<attribute name="label" translatable="yes">_Go</attribute>
|
|
<placeholder id="CommonGo">
|
|
</placeholder>
|
|
<section id="CommonHistory">
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m6' groups='RW'>
|
|
<attribute name="label" translatable="yes">_Bookmarks</attribute>
|
|
<section id="AddEditBook">
|
|
</section>
|
|
<section id="GoToBook">
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m7' groups='RO'>
|
|
<attribute name="label" translatable="yes">_Reports</attribute>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.Books</attribute>
|
|
<attribute name="label" translatable="yes">Books...</attribute>
|
|
</item>
|
|
</section>
|
|
<section id="P_ReportsMenu">
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m8' groups='RW'>
|
|
<attribute name="label" translatable="yes">_Tools</attribute>
|
|
<section id="P_ToolsMenu">
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m9' groups='RO'>
|
|
<attribute name="label" translatable="yes">_Windows</attribute>
|
|
<section id="WinMenu">
|
|
</section>
|
|
</submenu>
|
|
<submenu id='m10'>
|
|
<attribute name="label" translatable="yes">_Help</attribute>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.UserManual</attribute>
|
|
<attribute name="label" translatable="yes">_User Manual</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.FAQ</attribute>
|
|
<attribute name="label" translatable="yes">_FAQ</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.KeyBindings</attribute>
|
|
<attribute name="label" translatable="yes">_Key Bindings</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.TipOfDay</attribute>
|
|
<attribute name="label" translatable="yes">Tip of the Day</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.PluginStatus</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''_Plugin Manager</attribute>
|
|
</item>
|
|
</section>
|
|
<section>
|
|
<item>
|
|
<attribute name="action">win.HomePage</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''Gramps _Home Page</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.MailingLists</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''Gramps _Mailing Lists</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.ReportBug</attribute>
|
|
<attribute name="label" translatable="yes">_Report a Bug</attribute>
|
|
</item>
|
|
<item>
|
|
<attribute name="action">win.ExtraPlugins</attribute>
|
|
<attribute name="label" translatable="yes">'''
|
|
'''_Extra Reports/Tools</attribute>
|
|
</item>
|
|
</section>
|
|
<section groups='OSX'>
|
|
<item>
|
|
<attribute name="action">app.about</attribute>
|
|
<attribute name="label" translatable="yes">_About</attribute>
|
|
</item>
|
|
</section>
|
|
</submenu>
|
|
</section></menu>
|
|
|
|
<object class="GtkMenu" id="OpenBtnMenu">
|
|
<property name="visible">True</property>
|
|
<property name="can_focus">False</property>
|
|
</object>
|
|
|
|
<object class="GtkToolbar" id="ToolBar">
|
|
<property name="hexpand">1</property>
|
|
<property name="toolbar-style">GTK_TOOLBAR_ICONS</property>
|
|
<style>
|
|
<class name="primary-toolbar"/>
|
|
</style>
|
|
<child>
|
|
<object class="GtkToolButton">
|
|
<property name="icon-name">gramps</property>
|
|
<property name="action-name">win.Open</property>
|
|
<property name="tooltip_text" translatable="yes">'''
|
|
'''Manage databases</property>
|
|
<property name="label" translatable="yes">_Family Trees</property>
|
|
<property name="use-underline">True</property>
|
|
</object>
|
|
<packing>
|
|
<property name="homogeneous">False</property>
|
|
</packing>
|
|
</child>
|
|
<child>
|
|
<object class="GtkToolItem">
|
|
<child>
|
|
<object class="GtkMenuButton">
|
|
<property name="tooltip_text" translatable="yes">'''
|
|
'''Connect to a recent database</property>
|
|
<property name="popup">OpenBtnMenu</property>
|
|
<property name="relief">GTK_RELIEF_NONE</property>
|
|
</object>
|
|
</child>
|
|
</object>
|
|
</child>
|
|
<placeholder id='CommonNavigation'>
|
|
</placeholder>
|
|
<placeholder id='BarCommonEdit'>
|
|
</placeholder>
|
|
<placeholder id='TagTool'>
|
|
</placeholder>
|
|
<child>
|
|
<object class="GtkSeparatorToolItem"/>
|
|
</child>
|
|
<child>
|
|
<object class="GtkToolButton" id="ConfigView">
|
|
<property name="icon-name">gramps-config</property>
|
|
<property name="action-name">win.ConfigView</property>
|
|
<property name="tooltip_text" translatable="yes">'''
|
|
'''Configure the active view</property>
|
|
<property name="label" translatable="yes">_Configure...</property>
|
|
<property name="use-underline">True</property>
|
|
</object>
|
|
<packing>
|
|
<property name="homogeneous">False</property>
|
|
</packing>
|
|
</child>
|
|
<placeholder id='ViewsInCategoryBar'>
|
|
</placeholder>
|
|
<child>
|
|
<object class="GtkSeparatorToolItem" id="sep"/>
|
|
</child>
|
|
<placeholder id="MoreButtons">
|
|
</placeholder>
|
|
<placeholder id='AfterTools'>
|
|
</placeholder>
|
|
</object>
|
|
|
|
<menu id="Popup">
|
|
</menu>
|
|
|
|
</interface>
|
|
''')
|
|
|
|
try:
|
|
#import gnome introspection, part of pygobject
|
|
import gi
|
|
GIVERSION = gi.require_version
|
|
except:
|
|
print(_("Your version of gi (gnome-introspection) seems to be too old. "
|
|
"You need a version which has the function 'require_version' "
|
|
"to start Gramps"))
|
|
sys.exit(1)
|
|
|
|
if not PYGOBJ_ERR:
|
|
try:
|
|
from gi.repository import GObject, GLib
|
|
if not GObject.pygobject_version >= MIN_PYGOBJECT_VERSION:
|
|
PYGOBJ_ERR = True
|
|
except:
|
|
PYGOBJ_ERR = True
|
|
|
|
if PYGOBJ_ERR:
|
|
print(_("Your pygobject version does not meet the requirements.\n"
|
|
"At least pygobject %(major)d.%(feature)d.%(minor)d "
|
|
"is needed to start Gramps with a GUI.\n\n"
|
|
"Gramps will terminate now."
|
|
) % {'major' : MIN_PYGOBJECT_VERSION[0],
|
|
'feature' : MIN_PYGOBJECT_VERSION[1],
|
|
'minor' : MIN_PYGOBJECT_VERSION[2]})
|
|
sys.exit(1)
|
|
|
|
try:
|
|
gi.require_version('Pango', '1.0')
|
|
gi.require_version('PangoCairo', '1.0')
|
|
gi.require_version('Gtk', '3.0')
|
|
#It is important to import Pango before Gtk, or some things start to go
|
|
#wrong in GTK3 !
|
|
from gi.repository import Pango, PangoCairo
|
|
from gi.repository import Gtk, Gdk
|
|
except (ImportError, ValueError):
|
|
print(_("Gdk, Gtk, Pango or PangoCairo typelib not installed.\n"
|
|
"Install Gnome Introspection, and "
|
|
"pygobject version 3.12 or later.\n"
|
|
"Then install introspection data for Gdk, Gtk, Pango and "
|
|
"PangoCairo\n\n"
|
|
"Gramps will terminate now."))
|
|
sys.exit(1)
|
|
|
|
GTK_MAJOR = Gtk.get_major_version()
|
|
GTK_MINOR = Gtk.get_minor_version()
|
|
if (GTK_MAJOR, GTK_MINOR) < MIN_GTK_VERSION:
|
|
print(_("Your Gtk version does not meet the requirements.\n"
|
|
"At least %(major)d.%(minor)d "
|
|
"is needed to start Gramps with a GUI.\n\n"
|
|
"Gramps will terminate now."
|
|
) % {'major' : MIN_GTK_VERSION[0],
|
|
'minor' : MIN_GTK_VERSION[1]})
|
|
sys.exit(1)
|
|
|
|
try:
|
|
import cairo
|
|
except ImportError:
|
|
print(_("\ncairo python support not installed. "
|
|
"Install cairo for your version of python\n\n"
|
|
"Gramps will terminate now."))
|
|
sys.exit(1)
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Functions
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
|
|
def _display_gtk_gettext_message(parent=None):
|
|
"""
|
|
Display a GTK-translations-missing message to the user.
|
|
|
|
Note: the warning dialog below will likely have wrong stock icons!
|
|
"""
|
|
LOG.warning("GTK translations missing, GUI will be broken, "
|
|
"especially for RTL languages!")
|
|
from .dialog import WarningDialog
|
|
WarningDialog(_("Gramps detected "
|
|
"an incomplete GTK installation"),
|
|
_("GTK translations for the current language (%(language)s) "
|
|
"are missing.\n"
|
|
"%(bold_start)sGramps%(bold_end)s will "
|
|
"proceed nevertheless.\n"
|
|
"The GUI will likely be broken "
|
|
"as a result, especially for RTL languages!\n\n"
|
|
"See the Gramps README documentation for installation "
|
|
"prerequisites,\n"
|
|
"typically located in "
|
|
"/usr/share/doc/gramps.\n"
|
|
) % {'language' : glocale.lang,
|
|
'bold_start' : '<b>',
|
|
'bold_end' : '</b>'},
|
|
parent=parent)
|
|
|
|
def _display_translator_message(parent=None):
|
|
"""
|
|
Display a translator-wanted message to the user.
|
|
"""
|
|
if config.get('behavior.translator-needed'):
|
|
config.set('behavior.translator-needed', False)
|
|
from gramps.gen.utils.grampslocale import INCOMPLETE_TRANSLATIONS
|
|
language = None
|
|
if glocale.lang in INCOMPLETE_TRANSLATIONS:
|
|
language = glocale.lang
|
|
elif glocale.lang[:2] in INCOMPLETE_TRANSLATIONS:
|
|
language = glocale.lang[:2]
|
|
if language:
|
|
from .dialog import WarningDialog
|
|
from gramps.gen.const import URL_MAILINGLIST
|
|
# we are looking for a translator so leave this in English
|
|
WarningDialog("This Gramps has an incomplete translation",
|
|
"The translation for the "
|
|
"current language (%(language)s) is incomplete.\n\n"
|
|
"%(bold_start)sGramps%(bold_end)s "
|
|
"will start anyway, but if you would like "
|
|
"to improve\nGramps by doing some translating, "
|
|
"please contact us!\n\n"
|
|
"Subscribe to gramps-devel at\n%(mailing_list_url)s"
|
|
"\n" % {'language' : language,
|
|
'bold_start' : '<b>',
|
|
'bold_end' : '</b>',
|
|
'mailing_list_url' : URL_MAILINGLIST},
|
|
parent=parent)
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Main Gramps class
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
class Gramps:
|
|
"""
|
|
Main class corresponding to a running gramps process.
|
|
|
|
There can be only one instance of this class per gramps application
|
|
process. It may spawn several windows and control several databases.
|
|
"""
|
|
|
|
def __init__(self, argparser, app):
|
|
from gramps.gen.dbstate import DbState
|
|
from .viewmanager import ViewManager
|
|
from gramps.cli.arghandler import ArgHandler
|
|
from .tipofday import TipOfDay
|
|
import gettext
|
|
|
|
# Append image directory to the theme search path
|
|
theme = Gtk.IconTheme.get_default()
|
|
theme.append_search_path(IMAGE_DIR)
|
|
|
|
dbstate = DbState()
|
|
self._vm = ViewManager(app, dbstate,
|
|
config.get("interface.view-categories"))
|
|
|
|
if (lin()
|
|
and glocale.lang != 'C'
|
|
and not gettext.find(GTK_GETTEXT_DOMAIN)):
|
|
_display_gtk_gettext_message(parent=self._vm.window)
|
|
|
|
_display_translator_message(parent=self._vm.window)
|
|
|
|
self._vm.init_interface()
|
|
|
|
#act based on the given arguments
|
|
arg_h = ArgHandler(dbstate, argparser, self._vm, self.argerrorfunc,
|
|
gui=True)
|
|
arg_h.handle_args_gui()
|
|
if arg_h.open or arg_h.imp_db_path:
|
|
# if we opened or imported something, only show the interface
|
|
self._vm.post_init_interface(show_manager=False)
|
|
elif (config.get('paths.recent-file')
|
|
and config.get('behavior.autoload')):
|
|
# if we need to autoload last seen file, do so
|
|
filename = config.get('paths.recent-file')
|
|
if (os.path.isdir(filename)
|
|
and os.path.isfile(os.path.join(filename, "name.txt"))
|
|
and arg_h.check_db(filename)):
|
|
self._vm.post_init_interface(show_manager=False)
|
|
self._vm.open_activate(filename)
|
|
else:
|
|
self._vm.post_init_interface()
|
|
else:
|
|
# open without fam tree loaded
|
|
self._vm.post_init_interface()
|
|
|
|
if config.get('behavior.use-tips'):
|
|
TipOfDay(self._vm.uistate)
|
|
|
|
def argerrorfunc(self, string):
|
|
""" Show basic errors in argument handling in GUI fashion"""
|
|
from .dialog import ErrorDialog
|
|
parent = None
|
|
if hasattr(self, '_vm'):
|
|
if hasattr(self._vm, 'window'):
|
|
parent = self._vm.window
|
|
ErrorDialog(_("Error parsing arguments"), string,
|
|
parent=parent)
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Main startup functions
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
|
|
def startgramps(errors, argparser):
|
|
"""
|
|
Main startup function started via GObject.timeout_add
|
|
First action inside the gtk loop
|
|
"""
|
|
app = GrampsApplication(errors, argparser)
|
|
try:
|
|
quit_now = False
|
|
exit_code = 0
|
|
if app.run():
|
|
print(_("Gramps terminated because of no DISPLAY"))
|
|
|
|
except SystemExit as err:
|
|
quit_now = True
|
|
if err.code:
|
|
exit_code = err.code
|
|
if isinstance(err.code, str):
|
|
LOG.error(err.code, exc_info=False)
|
|
else:
|
|
LOG.error("Gramps terminated with exit code: %d.", err.code,
|
|
exc_info=False)
|
|
except OSError as err:
|
|
quit_now = True
|
|
exit_code = err.errno or 1
|
|
try:
|
|
fname = err.filename
|
|
except AttributeError:
|
|
fname = ""
|
|
LOG.error("Gramps terminated because of OS Error\n" +
|
|
"Error details: %s %s", repr(err), fname, exc_info=True)
|
|
except:
|
|
quit_now = True
|
|
exit_code = 1
|
|
LOG.error(_("\nGramps failed to start. "
|
|
"Please report a bug about this.\n"
|
|
"This could be because of an error "
|
|
"in a (third party) View on startup.\n"
|
|
"To use another view, don't load a Family Tree, "
|
|
"change view, and then load your Family Tree.\n"
|
|
"You can also change manually "
|
|
"the startup view in the gramps.ini file \n"
|
|
"by changing the last-view parameter.\n"
|
|
), exc_info=True)
|
|
|
|
if quit_now:
|
|
app.quit()
|
|
sys.exit(exit_code)
|
|
|
|
return False
|
|
|
|
# we do the following import here to avoid the Gtk require version warning
|
|
from .uimanager import UIManager
|
|
from gramps.gen.constfunc import is_quartz
|
|
|
|
|
|
class GrampsApplication(Gtk.Application):
|
|
|
|
def __init__(self, errors, argparser):
|
|
super().__init__(application_id="org.gramps_project.Gramps")
|
|
self.window = None
|
|
self.errors = errors
|
|
self.argparser = argparser
|
|
|
|
def do_startup(self):
|
|
Gtk.Application.do_startup(self)
|
|
self.uimanager = UIManager(self, UIDEFAULT)
|
|
if not is_quartz():
|
|
self.uimanager.show_groups = ['OSX']
|
|
self.uimanager.update_menu(init=True)
|
|
|
|
if os.path.exists(os.path.join(DATA_DIR, "gramps.accel")):
|
|
self.uimanager.load_accels(os.path.join(DATA_DIR, "gramps.accel"))
|
|
try:
|
|
from .dialog import ErrorDialog
|
|
#handle first existing errors in GUI fashion
|
|
if self.errors:
|
|
for error in self.errors:
|
|
ErrorDialog(error[0], error[1]) # TODO no-parent
|
|
Gtk.main_quit()
|
|
sys.exit(1)
|
|
|
|
if self.argparser.errors:
|
|
for error in self.argparser.errors:
|
|
ErrorDialog(error[0], error[1]) # TODO no-parent
|
|
Gtk.main_quit()
|
|
sys.exit(1)
|
|
|
|
# add gui logger
|
|
from .logger import RotateHandler, GtkHandler
|
|
form = logging.Formatter(
|
|
fmt="%(relativeCreated)d: %(levelname)s: "
|
|
"%(filename)s: line %(lineno)d: %(message)s")
|
|
# Create the log handlers
|
|
rot_h = RotateHandler(capacity=20)
|
|
rot_h.setFormatter(form)
|
|
# Only error and critical log records should
|
|
# trigger the GUI handler.
|
|
gtkh = GtkHandler(rotate_handler=rot_h)
|
|
gtkh.setFormatter(form)
|
|
gtkh.setLevel(logging.ERROR)
|
|
logger = logging.getLogger()
|
|
logger.addHandler(rot_h)
|
|
logger.addHandler(gtkh)
|
|
|
|
except:
|
|
#make sure there is a clean exit if error in above steps
|
|
exit_code = 1
|
|
LOG.error(_("\nGramps failed to start. "
|
|
"Please report a bug about this.\n"
|
|
"This could be because of an error "
|
|
"in a (third party) View on startup.\n"
|
|
"To use another view, don't load a Family Tree, "
|
|
"change view, and then load your Family Tree.\n"
|
|
"You can also change manually "
|
|
"the startup view in the gramps.ini file \n"
|
|
"by changing the last-view parameter.\n"
|
|
), exc_info=True)
|
|
#stop gtk loop and quit
|
|
self.quit()
|
|
sys.exit(exit_code)
|
|
|
|
def do_activate(self):
|
|
# We only allow a single window and raise any existing ones
|
|
if not self.window:
|
|
# Windows are associated with the application
|
|
# when the last one is closed the application shuts down
|
|
Gramps(self.argparser, self)
|
|
else:
|
|
print('Gramps is already running.')
|
|
self.window.present()
|