From 470ce6cff70e9b4248e9b105c9c060eeb6a37ae2 Mon Sep 17 00:00:00 2001 From: Mads Kruse Johnsen Date: Thu, 19 Nov 2020 16:22:47 +0100 Subject: [PATCH] Invoke Destroy in Dispose if the Widget IsToplevel and it is not already destroyed If the Widget is a toplevel then we have not ref'ed the Object, so ref before destroying it, and let the freeing of the ToggleRef undo the ref. --- Source/Libs/GtkSharp/Widget.cs | 31 ++++++++++++++++++++++++++++++- 1 file changed, 30 insertions(+), 1 deletion(-) diff --git a/Source/Libs/GtkSharp/Widget.cs b/Source/Libs/GtkSharp/Widget.cs index 845495fba..78c74abd0 100644 --- a/Source/Libs/GtkSharp/Widget.cs +++ b/Source/Libs/GtkSharp/Widget.cs @@ -370,6 +370,7 @@ namespace Gtk { Gtk.Widget widget = o as Gtk.Widget; if (widget == null) return; + widget.OnDestroyed (); } @@ -387,20 +388,41 @@ namespace Gtk { base.CreateNativeObject (names, vals); } + [UnmanagedFunctionPointer(CallingConvention.Cdecl)] + delegate IntPtr d_g_object_ref(IntPtr raw); + static d_g_object_ref g_object_ref = FuncLoader.LoadFunction(FuncLoader.GetProcAddress(GLibrary.Load(Library.GObject), "g_object_ref")); + + private bool destroyed; protected override void Dispose (bool disposing) { if (Handle == IntPtr.Zero) return; + + if (disposing && !destroyed && IsToplevel) + { + //If this is a TopLevel widget, then we do not hold a ref, only a toggle ref. + //Freeing our toggle ref expects a normal ref to exist, and therefore does not check if the object still exists. + //Take a ref here and let our toggle ref unref it. + g_object_ref (Handle); + gtk_widget_destroy (Handle); + destroyed = true; + } + InternalDestroyed -= NativeDestroyHandler; + base.Dispose (disposing); } - protected override IntPtr Raw { + protected override IntPtr Raw { get { return base.Raw; } set { + if (Handle == value) + return; + base.Raw = value; + if (value != IntPtr.Zero) InternalDestroyed += NativeDestroyHandler; } @@ -409,11 +431,18 @@ namespace Gtk { delegate void d_gtk_widget_destroy(IntPtr raw); static d_gtk_widget_destroy gtk_widget_destroy = FuncLoader.LoadFunction(FuncLoader.GetProcAddress(GLibrary.Load(Library.Gtk), "gtk_widget_destroy")); + public virtual void Destroy () { if (Handle == IntPtr.Zero) return; + + if (destroyed) + return; + gtk_widget_destroy (Handle); + destroyed = true; + InternalDestroyed -= NativeDestroyHandler; } }