diff options
author | estade@chromium.org <estade@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-04-16 21:10:32 +0000 |
---|---|---|
committer | estade@chromium.org <estade@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-04-16 21:10:32 +0000 |
commit | 05bcb65e5b8d1ddc63f19ecc6802c312b7c54243 (patch) | |
tree | ad5e5d269de770fd3d2e2f083b6f080f66eae778 /app/gtk_signal.cc | |
parent | b69e649feb5dc5f7daa6daa84c98b3a2d9a28124 (diff) | |
download | chromium_src-05bcb65e5b8d1ddc63f19ecc6802c312b7c54243.zip chromium_src-05bcb65e5b8d1ddc63f19ecc6802c312b7c54243.tar.gz chromium_src-05bcb65e5b8d1ddc63f19ecc6802c312b7c54243.tar.bz2 |
GTK: Add a scoping class for g signals.
BUG=40735
TEST=compile; trybots; manually opening and closing a lot of infobubbles
Review URL: http://codereview.chromium.org/1652005
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@44821 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'app/gtk_signal.cc')
-rw-r--r-- | app/gtk_signal.cc | 72 |
1 files changed, 72 insertions, 0 deletions
diff --git a/app/gtk_signal.cc b/app/gtk_signal.cc new file mode 100644 index 0000000..4db611d --- /dev/null +++ b/app/gtk_signal.cc @@ -0,0 +1,72 @@ +// Copyright (c) 2010 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "app/gtk_signal.h" + +#include "base/logging.h" + +GtkSignalRegistrar::GtkSignalRegistrar() { +} + +GtkSignalRegistrar::~GtkSignalRegistrar() { + for (HandlerMap::iterator list_iter = handler_lists_.begin(); + list_iter != handler_lists_.end(); ++list_iter) { + GObject* object = list_iter->first; + g_object_weak_unref(object, WeakNotifyThunk, this); + + HandlerList& handlers = list_iter->second; + for (HandlerList::iterator ids_iter = handlers.begin(); + ids_iter != handlers.end(); ids_iter++) { + g_signal_handler_disconnect(list_iter->first, *ids_iter); + } + } +} + +glong GtkSignalRegistrar::Connect(gpointer instance, + const gchar* detailed_signal, + GCallback signal_handler, + gpointer data) { + return ConnectInternal(instance, detailed_signal, signal_handler, data, + false); +} + +glong GtkSignalRegistrar::ConnectAfter(gpointer instance, + const gchar* detailed_signal, + GCallback signal_handler, + gpointer data) { + return ConnectInternal(instance, detailed_signal, signal_handler, data, true); +} + +glong GtkSignalRegistrar::ConnectInternal(gpointer instance, + const gchar* detailed_signal, + GCallback signal_handler, + gpointer data, + bool after) { + GObject* object = G_OBJECT(instance); + + HandlerMap::iterator iter = handler_lists_.find(object); + if (iter == handler_lists_.end()) { + g_object_weak_ref(object, WeakNotifyThunk, this); + handler_lists_[object] = HandlerList(); + iter = handler_lists_.find(object); + } + + glong handler_id = after ? + g_signal_connect_after(instance, detailed_signal, signal_handler, data) : + g_signal_connect(instance, detailed_signal, signal_handler, data); + iter->second.push_back(handler_id); + + return handler_id; +} + +void GtkSignalRegistrar::WeakNotify(GObject* where_the_object_was) { + HandlerMap::iterator iter = handler_lists_.find(where_the_object_was); + if (iter == handler_lists_.end()) { + NOTREACHED(); + return; + } + // The signal handlers will be disconnected automatically. Just erase the + // handler id list. + handler_lists_.erase(iter); +} |