summaryrefslogtreecommitdiffstats
path: root/app/gtk_signal.cc
diff options
context:
space:
mode:
authorestade@chromium.org <estade@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-04-16 21:10:32 +0000
committerestade@chromium.org <estade@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-04-16 21:10:32 +0000
commit05bcb65e5b8d1ddc63f19ecc6802c312b7c54243 (patch)
treead5e5d269de770fd3d2e2f083b6f080f66eae778 /app/gtk_signal.cc
parentb69e649feb5dc5f7daa6daa84c98b3a2d9a28124 (diff)
downloadchromium_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.cc72
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);
+}