Fixes ManagedWindow for a modal window with another modal elsewhere
During Gramps startup, it is possible to start two modal dialogs; the 'Family Trees' and the 'Update Addons' (when preferences is set to check on startup). Both are modal. This upgrades the Window manager and ManagedWindow to support this case. If this occurs, then the most recent modal window is left that way, a previous modal window is temporarily downgraded to non-modal, and the newest window is set 'transient' for the previous modal window, even if it was not the parent. This means that there can be only one modal window, and it is always on top and active.
This commit is contained in:
parent
56fa449de8
commit
659329c879
@ -154,6 +154,16 @@ class GrampsWindowManager:
|
|||||||
return self.id2item[key]
|
return self.id2item[key]
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
def find_modal_window(self, window):
|
||||||
|
""" This finds a ManagedWindow that is modal, if any, excluding the
|
||||||
|
'window' that is a parameter. There should be only one.
|
||||||
|
If no ManagedWindow is modal, returns None.
|
||||||
|
"""
|
||||||
|
for dummy, item in self.id2item.items():
|
||||||
|
if item.window != window and item.window.get_modal():
|
||||||
|
return item.window
|
||||||
|
return None
|
||||||
|
|
||||||
def close_track(self, track):
|
def close_track(self, track):
|
||||||
# This is called when item needs to be closed
|
# This is called when item needs to be closed
|
||||||
# Closes all its children and then removes the item from the tree.
|
# Closes all its children and then removes the item from the tree.
|
||||||
@ -256,7 +266,7 @@ class GrampsWindowManager:
|
|||||||
if not isinstance(item, list):
|
if not isinstance(item, list):
|
||||||
def func(obj):
|
def func(obj):
|
||||||
if item.window_id and self.id2item.get(item.window_id):
|
if item.window_id and self.id2item.get(item.window_id):
|
||||||
self.id2item[item.window_id].present()
|
self.id2item[item.window_id].__present()
|
||||||
else:
|
else:
|
||||||
def func(obj):
|
def func(obj):
|
||||||
pass
|
pass
|
||||||
@ -346,7 +356,10 @@ class ManagedWindow:
|
|||||||
# Proceed with the class.
|
# Proceed with the class.
|
||||||
window = Gtk.Dialog() # Some Gtk window object to manage
|
window = Gtk.Dialog() # Some Gtk window object to manage
|
||||||
self.set_window(window, None, None) # See set_window def below
|
self.set_window(window, None, None) # See set_window def below
|
||||||
|
# setup window size, position tracking configuration
|
||||||
|
self.setup_configs(self, "interface.mywindow", 680, 400)
|
||||||
...
|
...
|
||||||
|
self.close()
|
||||||
|
|
||||||
def build_menu_names(self, obj):
|
def build_menu_names(self, obj):
|
||||||
''' Define menu labels. If your ManagedWindow can have
|
''' Define menu labels. If your ManagedWindow can have
|
||||||
@ -402,9 +415,10 @@ class ManagedWindow:
|
|||||||
self.vert_position_key = None
|
self.vert_position_key = None
|
||||||
self.__refs_for_deletion = []
|
self.__refs_for_deletion = []
|
||||||
self.modal = modal
|
self.modal = modal
|
||||||
|
self.other_modal_window = None
|
||||||
|
|
||||||
if uistate and uistate.gwm.get_item_from_id(window_key):
|
if uistate and uistate.gwm.get_item_from_id(window_key):
|
||||||
uistate.gwm.get_item_from_id(window_key).present()
|
uistate.gwm.get_item_from_id(window_key).__present()
|
||||||
raise WindowActiveError('This window is already active')
|
raise WindowActiveError('This window is already active')
|
||||||
else:
|
else:
|
||||||
self.window_id = window_key
|
self.window_id = window_key
|
||||||
@ -433,10 +447,8 @@ class ManagedWindow:
|
|||||||
managed_parent = self.uistate.gwm.get_item_from_track(
|
managed_parent = self.uistate.gwm.get_item_from_track(
|
||||||
parent_item_track)
|
parent_item_track)
|
||||||
self.parent_window = managed_parent.window
|
self.parent_window = managed_parent.window
|
||||||
self.parent_modal = managed_parent.modal
|
|
||||||
else:
|
else:
|
||||||
# On the top level: we use gramps top window
|
# On the top level: we use gramps top window
|
||||||
self.parent_modal = False
|
|
||||||
if self.uistate:
|
if self.uistate:
|
||||||
self.parent_window = self.uistate.window
|
self.parent_window = self.uistate.window
|
||||||
else:
|
else:
|
||||||
@ -474,11 +486,15 @@ class ManagedWindow:
|
|||||||
self.window.connect('delete-event', self.close)
|
self.window.connect('delete-event', self.close)
|
||||||
if self.modal:
|
if self.modal:
|
||||||
self.window.set_modal(True)
|
self.window.set_modal(True)
|
||||||
if self.parent_modal:
|
# The following makes sure that we only have one modal window open;
|
||||||
self.parent_window.set_modal(False)
|
# if more the older ones get temporarily made non-modal.
|
||||||
|
self.other_modal_window = self.uistate.gwm.find_modal_window(window)
|
||||||
|
if self.other_modal_window:
|
||||||
|
self.other_modal_window.set_modal(False)
|
||||||
self.window.set_modal(True)
|
self.window.set_modal(True)
|
||||||
self.modal = True
|
self.modal = True
|
||||||
|
|
||||||
|
|
||||||
def get_window(self):
|
def get_window(self):
|
||||||
"""
|
"""
|
||||||
Return the managed window.
|
Return the managed window.
|
||||||
@ -522,13 +538,16 @@ class ManagedWindow:
|
|||||||
self.get_widget(button_name).connect('clicked', function)
|
self.get_widget(button_name).connect('clicked', function)
|
||||||
|
|
||||||
def show(self):
|
def show(self):
|
||||||
if self.isWindow :
|
""" The following covers a case where there are multiple modal windows
|
||||||
self.set_transient_for(self.parent_window)
|
to be open; possibly not in parent child relation. If this occurs,
|
||||||
self.opened = True
|
we use most recent modal window as parent. This occurs during startup
|
||||||
self.show_all()
|
when both the 'Available Gramps Updates for Addons' and 'Family Trees'
|
||||||
|
windows are started by the viewmanager.
|
||||||
else :
|
"""
|
||||||
assert self.window, "ManagedWindow: self.window does not exist!"
|
assert self.window, "ManagedWindow: self.window does not exist!"
|
||||||
|
if self.other_modal_window:
|
||||||
|
self.window.set_transient_for(self.other_modal_window)
|
||||||
|
else:
|
||||||
self.window.set_transient_for(self.parent_window)
|
self.window.set_transient_for(self.parent_window)
|
||||||
self.opened = True
|
self.opened = True
|
||||||
self.window.show_all()
|
self.window.show_all()
|
||||||
@ -544,17 +563,15 @@ class ManagedWindow:
|
|||||||
self.clean_up()
|
self.clean_up()
|
||||||
self.uistate.gwm.close_track(self.track)
|
self.uistate.gwm.close_track(self.track)
|
||||||
self.opened = False
|
self.opened = False
|
||||||
if self.parent_modal:
|
# put a previously modal window back to modal, now that we are closing
|
||||||
self.parent_window.set_modal(True)
|
if self.other_modal_window:
|
||||||
|
self.other_modal_window.set_modal(True)
|
||||||
self.parent_window.present()
|
self.parent_window.present()
|
||||||
|
|
||||||
def present(self):
|
def __present(self):
|
||||||
"""
|
"""
|
||||||
Present window (unroll/unminimize/bring to top).
|
Present window (unroll/unminimize/bring to top).
|
||||||
"""
|
"""
|
||||||
if self.isWindow :
|
|
||||||
self.present(self)
|
|
||||||
else :
|
|
||||||
assert hasattr(self, 'window'), \
|
assert hasattr(self, 'window'), \
|
||||||
"ManagedWindow: self.window does not exist!"
|
"ManagedWindow: self.window does not exist!"
|
||||||
self.window.present()
|
self.window.present()
|
||||||
|
Loading…
Reference in New Issue
Block a user