Generate bindings code for EventTarget.addEventListener() / removeEventListener()
[WebKit-https.git] / Source / WebCore / bindings / gobject / GObjectEventListener.cpp
1 /*
2  *  Copyright (C) 2010, 2011 Igalia S.L.
3  *
4  *  This library is free software; you can redistribute it and/or
5  *  modify it under the terms of the GNU Lesser General Public
6  *  License as published by the Free Software Foundation; either
7  *  version 2 of the License, or (at your option) any later version.
8  *
9  *  This library is distributed in the hope that it will be useful,
10  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
11  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  *  Lesser General Public License for more details.
13  *
14  *  You should have received a copy of the GNU Lesser General Public
15  *  License along with this library; if not, write to the Free Software
16  *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
17  */
18
19 #include "config.h"
20 #include "GObjectEventListener.h"
21
22 #include "Event.h"
23 #include "WebKitDOMEvent.h"
24 #include "WebKitDOMEventPrivate.h"
25 #include "WebKitDOMEventTarget.h"
26 #include <wtf/HashMap.h>
27
28 namespace WebCore {
29
30 GObjectEventListener::GObjectEventListener(GObject* target, EventTarget* coreTarget, const char* domEventName, GClosure* handler, bool capture)
31     : EventListener(GObjectEventListenerType)
32     , m_target(target)
33     , m_coreTarget(coreTarget)
34     , m_domEventName(domEventName)
35     , m_handler(handler)
36     , m_capture(capture)
37 {
38     ASSERT(m_coreTarget);
39     if (G_CLOSURE_NEEDS_MARSHAL(m_handler.get()))
40         g_closure_set_marshal(m_handler.get(), g_cclosure_marshal_generic);
41     g_object_weak_ref(m_target, reinterpret_cast<GWeakNotify>(GObjectEventListener::gobjectDestroyedCallback), this);
42 }
43
44 GObjectEventListener::~GObjectEventListener()
45 {
46     if (!m_coreTarget)
47         return;
48     g_object_weak_unref(m_target, reinterpret_cast<GWeakNotify>(GObjectEventListener::gobjectDestroyedCallback), this);
49 }
50
51 void GObjectEventListener::gobjectDestroyed()
52 {
53     ASSERT(m_coreTarget);
54
55     // Protect 'this' class in case the 'm_coreTarget' holds the last reference,
56     // which may cause, inside removeEventListener(), free of this object
57     // and later use-after-free with the m_handler = 0; assignment.
58     RefPtr<GObjectEventListener> protectedThis(this);
59
60     m_coreTarget->removeEventListener(m_domEventName.data(), *this, m_capture);
61     m_coreTarget = nullptr;
62     m_handler = nullptr;
63 }
64
65 void GObjectEventListener::handleEvent(ScriptExecutionContext*, Event* event)
66 {
67     GValue parameters[2] = { G_VALUE_INIT, G_VALUE_INIT };
68     g_value_init(&parameters[0], WEBKIT_DOM_TYPE_EVENT_TARGET);
69     g_value_set_object(&parameters[0], m_target);
70
71     GRefPtr<WebKitDOMEvent> domEvent = adoptGRef(WebKit::kit(event));
72     g_value_init(&parameters[1], WEBKIT_DOM_TYPE_EVENT);
73     g_value_set_object(&parameters[1], domEvent.get());
74
75     g_closure_invoke(m_handler.get(), 0, 2, parameters, NULL);
76     g_value_unset(parameters + 0);
77     g_value_unset(parameters + 1);
78 }
79
80 bool GObjectEventListener::operator==(const EventListener& listener) const
81 {
82     if (const GObjectEventListener* gobjectEventListener = GObjectEventListener::cast(&listener))
83         return m_target == gobjectEventListener->m_target
84             && reinterpret_cast<GCClosure*>(m_handler.get())->callback == reinterpret_cast<GCClosure*>(gobjectEventListener->m_handler.get())->callback;
85
86     return false;
87 }
88
89 }