Skip to content
Snippets Groups Projects
Commit 9b53c97b authored by Paul McCarthy's avatar Paul McCarthy
Browse files

DisplayContext now has a destroy method, which is called by the

FSLViewFrame when the associated ViewPanel is closed. Lots of fixes
still to be made.
parent 841c293c
No related branches found
No related tags found
No related merge requests found
...@@ -202,15 +202,17 @@ class ActionProvider(props.HasProperties): ...@@ -202,15 +202,17 @@ class ActionProvider(props.HasProperties):
act = Action(overlayList, displayCtx, action=func) act = Action(overlayList, displayCtx, action=func)
self.__actions[name] = act self.__actions[name] = act
log.memory('{}.init ({})'.format(type(self).__name__, id(self)))
def destroy(self):
"""This method should be called when this ``ActionProvider`` is
def __del__(self): about to be destroyed. It ensures that all ``Action`` instances
""" are cleared.
""" """
log.memory('{}.del ({})'.format(type(self).__name__, id(self))) for _, act in self.__actions.items():
act.unbindAllWidgets()
self.__actions = None self.__actions = None
def addActionToggleListener(self, name, listenerName, func): def addActionToggleListener(self, name, listenerName, func):
"""Add a listener function which will be called when the named action """Add a listener function which will be called when the named action
......
...@@ -93,7 +93,14 @@ class DisplayContext(props.SyncableHasProperties): ...@@ -93,7 +93,14 @@ class DisplayContext(props.SyncableHasProperties):
def __del__(self): def __del__(self):
log.memory('{}.del ({})'.format(type(self).__name__, id(self))) log.memory('{}.del ({})'.format(type(self).__name__, id(self)))
def destroy(self):
for overlay, display in self.__displays.items():
display.destroy()
self.__displays = None
def getDisplay(self, overlay, overlayType=None): def getDisplay(self, overlay, overlayType=None):
......
...@@ -98,6 +98,7 @@ class FSLViewFrame(wx.Frame): ...@@ -98,6 +98,7 @@ class FSLViewFrame(wx.Frame):
# Keeping track of all # Keeping track of all
# open view panels # open view panels
self.__viewPanels = [] self.__viewPanels = []
self.__viewPanelDCs = {}
self.__viewPanelTitles = {} self.__viewPanelTitles = {}
self.__viewPanelMenus = {} self.__viewPanelMenus = {}
self.__viewPanelCount = 0 self.__viewPanelCount = 0
...@@ -141,6 +142,7 @@ class FSLViewFrame(wx.Frame): ...@@ -141,6 +142,7 @@ class FSLViewFrame(wx.Frame):
self.__viewPanels.append(panel) self.__viewPanels.append(panel)
self.__viewPanelTitles[panel] = title self.__viewPanelTitles[panel] = title
self.__viewPanelDCs[ panel] = childDC
self.__centrePane.AddPage(panel, title, True) self.__centrePane.AddPage(panel, title, True)
self.__centrePane.Split( self.__centrePane.Split(
...@@ -184,6 +186,7 @@ class FSLViewFrame(wx.Frame): ...@@ -184,6 +186,7 @@ class FSLViewFrame(wx.Frame):
self.__viewPanels .remove(panel) self.__viewPanels .remove(panel)
self.__viewPanelMenus .pop( panel, None) self.__viewPanelMenus .pop( panel, None)
title = self.__viewPanelTitles.pop( panel) title = self.__viewPanelTitles.pop( panel)
dctx = self.__viewPanelDCs .pop( panel)
log.debug('Destroying view panel {} ({})'.format( log.debug('Destroying view panel {} ({})'.format(
title, type(panel).__name__)) title, type(panel).__name__))
...@@ -200,10 +203,9 @@ class FSLViewFrame(wx.Frame): ...@@ -200,10 +203,9 @@ class FSLViewFrame(wx.Frame):
# Calling fslpanel.FSLViewPanel.destroy() # Calling fslpanel.FSLViewPanel.destroy()
# - I think that the AUINotebook does the # - I think that the AUINotebook does the
# wx.Window.Destroy side of things, which # wx.Window.Destroy side of things ...
# will eventually result in panel.__del__
# ...
panel.destroy() panel.destroy()
dctx .destroy()
def __onClose(self, ev): def __onClose(self, ev):
......
...@@ -113,20 +113,18 @@ class _FSLViewPanel(actions.ActionProvider): ...@@ -113,20 +113,18 @@ class _FSLViewPanel(actions.ActionProvider):
Overriding subclass implementations must call this base class Overriding subclass implementations must call this base class
method, otherwise memory leaks will probably occur, and warnings will method, otherwise memory leaks will probably occur, and warnings will
probably be output to the log (see :meth:`__del__`). probably be output to the log (see :meth:`__del__`). This
implememtation should be called after the subclass has performed its
own clean-up, as this method expliciltly clears the ``_overlayList``
and ``_displayCtx`` references.
""" """
self.__destroyed = True actions.ActionProvider.destroy(self)
self._displayCtx = None
self._overlayList = None
self.__destroyed = True
def __del__(self): def __del__(self):
"""Sub-classes which implement ``__del__`` must call this
implementation, otherwise memory leaks will occur.
"""
actions.ActionProvider.__del__(self)
self._overlayList = None
self._displayCtx = None
if not self.__destroyed: if not self.__destroyed:
log.warning('The {}.destroy() method has not been called ' log.warning('The {}.destroy() method has not been called '
'- unless the application is shutting down, ' '- unless the application is shutting down, '
......
...@@ -193,8 +193,6 @@ class Profile(actions.ActionProvider): ...@@ -193,8 +193,6 @@ class Profile(actions.ActionProvider):
is called by the :class:`ProfileManager` when this ``Profile`` is called by the :class:`ProfileManager` when this ``Profile``
instance is no longer needed. instance is no longer needed.
""" """
self.deregister()
self._viewPanel = None self._viewPanel = None
self._overlayList = None self._overlayList = None
self._displayCtx = None self._displayCtx = None
...@@ -591,6 +589,7 @@ class ProfileManager(object): ...@@ -591,6 +589,7 @@ class ProfileManager(object):
important object references to avoid memory leaks. important object references to avoid memory leaks.
""" """
if self._currentProfile is not None: if self._currentProfile is not None:
self._currentProfile.deregister()
self._currentProfile.destroy() self._currentProfile.destroy()
self._currentProfile = None self._currentProfile = None
...@@ -622,6 +621,7 @@ class ProfileManager(object): ...@@ -622,6 +621,7 @@ class ProfileManager(object):
log.debug('Deregistering {} profile from {}'.format( log.debug('Deregistering {} profile from {}'.format(
self._currentProfile.__class__.__name__, self._currentProfile.__class__.__name__,
self._viewCls.__name__)) self._viewCls.__name__))
self._currentProfile.deregister()
self._currentProfile.destroy() self._currentProfile.destroy()
self._currentProfile = profileCls(self._viewPanel, self._currentProfile = profileCls(self._viewPanel,
......
...@@ -21,8 +21,9 @@ class LightBoxViewProfile(profiles.Profile): ...@@ -21,8 +21,9 @@ class LightBoxViewProfile(profiles.Profile):
displayCtx, displayCtx,
modes=['view', 'zoom']) modes=['view', 'zoom'])
self._canvas = canvasPanel.getCanvas() self._canvas = viewPanel.getCanvas()
def getEventTargets(self): def getEventTargets(self):
return [self._canvas] return [self._canvas]
......
...@@ -245,11 +245,11 @@ class CanvasPanel(viewpanel.ViewPanel): ...@@ -245,11 +245,11 @@ class CanvasPanel(viewpanel.ViewPanel):
cleanly. cleanly.
""" """
viewpanel.ViewPanel.destroy(self)
if self.__colourBar is not None: if self.__colourBar is not None:
self.__colourBar.destroy() self.__colourBar.destroy()
viewpanel.ViewPanel.destroy(self)
def screenshot(self, *a): def screenshot(self, *a):
_takeScreenShot(self._overlayList, self._displayCtx, self) _takeScreenShot(self._overlayList, self._displayCtx, self)
......
...@@ -133,12 +133,13 @@ class LightBoxPanel(canvaspanel.CanvasPanel): ...@@ -133,12 +133,13 @@ class LightBoxPanel(canvaspanel.CanvasPanel):
def destroy(self): def destroy(self):
"""Removes property listeners""" """Removes property listeners"""
canvaspanel.CanvasPanel.destroy(self)
self._displayCtx .removeListener('location', self._name) self._displayCtx .removeListener('location', self._name)
self._displayCtx .removeListener('selectedOverlay', self._name) self._displayCtx .removeListener('selectedOverlay', self._name)
self._overlayList.removeListener('overlays', self._name) self._overlayList.removeListener('overlays', self._name)
canvaspanel.CanvasPanel.destroy(self)
def _selectedOverlayChanged(self, *a): def _selectedOverlayChanged(self, *a):
"""Called when the selected overlay changes. """Called when the selected overlay changes.
......
...@@ -138,8 +138,6 @@ class ViewPanel(fslpanel.FSLViewPanel): ...@@ -138,8 +138,6 @@ class ViewPanel(fslpanel.FSLViewPanel):
def destroy(self): def destroy(self):
""" """
""" """
fslpanel.FSLViewPanel.destroy(self)
# Make sure that any control panels are correctly destroyed # Make sure that any control panels are correctly destroyed
for panelType, panel in self.__panels.items(): for panelType, panel in self.__panels.items():
...@@ -159,9 +157,16 @@ class ViewPanel(fslpanel.FSLViewPanel): ...@@ -159,9 +157,16 @@ class ViewPanel(fslpanel.FSLViewPanel):
# Un-initialise the AUI manager # Un-initialise the AUI manager
self.__auiMgr.UnInit() self.__auiMgr.UnInit()
# The AUI manager does not clear its
# reference to this panel, so let's
# do it here.
self.__auiMgr._frame = None
self.__profileManager = None self.__profileManager = None
self.__auiMgr = None self.__auiMgr = None
fslpanel.FSLViewPanel.destroy(self)
def setCentrePanel(self, panel): def setCentrePanel(self, panel):
panel.Reparent(self) panel.Reparent(self)
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment